// Copyright (c) 2010 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/cookies_tree_model.h" #include #include #include #include "app/l10n_util.h" #include "app/resource_bundle.h" #include "app/table_model_observer.h" #include "app/tree_node_model.h" #include "base/callback.h" #include "base/i18n/rtl.h" #include "base/linked_ptr.h" #include "base/string_util.h" #include "chrome/browser/extensions/extensions_service.h" #include "chrome/browser/in_process_webkit/webkit_context.h" #include "chrome/browser/net/chrome_url_request_context.h" #include "chrome/browser/profile.h" #include "grit/app_resources.h" #include "grit/generated_resources.h" #include "grit/theme_resources.h" #include "net/base/cookie_monster.h" #include "net/base/registry_controlled_domain.h" #include "net/url_request/url_request_context.h" #include "third_party/skia/include/core/SkBitmap.h" static const char kFileOriginNodeName[] = "file://"; /////////////////////////////////////////////////////////////////////////////// // CookieTreeNode, public: void CookieTreeNode::DeleteStoredObjects() { std::for_each(children().begin(), children().end(), std::mem_fun(&CookieTreeNode::DeleteStoredObjects)); } CookiesTreeModel* CookieTreeNode::GetModel() const { if (GetParent()) return GetParent()->GetModel(); else return NULL; } /////////////////////////////////////////////////////////////////////////////// // CookieTreeCookieNode, public: CookieTreeCookieNode::CookieTreeCookieNode( net::CookieMonster::CookieListPair* cookie) : CookieTreeNode(UTF8ToWide(cookie->second.Name())), cookie_(cookie) { } void CookieTreeCookieNode::DeleteStoredObjects() { // notify CookieMonster that we should delete this cookie // Since we are running on the UI thread don't call GetURLRequestContext(). net::CookieMonster* monster = GetModel()->profile_-> GetRequestContext()->GetCookieStore()->GetCookieMonster(); // We have stored a copy of all the cookies in the model, and our model is // never re-calculated. Thus, we just need to delete the nodes from our // model, and tell CookieMonster to delete the cookies. We can keep the // vector storing the cookies in-tact and not delete from there (that would // invalidate our pointers), and the fact that it contains semi out-of-date // data is not problematic as we don't re-build the model based on that. monster->DeleteCookie(cookie_->first, cookie_->second, true); } namespace { // comparison functor, for use in CookieTreeRootNode class OriginNodeComparator { public: bool operator() (const CookieTreeNode* lhs, const CookieTreeNode* rhs) { // We want to order by registry controlled domain, so we would get // google.com, ad.google.com, www.google.com, // microsoft.com, ad.microsoft.com. CanonicalizeHost transforms the origins // into a form like google.com.www so that string comparisons work. return (CanonicalizeHost(lhs->GetTitleAsString16()) < CanonicalizeHost(rhs->GetTitleAsString16())); } private: static std::string CanonicalizeHost(const string16& host16) { // The canonicalized representation makes the registry controlled domain // come first, and then adds subdomains in reverse order, e.g. // 1.mail.google.com would become google.com.mail.1, and then a standard // string comparison works to order hosts by registry controlled domain // first. Leading dots are ignored, ".google.com" is the same as // "google.com". std::string host = UTF16ToUTF8(host16); std::string retval = net::RegistryControlledDomainService:: GetDomainAndRegistry(host); if (!retval.length()) // Is an IP address or other special origin. return host; std::string::size_type position = host.rfind(retval); // The host may be the registry controlled domain, in which case fail fast. if (position == 0 || position == std::string::npos) return host; // If host is www.google.com, retval will contain google.com at this point. // Start operating to the left of the registry controlled domain, e.g. in // the www.google.com example, start at index 3. --position; // If position == 0, that means it's a dot; this will be ignored to treat // ".google.com" the same as "google.com". while (position > 0) { retval += std::string("."); // Copy up to the next dot. host[position] is a dot so start after it. std::string::size_type next_dot = host.rfind(".", position - 1); if (next_dot == std::string::npos) { retval += host.substr(0, position); break; } retval += host.substr(next_dot + 1, position - (next_dot + 1)); position = next_dot; } return retval; } }; } // namespace /////////////////////////////////////////////////////////////////////////////// // CookieTreeAppCacheNode, public: CookieTreeAppCacheNode::CookieTreeAppCacheNode( const appcache::AppCacheInfo* appcache_info) : CookieTreeNode(UTF8ToWide(appcache_info->manifest_url.spec())), appcache_info_(appcache_info) { } void CookieTreeAppCacheNode::DeleteStoredObjects() { DCHECK(GetModel()->appcache_helper_); GetModel()->appcache_helper_->DeleteAppCacheGroup( appcache_info_->manifest_url); } /////////////////////////////////////////////////////////////////////////////// // CookieTreeDatabaseNode, public: CookieTreeDatabaseNode::CookieTreeDatabaseNode( BrowsingDataDatabaseHelper::DatabaseInfo* database_info) : CookieTreeNode(database_info->database_name.empty() ? l10n_util::GetString(IDS_COOKIES_WEB_DATABASE_UNNAMED_NAME) : UTF8ToWide(database_info->database_name)), database_info_(database_info) { } void CookieTreeDatabaseNode::DeleteStoredObjects() { GetModel()->database_helper_->DeleteDatabase( database_info_->origin_identifier, database_info_->database_name); } /////////////////////////////////////////////////////////////////////////////// // CookieTreeLocalStorageNode, public: CookieTreeLocalStorageNode::CookieTreeLocalStorageNode( BrowsingDataLocalStorageHelper::LocalStorageInfo* local_storage_info) : CookieTreeNode(UTF8ToWide( local_storage_info->origin.empty() ? local_storage_info->database_identifier : local_storage_info->origin)), local_storage_info_(local_storage_info) { } void CookieTreeLocalStorageNode::DeleteStoredObjects() { GetModel()->local_storage_helper_->DeleteLocalStorageFile( local_storage_info_->file_path); } /////////////////////////////////////////////////////////////////////////////// // CookieTreeRootNode, public: CookieTreeOriginNode* CookieTreeRootNode::GetOrCreateOriginNode( const std::wstring& origin) { // Strip the leading dot if it exists. std::wstring rewritten_origin = origin; if (origin.length() >= 1 && origin[0] == '.') rewritten_origin = origin.substr(1); CookieTreeOriginNode rewritten_origin_node(rewritten_origin); // First see if there is an existing match. std::vector::iterator origin_node_iterator = lower_bound(children().begin(), children().end(), &rewritten_origin_node, OriginNodeComparator()); if (origin_node_iterator != children().end() && rewritten_origin == (*origin_node_iterator)->GetTitle()) return static_cast(*origin_node_iterator); // Node doesn't exist, create a new one and insert it into the (ordered) // children. CookieTreeOriginNode* retval = new CookieTreeOriginNode(rewritten_origin); DCHECK(model_); model_->Add(this, (origin_node_iterator - children().begin()), retval); return retval; } /////////////////////////////////////////////////////////////////////////////// // CookieTreeOriginNode, public: CookieTreeCookiesNode* CookieTreeOriginNode::GetOrCreateCookiesNode() { if (cookies_child_) return cookies_child_; cookies_child_ = new CookieTreeCookiesNode; AddChildSortedByTitle(cookies_child_); return cookies_child_; } CookieTreeDatabasesNode* CookieTreeOriginNode::GetOrCreateDatabasesNode() { if (databases_child_) return databases_child_; databases_child_ = new CookieTreeDatabasesNode; AddChildSortedByTitle(databases_child_); return databases_child_; } CookieTreeLocalStoragesNode* CookieTreeOriginNode::GetOrCreateLocalStoragesNode() { if (local_storages_child_) return local_storages_child_; local_storages_child_ = new CookieTreeLocalStoragesNode; AddChildSortedByTitle(local_storages_child_); return local_storages_child_; } CookieTreeAppCachesNode* CookieTreeOriginNode::GetOrCreateAppCachesNode() { if (appcaches_child_) return appcaches_child_; appcaches_child_ = new CookieTreeAppCachesNode; AddChildSortedByTitle(appcaches_child_); return appcaches_child_; } /////////////////////////////////////////////////////////////////////////////// // CookieTreeCookiesNode, public: CookieTreeCookiesNode::CookieTreeCookiesNode() : CookieTreeNode(l10n_util::GetString(IDS_COOKIES_COOKIES)) { } /////////////////////////////////////////////////////////////////////////////// // CookieTreeAppCachesNode, public: CookieTreeAppCachesNode::CookieTreeAppCachesNode() : CookieTreeNode(l10n_util::GetString(IDS_COOKIES_APPLICATION_CACHES)) { } /////////////////////////////////////////////////////////////////////////////// // CookieTreeDatabasesNode, public: CookieTreeDatabasesNode::CookieTreeDatabasesNode() : CookieTreeNode(l10n_util::GetString(IDS_COOKIES_WEB_DATABASES)) { } /////////////////////////////////////////////////////////////////////////////// // CookieTreeLocalStoragesNode, public: CookieTreeLocalStoragesNode::CookieTreeLocalStoragesNode() : CookieTreeNode(l10n_util::GetString(IDS_COOKIES_LOCAL_STORAGE)) { } /////////////////////////////////////////////////////////////////////////////// // CookieTreeNode, protected bool CookieTreeNode::NodeTitleComparator::operator() ( const CookieTreeNode* lhs, const CookieTreeNode* rhs) { const CookieTreeNode* left = static_cast(lhs); const CookieTreeNode* right = static_cast(rhs); return (left->GetTitleAsString16() < right->GetTitleAsString16()); } void CookieTreeNode::AddChildSortedByTitle(CookieTreeNode* new_child) { std::vector::iterator iter = lower_bound(children().begin(), children().end(), new_child, NodeTitleComparator()); GetModel()->Add(this, iter - children().begin(), new_child); } /////////////////////////////////////////////////////////////////////////////// // CookiesTreeModel, public: CookiesTreeModel::CookiesTreeModel( Profile* profile, BrowsingDataDatabaseHelper* database_helper, BrowsingDataLocalStorageHelper* local_storage_helper, BrowsingDataAppCacheHelper* appcache_helper) : ALLOW_THIS_IN_INITIALIZER_LIST(TreeNodeModel( new CookieTreeRootNode(this))), profile_(profile), appcache_helper_(appcache_helper), database_helper_(database_helper), local_storage_helper_(local_storage_helper), batch_update_(0) { LoadCookies(); DCHECK(database_helper_); database_helper_->StartFetching(NewCallback( this, &CookiesTreeModel::OnDatabaseModelInfoLoaded)); DCHECK(local_storage_helper_); local_storage_helper_->StartFetching(NewCallback( this, &CookiesTreeModel::OnStorageModelInfoLoaded)); // TODO(michaeln): when all of the ui impls have been updated, // make this a required parameter. if (appcache_helper_) { appcache_helper_->StartFetching(NewCallback( this, &CookiesTreeModel::OnAppCacheModelInfoLoaded)); } } CookiesTreeModel::~CookiesTreeModel() { database_helper_->CancelNotification(); local_storage_helper_->CancelNotification(); if (appcache_helper_) appcache_helper_->CancelNotification(); } /////////////////////////////////////////////////////////////////////////////// // CookiesTreeModel, TreeModel methods (public): // TreeModel methods: // Returns the set of icons for the nodes in the tree. You only need override // this if you don't want to use the default folder icons. void CookiesTreeModel::GetIcons(std::vector* icons) { icons->push_back(*ResourceBundle::GetSharedInstance().GetBitmapNamed( IDR_DEFAULT_FAVICON)); icons->push_back(*ResourceBundle::GetSharedInstance().GetBitmapNamed( IDR_COOKIE_ICON)); icons->push_back(*ResourceBundle::GetSharedInstance().GetBitmapNamed( IDR_COOKIE_STORAGE_ICON)); } // Returns the index of the icon to use for |node|. Return -1 to use the // default icon. The index is relative to the list of icons returned from // GetIcons. int CookiesTreeModel::GetIconIndex(TreeModelNode* node) { CookieTreeNode* ct_node = static_cast(node); switch (ct_node->GetDetailedInfo().node_type) { case CookieTreeNode::DetailedInfo::TYPE_ORIGIN: return ORIGIN; break; case CookieTreeNode::DetailedInfo::TYPE_COOKIE: return COOKIE; break; case CookieTreeNode::DetailedInfo::TYPE_DATABASE: return DATABASE; break; case CookieTreeNode::DetailedInfo::TYPE_LOCAL_STORAGE: return DATABASE; // close enough break; case CookieTreeNode::DetailedInfo::TYPE_APPCACHE: return DATABASE; // ditto break; default: return -1; } } void CookiesTreeModel::LoadCookies() { LoadCookiesWithFilter(std::wstring()); } void CookiesTreeModel::LoadCookiesWithFilter(const std::wstring& filter) { // mmargh mmargh mmargh! // Since we are running on the UI thread don't call GetURLRequestContext(). net::CookieMonster* cookie_monster = profile_->GetRequestContext()->GetCookieStore()->GetCookieMonster(); all_cookies_ = cookie_monster->GetAllCookies(); CookieTreeRootNode* root = static_cast(GetRoot()); for (CookieList::iterator it = all_cookies_.begin(); it != all_cookies_.end(); ++it) { std::wstring origin_node_name = UTF8ToWide(it->first); if (!filter.size() || (origin_node_name.find(filter) != std::wstring::npos)) { CookieTreeOriginNode* origin_node = root->GetOrCreateOriginNode(origin_node_name); CookieTreeCookiesNode* cookies_node = origin_node->GetOrCreateCookiesNode(); CookieTreeCookieNode* new_cookie = new CookieTreeCookieNode(&*it); cookies_node->AddCookieNode(new_cookie); } } } void CookiesTreeModel::DeleteAllStoredObjects() { NotifyObserverBeginBatch(); CookieTreeNode* root = GetRoot(); root->DeleteStoredObjects(); int num_children = root->GetChildCount(); for (int i = num_children - 1; i >= 0; --i) delete Remove(root, i); NotifyObserverTreeNodeChanged(root); NotifyObserverEndBatch(); } void CookiesTreeModel::DeleteCookieNode(CookieTreeNode* cookie_node) { cookie_node->DeleteStoredObjects(); // find the parent and index CookieTreeNode* parent_node = cookie_node->GetParent(); int cookie_node_index = parent_node->IndexOfChild(cookie_node); delete Remove(parent_node, cookie_node_index); } void CookiesTreeModel::UpdateSearchResults(const std::wstring& filter) { CookieTreeNode* root = GetRoot(); int num_children = root->GetChildCount(); NotifyObserverBeginBatch(); for (int i = num_children - 1; i >= 0; --i) delete Remove(root, i); LoadCookiesWithFilter(filter); PopulateDatabaseInfoWithFilter(filter); PopulateLocalStorageInfoWithFilter(filter); PopulateAppCacheInfoWithFilter(filter); NotifyObserverTreeNodeChanged(root); NotifyObserverEndBatch(); } void CookiesTreeModel::AddObserver(Observer* observer) { cookies_observer_list_.AddObserver(observer); // Call super so that TreeNodeModel can notify, too. TreeNodeModel::AddObserver(observer); } void CookiesTreeModel::RemoveObserver(Observer* observer) { cookies_observer_list_.RemoveObserver(observer); // Call super so that TreeNodeModel doesn't have dead pointers. TreeNodeModel::RemoveObserver(observer); } void CookiesTreeModel::OnAppCacheModelInfoLoaded() { appcache_info_ = appcache_helper_->info_collection(); PopulateAppCacheInfoWithFilter(std::wstring()); } void CookiesTreeModel::PopulateAppCacheInfoWithFilter( const std::wstring& filter) { using appcache::AppCacheInfo; using appcache::AppCacheInfoVector; typedef std::map InfoByOrigin; if (!appcache_info_ || appcache_info_->infos_by_origin.empty()) return; CookieTreeRootNode* root = static_cast(GetRoot()); NotifyObserverBeginBatch(); for (InfoByOrigin::const_iterator origin = appcache_info_->infos_by_origin.begin(); origin != appcache_info_->infos_by_origin.end(); ++origin) { std::wstring origin_node_name = UTF8ToWide(origin->first.host()); if (filter.empty() || (origin_node_name.find(filter) != std::wstring::npos)) { CookieTreeOriginNode* origin_node = root->GetOrCreateOriginNode(origin_node_name); CookieTreeAppCachesNode* appcaches_node = origin_node->GetOrCreateAppCachesNode(); for (AppCacheInfoVector::const_iterator info = origin->second.begin(); info != origin->second.end(); ++info) { appcaches_node->AddAppCacheNode( new CookieTreeAppCacheNode(&(*info))); } } } NotifyObserverTreeNodeChanged(root); NotifyObserverEndBatch(); } void CookiesTreeModel::OnDatabaseModelInfoLoaded( const DatabaseInfoList& database_info) { database_info_list_ = database_info; PopulateDatabaseInfoWithFilter(std::wstring()); } void CookiesTreeModel::PopulateDatabaseInfoWithFilter( const std::wstring& filter) { if (database_info_list_.empty()) return; CookieTreeRootNode* root = static_cast(GetRoot()); NotifyObserverBeginBatch(); for (DatabaseInfoList::iterator database_info = database_info_list_.begin(); database_info != database_info_list_.end(); ++database_info) { // Determine which 'origin' node to place each 'info' in. std::wstring origin_node_name; if (database_info->IsFileSchemeData()) origin_node_name = UTF8ToWide(kFileOriginNodeName); else if (database_info->IsExtensionSchemeData()) origin_node_name = FormExtensionNodeName(database_info->host); else origin_node_name = UTF8ToWide(database_info->host); if (!filter.size() || (origin_node_name.find(filter) != std::wstring::npos)) { CookieTreeOriginNode* origin_node = root->GetOrCreateOriginNode(origin_node_name); CookieTreeDatabasesNode* databases_node = origin_node->GetOrCreateDatabasesNode(); databases_node->AddDatabaseNode( new CookieTreeDatabaseNode(&(*database_info))); } } NotifyObserverTreeNodeChanged(root); NotifyObserverEndBatch(); } void CookiesTreeModel::OnStorageModelInfoLoaded( const LocalStorageInfoList& local_storage_info) { local_storage_info_list_ = local_storage_info; PopulateLocalStorageInfoWithFilter(std::wstring()); } void CookiesTreeModel::PopulateLocalStorageInfoWithFilter( const std::wstring& filter) { if (local_storage_info_list_.empty()) return; CookieTreeRootNode* root = static_cast(GetRoot()); NotifyObserverBeginBatch(); for (LocalStorageInfoList::iterator local_storage_info = local_storage_info_list_.begin(); local_storage_info != local_storage_info_list_.end(); ++local_storage_info) { // Determine which 'origin' node to place each 'info' in. std::wstring origin_node_name; if (local_storage_info->IsFileSchemeData()) origin_node_name = UTF8ToWide(kFileOriginNodeName); else if (local_storage_info->IsExtensionSchemeData()) origin_node_name = FormExtensionNodeName(local_storage_info->host); else origin_node_name = UTF8ToWide(local_storage_info->host); if (!filter.size() || (origin_node_name.find(filter) != std::wstring::npos)) { CookieTreeOriginNode* origin_node = root->GetOrCreateOriginNode(origin_node_name); CookieTreeLocalStoragesNode* local_storages_node = origin_node->GetOrCreateLocalStoragesNode(); local_storages_node->AddLocalStorageNode( new CookieTreeLocalStorageNode(&(*local_storage_info))); } } NotifyObserverTreeNodeChanged(root); NotifyObserverEndBatch(); } void CookiesTreeModel::NotifyObserverBeginBatch() { // Only notify the model once if we're batching in a nested manner. if (batch_update_++ == 0) { FOR_EACH_OBSERVER(Observer, cookies_observer_list_, TreeModelBeginBatch(this)); } } void CookiesTreeModel::NotifyObserverEndBatch() { // Only notify the observers if this is the outermost call to EndBatch() if // called in a nested manner. if (--batch_update_ == 0) { FOR_EACH_OBSERVER(Observer, cookies_observer_list_, TreeModelEndBatch(this)); } } std::wstring CookiesTreeModel::FormExtensionNodeName( const std::string& extension_id) { Extension* extension = profile_->GetExtensionsService()->GetExtensionById(extension_id, true); std::wstring extension_name = extension ? UTF8ToWide(extension->name()) : l10n_util::GetString(IDS_UNKNOWN_PLUGIN_NAME); // Since the extension_name will be concatenated with a prefix, we need // to explicitly set the extension_name to be LTR format if there is no // strong RTL charater in it. Otherwise, if the prefix is an RTL word, // the concatenated result might be wrong. For extension named // "Great Extension!" the concatenated result would be something like // "!Great Extension :NOISNETXE", in which capital letters "NOISNETXE" // stand for the Hebrew word for "extension". base::i18n::AdjustStringForLocaleDirection(extension_name, &extension_name); return l10n_util::GetStringF(IDS_TASK_MANAGER_EXTENSION_PREFIX, extension_name); }