// Copyright (c) 2011 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/search_engines/util.h" #include <set> #include <vector> #include "base/logging.h" #include "chrome/browser/search_engines/template_url.h" #include "chrome/browser/search_engines/template_url_model.h" #include "chrome/browser/search_engines/template_url_prepopulate_data.h" #include "chrome/browser/prefs/pref_service.h" #include "chrome/browser/profiles/profile.h" #include "content/browser/browser_thread.h" string16 GetDefaultSearchEngineName(Profile* profile) { if (!profile) { NOTREACHED(); return string16(); } const TemplateURL* const default_provider = profile->GetTemplateURLModel()->GetDefaultSearchProvider(); if (!default_provider) { // TODO(cpu): bug 1187517. It is possible to have no default provider. // returning an empty string is a stopgap measure for the crash // http://code.google.com/p/chromium/issues/detail?id=2573 return string16(); } return default_provider->short_name(); } // Removes (and deletes) TemplateURLs from |urls| that have duplicate // prepopulate ids. Duplicate prepopulate ids are not allowed, but due to a // bug it was possible get dups. This step is only called when the version // number changes. Only pass in a non-NULL value for |service| if the removed // items should be removed from the DB. static void RemoveDuplicatePrepopulateIDs( std::vector<TemplateURL*>* template_urls, WebDataService* service) { DCHECK(template_urls); DCHECK(service == NULL || BrowserThread::CurrentlyOn(BrowserThread::UI)); std::set<int> ids; for (std::vector<TemplateURL*>::iterator i = template_urls->begin(); i != template_urls->end(); ) { int prepopulate_id = (*i)->prepopulate_id(); if (prepopulate_id) { if (ids.find(prepopulate_id) != ids.end()) { if (service) service->RemoveKeyword(**i); delete *i; i = template_urls->erase(i); } else { ids.insert(prepopulate_id); ++i; } } else { ++i; } } } // Loads engines from prepopulate data and merges them in with the existing // engines. This is invoked when the version of the prepopulate data changes. void MergeEnginesFromPrepopulateData( PrefService* prefs, WebDataService* service, std::vector<TemplateURL*>* template_urls, const TemplateURL** default_search_provider) { DCHECK(service == NULL || BrowserThread::CurrentlyOn(BrowserThread::UI)); DCHECK(template_urls); DCHECK(default_search_provider); // Build a map from prepopulate id to TemplateURL of existing urls. typedef std::map<int, TemplateURL*> IDMap; IDMap id_to_turl; for (std::vector<TemplateURL*>::iterator i(template_urls->begin()); i != template_urls->end(); ++i) { int prepopulate_id = (*i)->prepopulate_id(); if (prepopulate_id > 0) id_to_turl[prepopulate_id] = *i; } std::vector<TemplateURL*> prepopulated_urls; size_t default_search_index; TemplateURLPrepopulateData::GetPrepopulatedEngines(prefs, &prepopulated_urls, &default_search_index); std::set<int> updated_ids; for (size_t i = 0; i < prepopulated_urls.size(); ++i) { // We take ownership of |prepopulated_urls[i]|. scoped_ptr<TemplateURL> prepopulated_url(prepopulated_urls[i]); const int prepopulated_id = prepopulated_url->prepopulate_id(); if (!prepopulated_id || updated_ids.count(prepopulated_id)) { // Prepopulate engines need a unique id. NOTREACHED(); continue; } TemplateURL* existing_url = NULL; IDMap::iterator existing_url_iter(id_to_turl.find(prepopulated_id)); if (existing_url_iter != id_to_turl.end()) { existing_url = existing_url_iter->second; if (!existing_url->safe_for_autoreplace()) { // User edited the entry, preserve the keyword and description. prepopulated_url->set_safe_for_autoreplace(false); prepopulated_url->set_keyword(existing_url->keyword()); prepopulated_url->set_autogenerate_keyword( existing_url->autogenerate_keyword()); prepopulated_url->set_short_name(existing_url->short_name()); } prepopulated_url->set_id(existing_url->id()); *existing_url = *prepopulated_url; if (service) { service->UpdateKeyword(*existing_url); } id_to_turl.erase(existing_url_iter); } else { existing_url = prepopulated_url.get(); template_urls->push_back(prepopulated_url.release()); } DCHECK(existing_url); if (i == default_search_index && !*default_search_provider) *default_search_provider = existing_url; updated_ids.insert(prepopulated_id); } // Remove any prepopulated engines which are no longer in the master list, as // long as the user hasn't modified them or made them the default engine. for (IDMap::iterator i(id_to_turl.begin()); i != id_to_turl.end(); ++i) { const TemplateURL* template_url = i->second; if ((template_url->safe_for_autoreplace()) && (template_url != *default_search_provider)) { std::vector<TemplateURL*>::iterator i = find(template_urls->begin(), template_urls->end(), template_url); DCHECK(i != template_urls->end()); template_urls->erase(i); if (service) service->RemoveKeyword(*template_url); delete template_url; } } } void GetSearchProvidersUsingKeywordResult( const WDTypedResult& result, WebDataService* service, PrefService* prefs, std::vector<TemplateURL*>* template_urls, const TemplateURL** default_search_provider, int* new_resource_keyword_version) { DCHECK(service == NULL || BrowserThread::CurrentlyOn(BrowserThread::UI)); DCHECK(template_urls); DCHECK(template_urls->empty()); DCHECK(default_search_provider); DCHECK(*default_search_provider == NULL); DCHECK(result.GetType() == KEYWORDS_RESULT); DCHECK(new_resource_keyword_version); *new_resource_keyword_version = 0; WDKeywordsResult keyword_result = reinterpret_cast< const WDResult<WDKeywordsResult>*>(&result)->GetValue(); template_urls->swap(keyword_result.keywords); const int resource_keyword_version = TemplateURLPrepopulateData::GetDataVersion(prefs); if (keyword_result.builtin_keyword_version != resource_keyword_version) { // There should never be duplicate TemplateURLs. We had a bug such that // duplicate TemplateURLs existed for one locale. As such we invoke // RemoveDuplicatePrepopulateIDs to nuke the duplicates. RemoveDuplicatePrepopulateIDs(template_urls, service); } if (keyword_result.default_search_provider_id) { // See if we can find the default search provider. for (std::vector<TemplateURL*>::iterator i = template_urls->begin(); i != template_urls->end(); ++i) { if ((*i)->id() == keyword_result.default_search_provider_id) { *default_search_provider = *i; break; } } } if (keyword_result.builtin_keyword_version != resource_keyword_version) { MergeEnginesFromPrepopulateData(prefs, service, template_urls, default_search_provider); *new_resource_keyword_version = resource_keyword_version; } }