Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(6168)

Unified Diff: chrome/browser/extensions/api/bookmarks/bookmark_api.cc

Issue 11778096: Revert 176047 (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src/
Patch Set: Created 7 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: chrome/browser/extensions/api/bookmarks/bookmark_api.cc
===================================================================
--- chrome/browser/extensions/api/bookmarks/bookmark_api.cc (revision 176049)
+++ chrome/browser/extensions/api/bookmarks/bookmark_api.cc (working copy)
@@ -1,996 +0,0 @@
-// Copyright (c) 2012 The Chromium Authors. All rights reserved.
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#include "chrome/browser/extensions/api/bookmarks/bookmark_api.h"
-
-#include "base/bind.h"
-#include "base/file_path.h"
-#include "base/i18n/file_util_icu.h"
-#include "base/i18n/time_formatting.h"
-#include "base/json/json_writer.h"
-#include "base/memory/scoped_ptr.h"
-#include "base/path_service.h"
-#include "base/prefs/public/pref_service_base.h"
-#include "base/sha1.h"
-#include "base/stl_util.h"
-#include "base/string16.h"
-#include "base/string_number_conversions.h"
-#include "base/string_util.h"
-#include "base/time.h"
-#include "base/utf_string_conversions.h"
-#include "chrome/browser/bookmarks/bookmark_codec.h"
-#include "chrome/browser/bookmarks/bookmark_html_writer.h"
-#include "chrome/browser/bookmarks/bookmark_model.h"
-#include "chrome/browser/bookmarks/bookmark_model_factory.h"
-#include "chrome/browser/bookmarks/bookmark_utils.h"
-#include "chrome/browser/extensions/api/bookmarks/bookmark_api_constants.h"
-#include "chrome/browser/extensions/api/bookmarks/bookmark_api_helpers.h"
-#include "chrome/browser/extensions/event_router.h"
-#include "chrome/browser/extensions/extension_function_dispatcher.h"
-#include "chrome/browser/extensions/extension_system.h"
-#include "chrome/browser/extensions/extensions_quota_service.h"
-#include "chrome/browser/importer/importer_data_types.h"
-#include "chrome/browser/importer/importer_host.h"
-#include "chrome/browser/profiles/profile.h"
-#include "chrome/browser/ui/chrome_select_file_policy.h"
-#include "chrome/common/chrome_notification_types.h"
-#include "chrome/common/chrome_paths.h"
-#include "chrome/common/extensions/api/bookmarks.h"
-#include "chrome/common/pref_names.h"
-#include "content/public/browser/notification_service.h"
-#include "grit/generated_resources.h"
-#include "ui/base/l10n/l10n_util.h"
-
-namespace extensions {
-
-namespace keys = bookmark_api_constants;
-namespace bookmarks = api::bookmarks;
-
-using base::TimeDelta;
-using bookmarks::BookmarkTreeNode;
-using content::BrowserThread;
-using content::WebContents;
-
-typedef QuotaLimitHeuristic::Bucket Bucket;
-typedef QuotaLimitHeuristic::Config Config;
-typedef QuotaLimitHeuristic::BucketList BucketList;
-typedef ExtensionsQuotaService::TimedLimit TimedLimit;
-typedef ExtensionsQuotaService::SustainedLimit SustainedLimit;
-typedef QuotaLimitHeuristic::BucketMapper BucketMapper;
-
-namespace {
-
-// Generates a default path (including a default filename) that will be
-// used for pre-populating the "Export Bookmarks" file chooser dialog box.
-FilePath GetDefaultFilepathForBookmarkExport() {
- base::Time time = base::Time::Now();
-
- // Concatenate a date stamp to the filename.
-#if defined(OS_POSIX)
- FilePath::StringType filename =
- l10n_util::GetStringFUTF8(IDS_EXPORT_BOOKMARKS_DEFAULT_FILENAME,
- base::TimeFormatShortDateNumeric(time));
-#elif defined(OS_WIN)
- FilePath::StringType filename =
- l10n_util::GetStringFUTF16(IDS_EXPORT_BOOKMARKS_DEFAULT_FILENAME,
- base::TimeFormatShortDateNumeric(time));
-#endif
-
- file_util::ReplaceIllegalCharactersInPath(&filename, '_');
-
- FilePath default_path;
- PathService::Get(chrome::DIR_USER_DOCUMENTS, &default_path);
- return default_path.Append(filename);
-}
-
-} // namespace
-
-void BookmarksFunction::Run() {
- BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
- if (!model->IsLoaded()) {
- // Bookmarks are not ready yet. We'll wait.
- registrar_.Add(
- this, chrome::NOTIFICATION_BOOKMARK_MODEL_LOADED,
- content::NotificationService::AllBrowserContextsAndSources());
- AddRef(); // Balanced in Observe().
- return;
- }
-
- bool success = RunImpl();
- if (success) {
- content::NotificationService::current()->Notify(
- chrome::NOTIFICATION_EXTENSION_BOOKMARKS_API_INVOKED,
- content::Source<const Extension>(GetExtension()),
- content::Details<const BookmarksFunction>(this));
- }
- SendResponse(success);
-}
-
-bool BookmarksFunction::GetBookmarkIdAsInt64(
- const std::string& id_string, int64* id) {
- if (base::StringToInt64(id_string, id))
- return true;
-
- error_ = keys::kInvalidIdError;
- return false;
-}
-
-bool BookmarksFunction::EditBookmarksEnabled() {
- PrefServiceBase* prefs = PrefServiceBase::FromBrowserContext(profile_);
- if (prefs->GetBoolean(prefs::kEditBookmarksEnabled))
- return true;
- error_ = keys::kEditBookmarksDisabled;
- return false;
-}
-
-void BookmarksFunction::Observe(int type,
- const content::NotificationSource& source,
- const content::NotificationDetails& details) {
- DCHECK(type == chrome::NOTIFICATION_BOOKMARK_MODEL_LOADED);
- Profile* source_profile = content::Source<Profile>(source).ptr();
- if (!source_profile || !source_profile->IsSameProfile(profile()))
- return;
-
- DCHECK(BookmarkModelFactory::GetForProfile(profile())->IsLoaded());
- Run();
- Release(); // Balanced in Run().
-}
-
-BookmarkEventRouter::BookmarkEventRouter(BookmarkModel* model) : model_(model) {
- model_->AddObserver(this);
-}
-
-BookmarkEventRouter::~BookmarkEventRouter() {
- if (model_) {
- model_->RemoveObserver(this);
- }
-}
-
-void BookmarkEventRouter::DispatchEvent(
- Profile* profile,
- const char* event_name,
- scoped_ptr<ListValue> event_args) {
- if (extensions::ExtensionSystem::Get(profile)->event_router()) {
- extensions::ExtensionSystem::Get(profile)->event_router()->BroadcastEvent(
- make_scoped_ptr(new extensions::Event(event_name, event_args.Pass())));
- }
-}
-
-void BookmarkEventRouter::Loaded(BookmarkModel* model, bool ids_reassigned) {
- // TODO(erikkay): Perhaps we should send this event down to the extension
- // so they know when it's safe to use the API?
-}
-
-void BookmarkEventRouter::BookmarkModelBeingDeleted(BookmarkModel* model) {
- model_ = NULL;
-}
-
-void BookmarkEventRouter::BookmarkNodeMoved(BookmarkModel* model,
- const BookmarkNode* old_parent,
- int old_index,
- const BookmarkNode* new_parent,
- int new_index) {
- scoped_ptr<ListValue> args(new ListValue());
- const BookmarkNode* node = new_parent->GetChild(new_index);
- args->Append(new StringValue(base::Int64ToString(node->id())));
- DictionaryValue* object_args = new DictionaryValue();
- object_args->SetString(keys::kParentIdKey,
- base::Int64ToString(new_parent->id()));
- object_args->SetInteger(keys::kIndexKey, new_index);
- object_args->SetString(keys::kOldParentIdKey,
- base::Int64ToString(old_parent->id()));
- object_args->SetInteger(keys::kOldIndexKey, old_index);
- args->Append(object_args);
-
- DispatchEvent(model->profile(), keys::kOnBookmarkMoved, args.Pass());
-}
-
-void BookmarkEventRouter::BookmarkNodeAdded(BookmarkModel* model,
- const BookmarkNode* parent,
- int index) {
- scoped_ptr<ListValue> args(new ListValue());
- const BookmarkNode* node = parent->GetChild(index);
- args->Append(new StringValue(base::Int64ToString(node->id())));
- scoped_ptr<BookmarkTreeNode> tree_node(
- bookmark_api_helpers::GetBookmarkTreeNode(node, false, false));
- args->Append(tree_node->ToValue().release());
-
- DispatchEvent(model->profile(), keys::kOnBookmarkCreated, args.Pass());
-}
-
-void BookmarkEventRouter::BookmarkNodeRemoved(BookmarkModel* model,
- const BookmarkNode* parent,
- int index,
- const BookmarkNode* node) {
- scoped_ptr<ListValue> args(new ListValue());
- args->Append(new StringValue(base::Int64ToString(node->id())));
- DictionaryValue* object_args = new DictionaryValue();
- object_args->SetString(keys::kParentIdKey,
- base::Int64ToString(parent->id()));
- object_args->SetInteger(keys::kIndexKey, index);
- args->Append(object_args);
-
- DispatchEvent(model->profile(), keys::kOnBookmarkRemoved, args.Pass());
-}
-
-void BookmarkEventRouter::BookmarkNodeChanged(BookmarkModel* model,
- const BookmarkNode* node) {
- scoped_ptr<ListValue> args(new ListValue());
- args->Append(new StringValue(base::Int64ToString(node->id())));
-
- // TODO(erikkay) The only three things that BookmarkModel sends this
- // notification for are title, url and favicon. Since we're currently
- // ignoring favicon and since the notification doesn't say which one anyway,
- // for now we only include title and url. The ideal thing would be to change
- // BookmarkModel to indicate what changed.
- DictionaryValue* object_args = new DictionaryValue();
- object_args->SetString(keys::kTitleKey, node->GetTitle());
- if (node->is_url())
- object_args->SetString(keys::kUrlKey, node->url().spec());
- args->Append(object_args);
-
- DispatchEvent(model->profile(), keys::kOnBookmarkChanged, args.Pass());
-}
-
-void BookmarkEventRouter::BookmarkNodeFaviconChanged(BookmarkModel* model,
- const BookmarkNode* node) {
- // TODO(erikkay) anything we should do here?
-}
-
-void BookmarkEventRouter::BookmarkNodeChildrenReordered(
- BookmarkModel* model,
- const BookmarkNode* node) {
- scoped_ptr<ListValue> args(new ListValue());
- args->Append(new StringValue(base::Int64ToString(node->id())));
- int childCount = node->child_count();
- ListValue* children = new ListValue();
- for (int i = 0; i < childCount; ++i) {
- const BookmarkNode* child = node->GetChild(i);
- Value* child_id = new StringValue(base::Int64ToString(child->id()));
- children->Append(child_id);
- }
- DictionaryValue* reorder_info = new DictionaryValue();
- reorder_info->Set(keys::kChildIdsKey, children);
- args->Append(reorder_info);
-
- DispatchEvent(model->profile(), keys::kOnBookmarkChildrenReordered,
- args.Pass());
-}
-
-void BookmarkEventRouter::ExtensiveBookmarkChangesBeginning(
- BookmarkModel* model) {
- scoped_ptr<ListValue> args(new ListValue());
- DispatchEvent(model->profile(),
- keys::kOnBookmarkImportBegan,
- args.Pass());
-}
-
-void BookmarkEventRouter::ExtensiveBookmarkChangesEnded(BookmarkModel* model) {
- scoped_ptr<ListValue> args(new ListValue());
- DispatchEvent(model->profile(),
- keys::kOnBookmarkImportEnded,
- args.Pass());
-}
-
-BookmarkAPI::BookmarkAPI(Profile* profile) : profile_(profile) {
- ExtensionSystem::Get(profile_)->event_router()->RegisterObserver(
- this, keys::kOnBookmarkCreated);
- ExtensionSystem::Get(profile_)->event_router()->RegisterObserver(
- this, keys::kOnBookmarkRemoved);
- ExtensionSystem::Get(profile_)->event_router()->RegisterObserver(
- this, keys::kOnBookmarkChanged);
- ExtensionSystem::Get(profile_)->event_router()->RegisterObserver(
- this, keys::kOnBookmarkMoved);
- ExtensionSystem::Get(profile_)->event_router()->RegisterObserver(
- this, keys::kOnBookmarkChildrenReordered);
- ExtensionSystem::Get(profile_)->event_router()->RegisterObserver(
- this, keys::kOnBookmarkImportBegan);
- ExtensionSystem::Get(profile_)->event_router()->RegisterObserver(
- this, keys::kOnBookmarkImportEnded);
-}
-
-BookmarkAPI::~BookmarkAPI() {
-}
-
-void BookmarkAPI::Shutdown() {
- ExtensionSystem::Get(profile_)->event_router()->UnregisterObserver(this);
-}
-
-void BookmarkAPI::OnListenerAdded(const EventListenerInfo& details) {
- bookmark_event_router_.reset(new BookmarkEventRouter(
- BookmarkModelFactory::GetForProfile(profile_)));
- ExtensionSystem::Get(profile_)->event_router()->UnregisterObserver(this);
-}
-
-bool GetBookmarksFunction::RunImpl() {
- scoped_ptr<bookmarks::Get::Params> params(
- bookmarks::Get::Params::Create(*args_));
- EXTENSION_FUNCTION_VALIDATE(params.get());
-
- std::vector<linked_ptr<BookmarkTreeNode> > nodes;
- BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
- if (params->id_or_id_list_type ==
- bookmarks::Get::Params::ID_OR_ID_LIST_ARRAY) {
- std::vector<std::string>* ids = params->id_or_id_list_array.get();
- size_t count = ids->size();
- EXTENSION_FUNCTION_VALIDATE(count > 0);
- for (size_t i = 0; i < count; ++i) {
- int64 id;
- if (!GetBookmarkIdAsInt64(ids->at(i), &id))
- return false;
- const BookmarkNode* node = model->GetNodeByID(id);
- if (!node) {
- error_ = keys::kNoNodeError;
- return false;
- } else {
- bookmark_api_helpers::AddNode(node, &nodes, false);
- }
- }
- } else {
- int64 id;
- if (!GetBookmarkIdAsInt64(*params->id_or_id_list_string, &id))
- return false;
- const BookmarkNode* node = model->GetNodeByID(id);
- if (!node) {
- error_ = keys::kNoNodeError;
- return false;
- }
- bookmark_api_helpers::AddNode(node, &nodes, false);
- }
-
- results_ = bookmarks::Get::Results::Create(nodes);
- return true;
-}
-
-bool GetBookmarkChildrenFunction::RunImpl() {
- scoped_ptr<bookmarks::GetChildren::Params> params(
- bookmarks::GetChildren::Params::Create(*args_));
- EXTENSION_FUNCTION_VALIDATE(params.get());
-
- int64 id;
- if (!GetBookmarkIdAsInt64(params->id, &id))
- return false;
-
- std::vector<linked_ptr<BookmarkTreeNode> > nodes;
- const BookmarkNode* node =
- BookmarkModelFactory::GetForProfile(profile())->GetNodeByID(id);
- if (!node) {
- error_ = keys::kNoNodeError;
- return false;
- }
- int child_count = node->child_count();
- for (int i = 0; i < child_count; ++i) {
- const BookmarkNode* child = node->GetChild(i);
- bookmark_api_helpers::AddNode(child, &nodes, false);
- }
-
- results_ = bookmarks::GetChildren::Results::Create(nodes);
- return true;
-}
-
-bool GetBookmarkRecentFunction::RunImpl() {
- scoped_ptr<bookmarks::GetRecent::Params> params(
- bookmarks::GetRecent::Params::Create(*args_));
- EXTENSION_FUNCTION_VALIDATE(params.get());
- if (params->number_of_items < 1)
- return false;
-
- std::vector<const BookmarkNode*> nodes;
- bookmark_utils::GetMostRecentlyAddedEntries(
- BookmarkModelFactory::GetForProfile(profile()),
- params->number_of_items,
- &nodes);
-
- std::vector<linked_ptr<BookmarkTreeNode> > tree_nodes;
- std::vector<const BookmarkNode*>::iterator i = nodes.begin();
- for (; i != nodes.end(); ++i) {
- const BookmarkNode* node = *i;
- bookmark_api_helpers::AddNode(node, &tree_nodes, false);
- }
-
- results_ = bookmarks::GetRecent::Results::Create(tree_nodes);
- return true;
-}
-
-bool GetBookmarkTreeFunction::RunImpl() {
- std::vector<linked_ptr<BookmarkTreeNode> > nodes;
- const BookmarkNode* node =
- BookmarkModelFactory::GetForProfile(profile())->root_node();
- bookmark_api_helpers::AddNode(node, &nodes, true);
- results_ = bookmarks::GetTree::Results::Create(nodes);
- return true;
-}
-
-bool GetBookmarkSubTreeFunction::RunImpl() {
- scoped_ptr<bookmarks::GetSubTree::Params> params(
- bookmarks::GetSubTree::Params::Create(*args_));
- EXTENSION_FUNCTION_VALIDATE(params.get());
-
- int64 id;
- if (!GetBookmarkIdAsInt64(params->id, &id))
- return false;
-
- const BookmarkNode* node =
- BookmarkModelFactory::GetForProfile(profile())->GetNodeByID(id);
- if (!node) {
- error_ = keys::kNoNodeError;
- return false;
- }
-
- std::vector<linked_ptr<BookmarkTreeNode> > nodes;
- bookmark_api_helpers::AddNode(node, &nodes, true);
- results_ = bookmarks::GetSubTree::Results::Create(nodes);
- return true;
-}
-
-bool SearchBookmarksFunction::RunImpl() {
- scoped_ptr<bookmarks::Search::Params> params(
- bookmarks::Search::Params::Create(*args_));
- EXTENSION_FUNCTION_VALIDATE(params.get());
-
- PrefServiceBase* prefs = PrefServiceBase::FromBrowserContext(profile_);
- std::string lang = prefs->GetString(prefs::kAcceptLanguages);
- std::vector<const BookmarkNode*> nodes;
- bookmark_utils::GetBookmarksContainingText(
- BookmarkModelFactory::GetForProfile(profile()),
- UTF8ToUTF16(params->query),
- std::numeric_limits<int>::max(),
- lang,
- &nodes);
-
- std::vector<linked_ptr<BookmarkTreeNode> > tree_nodes;
- for (std::vector<const BookmarkNode*>::iterator node_iter = nodes.begin();
- node_iter != nodes.end(); ++node_iter) {
- bookmark_api_helpers::AddNode(*node_iter, &tree_nodes, false);
- }
-
- results_ = bookmarks::Search::Results::Create(tree_nodes);
- return true;
-}
-
-// static
-bool RemoveBookmarkFunction::ExtractIds(const ListValue* args,
- std::list<int64>* ids,
- bool* invalid_id) {
- std::string id_string;
- if (!args->GetString(0, &id_string))
- return false;
- int64 id;
- if (base::StringToInt64(id_string, &id))
- ids->push_back(id);
- else
- *invalid_id = true;
- return true;
-}
-
-bool RemoveBookmarkFunction::RunImpl() {
- if (!EditBookmarksEnabled())
- return false;
-
- scoped_ptr<bookmarks::Remove::Params> params(
- bookmarks::Remove::Params::Create(*args_));
- EXTENSION_FUNCTION_VALIDATE(params.get());
-
- int64 id;
- if (!base::StringToInt64(params->id, &id)) {
- error_ = keys::kInvalidIdError;
- return false;
- }
-
- bool recursive = false;
- if (name() == RemoveTreeBookmarkFunction::function_name())
- recursive = true;
-
- BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
- if (!bookmark_api_helpers::RemoveNode(model, id, recursive, &error_))
- return false;
-
- return true;
-}
-
-bool CreateBookmarkFunction::RunImpl() {
- if (!EditBookmarksEnabled())
- return false;
-
- scoped_ptr<bookmarks::Create::Params> params(
- bookmarks::Create::Params::Create(*args_));
- EXTENSION_FUNCTION_VALIDATE(params.get());
-
- BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
- int64 parentId;
-
- if (!params->bookmark.parent_id.get()) {
- // Optional, default to "other bookmarks".
- parentId = model->other_node()->id();
- } else {
- if (!GetBookmarkIdAsInt64(*params->bookmark.parent_id, &parentId))
- return false;
- }
- const BookmarkNode* parent = model->GetNodeByID(parentId);
- if (!parent) {
- error_ = keys::kNoParentError;
- return false;
- }
- if (parent->is_root()) { // Can't create children of the root.
- error_ = keys::kModifySpecialError;
- return false;
- }
-
- int index;
- if (!params->bookmark.index.get()) { // Optional (defaults to end).
- index = parent->child_count();
- } else {
- index = *params->bookmark.index;
- if (index > parent->child_count() || index < 0) {
- error_ = keys::kInvalidIndexError;
- return false;
- }
- }
-
- string16 title; // Optional.
- if (params->bookmark.title.get())
- title = UTF8ToUTF16(*params->bookmark.title.get());
-
- std::string url_string; // Optional.
- if (params->bookmark.url.get())
- url_string = *params->bookmark.url.get();
-
- GURL url(url_string);
- if (!url_string.empty() && !url.is_valid()) {
- error_ = keys::kInvalidUrlError;
- return false;
- }
-
- const BookmarkNode* node;
- if (url_string.length())
- node = model->AddURL(parent, index, title, url);
- else
- node = model->AddFolder(parent, index, title);
- DCHECK(node);
- if (!node) {
- error_ = keys::kNoNodeError;
- return false;
- }
-
- scoped_ptr<BookmarkTreeNode> ret(
- bookmark_api_helpers::GetBookmarkTreeNode(node, false, false));
- results_ = bookmarks::Create::Results::Create(*ret);
-
- return true;
-}
-
-// static
-bool MoveBookmarkFunction::ExtractIds(const ListValue* args,
- std::list<int64>* ids,
- bool* invalid_id) {
- // For now, Move accepts ID parameters in the same way as an Update.
- return UpdateBookmarkFunction::ExtractIds(args, ids, invalid_id);
-}
-
-bool MoveBookmarkFunction::RunImpl() {
- if (!EditBookmarksEnabled())
- return false;
-
- scoped_ptr<bookmarks::Move::Params> params(
- bookmarks::Move::Params::Create(*args_));
- EXTENSION_FUNCTION_VALIDATE(params.get());
-
- int64 id;
- if (!base::StringToInt64(params->id, &id)) {
- error_ = keys::kInvalidIdError;
- return false;
- }
-
- BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
- const BookmarkNode* node = model->GetNodeByID(id);
- if (!node) {
- error_ = keys::kNoNodeError;
- return false;
- }
- if (model->is_permanent_node(node)) {
- error_ = keys::kModifySpecialError;
- return false;
- }
-
- const BookmarkNode* parent = NULL;
- if (!params->destination.parent_id.get()) {
- // Optional, defaults to current parent.
- parent = node->parent();
- } else {
- int64 parentId;
- if (!GetBookmarkIdAsInt64(*params->destination.parent_id, &parentId))
- return false;
-
- parent = model->GetNodeByID(parentId);
- }
- if (!parent) {
- error_ = keys::kNoParentError;
- // TODO(erikkay) return an error message.
- return false;
- }
- if (parent == model->root_node()) {
- error_ = keys::kModifySpecialError;
- return false;
- }
-
- int index;
- if (params->destination.index.get()) { // Optional (defaults to end).
- index = *params->destination.index;
- if (index > parent->child_count() || index < 0) {
- error_ = keys::kInvalidIndexError;
- return false;
- }
- } else {
- index = parent->child_count();
- }
-
- model->Move(node, parent, index);
-
- scoped_ptr<BookmarkTreeNode> tree_node(
- bookmark_api_helpers::GetBookmarkTreeNode(node, false, false));
- results_ = bookmarks::Move::Results::Create(*tree_node);
-
- return true;
-}
-
-// static
-bool UpdateBookmarkFunction::ExtractIds(const ListValue* args,
- std::list<int64>* ids,
- bool* invalid_id) {
- // For now, Update accepts ID parameters in the same way as an Remove.
- return RemoveBookmarkFunction::ExtractIds(args, ids, invalid_id);
-}
-
-bool UpdateBookmarkFunction::RunImpl() {
- if (!EditBookmarksEnabled())
- return false;
-
- scoped_ptr<bookmarks::Update::Params> params(
- bookmarks::Update::Params::Create(*args_));
- EXTENSION_FUNCTION_VALIDATE(params.get());
-
- int64 id;
- if (!base::StringToInt64(params->id, &id)) {
- error_ = keys::kInvalidIdError;
- return false;
- }
-
- BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile());
-
- // Optional but we need to distinguish non present from an empty title.
- string16 title;
- bool has_title = false;
- if (params->changes.title.get()) {
- title = UTF8ToUTF16(*params->changes.title);
- has_title = true;
- }
-
- // Optional.
- std::string url_string;
- if (params->changes.url.get())
- url_string = *params->changes.url;
- GURL url(url_string);
- if (!url_string.empty() && !url.is_valid()) {
- error_ = keys::kInvalidUrlError;
- return false;
- }
-
- const BookmarkNode* node = model->GetNodeByID(id);
- if (!node) {
- error_ = keys::kNoNodeError;
- return false;
- }
- if (model->is_permanent_node(node)) {
- error_ = keys::kModifySpecialError;
- return false;
- }
- if (has_title)
- model->SetTitle(node, title);
- if (!url.is_empty())
- model->SetURL(node, url);
-
- scoped_ptr<BookmarkTreeNode> tree_node(
- bookmark_api_helpers::GetBookmarkTreeNode(node, false, false));
- results_ = bookmarks::Update::Results::Create(*tree_node);
- return true;
-}
-
-// Mapper superclass for BookmarkFunctions.
-template <typename BucketIdType>
-class BookmarkBucketMapper : public BucketMapper {
- public:
- virtual ~BookmarkBucketMapper() { STLDeleteValues(&buckets_); }
- protected:
- Bucket* GetBucket(const BucketIdType& id) {
- Bucket* b = buckets_[id];
- if (b == NULL) {
- b = new Bucket();
- buckets_[id] = b;
- }
- return b;
- }
- private:
- std::map<BucketIdType, Bucket*> buckets_;
-};
-
-// Mapper for 'bookmarks.create'. Maps "same input to bookmarks.create" to a
-// unique bucket.
-class CreateBookmarkBucketMapper : public BookmarkBucketMapper<std::string> {
- public:
- explicit CreateBookmarkBucketMapper(Profile* profile) : profile_(profile) {}
- // TODO(tim): This should share code with CreateBookmarkFunction::RunImpl,
- // but I can't figure out a good way to do that with all the macros.
- virtual void GetBucketsForArgs(const ListValue* args, BucketList* buckets) {
- const DictionaryValue* json;
- if (!args->GetDictionary(0, &json))
- return;
-
- std::string parent_id;
- if (json->HasKey(keys::kParentIdKey)) {
- if (!json->GetString(keys::kParentIdKey, &parent_id))
- return;
- }
- BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile_);
-
- int64 parent_id_int64;
- base::StringToInt64(parent_id, &parent_id_int64);
- const BookmarkNode* parent = model->GetNodeByID(parent_id_int64);
- if (!parent)
- return;
-
- std::string bucket_id = UTF16ToUTF8(parent->GetTitle());
- std::string title;
- json->GetString(keys::kTitleKey, &title);
- std::string url_string;
- json->GetString(keys::kUrlKey, &url_string);
-
- bucket_id += title;
- bucket_id += url_string;
- // 20 bytes (SHA1 hash length) is very likely less than most of the
- // |bucket_id| strings we construct here, so we hash it to save space.
- buckets->push_back(GetBucket(base::SHA1HashString(bucket_id)));
- }
- private:
- Profile* profile_;
-};
-
-// Mapper for 'bookmarks.remove'.
-class RemoveBookmarksBucketMapper : public BookmarkBucketMapper<std::string> {
- public:
- explicit RemoveBookmarksBucketMapper(Profile* profile) : profile_(profile) {}
- virtual void GetBucketsForArgs(const ListValue* args, BucketList* buckets) {
- typedef std::list<int64> IdList;
- IdList ids;
- bool invalid_id = false;
- if (!RemoveBookmarkFunction::ExtractIds(args, &ids, &invalid_id) ||
- invalid_id) {
- return;
- }
-
- for (IdList::iterator it = ids.begin(); it != ids.end(); ++it) {
- BookmarkModel* model = BookmarkModelFactory::GetForProfile(profile_);
- const BookmarkNode* node = model->GetNodeByID(*it);
- if (!node || node->is_root())
- return;
-
- std::string bucket_id;
- bucket_id += UTF16ToUTF8(node->parent()->GetTitle());
- bucket_id += UTF16ToUTF8(node->GetTitle());
- bucket_id += node->url().spec();
- buckets->push_back(GetBucket(base::SHA1HashString(bucket_id)));
- }
- }
- private:
- Profile* profile_;
-};
-
-// Mapper for any bookmark function accepting bookmark IDs as parameters, where
-// a distinct ID corresponds to a single item in terms of quota limiting. This
-// is inappropriate for bookmarks.remove, for example, since repeated removals
-// of the same item will actually have a different ID each time.
-template <class FunctionType>
-class BookmarkIdMapper : public BookmarkBucketMapper<int64> {
- public:
- typedef std::list<int64> IdList;
- virtual void GetBucketsForArgs(const ListValue* args, BucketList* buckets) {
- IdList ids;
- bool invalid_id = false;
- if (!FunctionType::ExtractIds(args, &ids, &invalid_id) || invalid_id)
- return;
- for (IdList::iterator it = ids.begin(); it != ids.end(); ++it)
- buckets->push_back(GetBucket(*it));
- }
-};
-
-// Builds heuristics for all BookmarkFunctions using specialized BucketMappers.
-class BookmarksQuotaLimitFactory {
- public:
- // For id-based bookmark functions.
- template <class FunctionType>
- static void Build(QuotaLimitHeuristics* heuristics) {
- BuildWithMappers(heuristics, new BookmarkIdMapper<FunctionType>(),
- new BookmarkIdMapper<FunctionType>());
- }
-
- // For bookmarks.create.
- static void BuildForCreate(QuotaLimitHeuristics* heuristics,
- Profile* profile) {
- BuildWithMappers(heuristics, new CreateBookmarkBucketMapper(profile),
- new CreateBookmarkBucketMapper(profile));
- }
-
- // For bookmarks.remove.
- static void BuildForRemove(QuotaLimitHeuristics* heuristics,
- Profile* profile) {
- BuildWithMappers(heuristics, new RemoveBookmarksBucketMapper(profile),
- new RemoveBookmarksBucketMapper(profile));
- }
-
- private:
- static void BuildWithMappers(QuotaLimitHeuristics* heuristics,
- BucketMapper* short_mapper, BucketMapper* long_mapper) {
- const Config kSustainedLimitConfig = {
- // See bookmarks.json for current value.
- bookmarks::MAX_SUSTAINED_WRITE_OPERATIONS_PER_MINUTE,
- TimeDelta::FromMinutes(1)
- };
- heuristics->push_back(new SustainedLimit(
- TimeDelta::FromMinutes(10),
- kSustainedLimitConfig,
- short_mapper,
- "MAX_SUSTAINED_WRITE_OPERATIONS_PER_MINUTE"));
-
- const Config kTimedLimitConfig = {
- // See bookmarks.json for current value.
- bookmarks::MAX_WRITE_OPERATIONS_PER_HOUR,
- TimeDelta::FromHours(1)
- };
- heuristics->push_back(new TimedLimit(
- kTimedLimitConfig,
- long_mapper,
- "MAX_WRITE_OPERATIONS_PER_HOUR"));
- }
-
- DISALLOW_IMPLICIT_CONSTRUCTORS(BookmarksQuotaLimitFactory);
-};
-
-// And finally, building the individual heuristics for each function.
-void RemoveBookmarkFunction::GetQuotaLimitHeuristics(
- QuotaLimitHeuristics* heuristics) const {
- BookmarksQuotaLimitFactory::BuildForRemove(heuristics, profile());
-}
-
-void MoveBookmarkFunction::GetQuotaLimitHeuristics(
- QuotaLimitHeuristics* heuristics) const {
- BookmarksQuotaLimitFactory::Build<MoveBookmarkFunction>(heuristics);
-}
-
-void UpdateBookmarkFunction::GetQuotaLimitHeuristics(
- QuotaLimitHeuristics* heuristics) const {
- BookmarksQuotaLimitFactory::Build<UpdateBookmarkFunction>(heuristics);
-};
-
-void CreateBookmarkFunction::GetQuotaLimitHeuristics(
- QuotaLimitHeuristics* heuristics) const {
- BookmarksQuotaLimitFactory::BuildForCreate(heuristics, profile());
-}
-
-BookmarksIOFunction::BookmarksIOFunction() {}
-
-BookmarksIOFunction::~BookmarksIOFunction() {
- // There may be pending file dialogs, we need to tell them that we've gone
- // away so they don't try and call back to us.
- if (select_file_dialog_.get())
- select_file_dialog_->ListenerDestroyed();
-}
-
-void BookmarksIOFunction::SelectFile(ui::SelectFileDialog::Type type) {
- // GetDefaultFilepathForBookmarkExport() might have to touch the filesystem
- // (stat or access, for example), so this requires a thread with IO allowed.
- if (!BrowserThread::CurrentlyOn(BrowserThread::FILE)) {
- BrowserThread::PostTask(BrowserThread::FILE, FROM_HERE,
- base::Bind(&BookmarksIOFunction::SelectFile, this, type));
- return;
- }
-
- // Pre-populating the filename field in case this is a SELECT_SAVEAS_FILE
- // dialog. If not, there is no filename field in the dialog box.
- FilePath default_path;
- if (type == ui::SelectFileDialog::SELECT_SAVEAS_FILE)
- default_path = GetDefaultFilepathForBookmarkExport();
- else
- DCHECK(type == ui::SelectFileDialog::SELECT_OPEN_FILE);
-
- // After getting the |default_path|, ask the UI to display the file dialog.
- BrowserThread::PostTask(BrowserThread::UI, FROM_HERE,
- base::Bind(&BookmarksIOFunction::ShowSelectFileDialog, this,
- type, default_path));
-}
-
-void BookmarksIOFunction::ShowSelectFileDialog(ui::SelectFileDialog::Type type,
- const FilePath& default_path) {
- // Balanced in one of the three callbacks of SelectFileDialog:
- // either FileSelectionCanceled, MultiFilesSelected, or FileSelected
- AddRef();
-
- WebContents* web_contents = dispatcher()->delegate()->
- GetAssociatedWebContents();
-
- select_file_dialog_ = ui::SelectFileDialog::Create(
- this, new ChromeSelectFilePolicy(web_contents));
- ui::SelectFileDialog::FileTypeInfo file_type_info;
- file_type_info.extensions.resize(1);
- file_type_info.extensions[0].push_back(FILE_PATH_LITERAL("html"));
- // TODO(kinaba): http://crbug.com/140425. Turn file_type_info.support_gdata
- // on for saving once Google Drive client on ChromeOS supports it.
- if (type == ui::SelectFileDialog::SELECT_OPEN_FILE)
- file_type_info.support_gdata = true;
- // |web_contents| can be NULL (for background pages), which is fine. In such
- // a case if file-selection dialogs are forbidden by policy, we will not
- // show an InfoBar, which is better than letting one appear out of the blue.
- select_file_dialog_->SelectFile(type,
- string16(),
- default_path,
- &file_type_info,
- 0,
- FILE_PATH_LITERAL(""),
- NULL,
- NULL);
-}
-
-void BookmarksIOFunction::FileSelectionCanceled(void* params) {
- Release(); // Balanced in BookmarksIOFunction::SelectFile()
-}
-
-void BookmarksIOFunction::MultiFilesSelected(
- const std::vector<FilePath>& files, void* params) {
- Release(); // Balanced in BookmarsIOFunction::SelectFile()
- NOTREACHED() << "Should not be able to select multiple files";
-}
-
-bool ImportBookmarksFunction::RunImpl() {
- if (!EditBookmarksEnabled())
- return false;
- SelectFile(ui::SelectFileDialog::SELECT_OPEN_FILE);
- return true;
-}
-
-void ImportBookmarksFunction::FileSelected(const FilePath& path,
- int index,
- void* params) {
-#if !defined(OS_ANDROID)
- // Android does not have support for the standard importers.
- // TODO(jgreenwald): remove ifdef once extensions are no longer built on
- // Android.
- scoped_refptr<ImporterHost> importer_host(new ImporterHost);
- importer::SourceProfile source_profile;
- source_profile.importer_type = importer::TYPE_BOOKMARKS_FILE;
- source_profile.source_path = path;
- importer_host->StartImportSettings(source_profile,
- profile(),
- importer::FAVORITES,
- new ProfileWriter(profile()),
- true);
-#endif
- Release(); // Balanced in BookmarksIOFunction::SelectFile()
-}
-
-bool ExportBookmarksFunction::RunImpl() {
- SelectFile(ui::SelectFileDialog::SELECT_SAVEAS_FILE);
- return true;
-}
-
-void ExportBookmarksFunction::FileSelected(const FilePath& path,
- int index,
- void* params) {
-#if !defined(OS_ANDROID)
- // Android does not have support for the standard exporter.
- // TODO(jgreenwald): remove ifdef once extensions are no longer built on
- // Android.
- bookmark_html_writer::WriteBookmarks(profile(), path, NULL);
-#endif
- Release(); // Balanced in BookmarksIOFunction::SelectFile()
-}
-
-} // namespace extensions

Powered by Google App Engine
This is Rietveld 408576698