blob: e6ec9c72e57d2b0310b6c0dfaf27baa2af1eb64a [file] [log] [blame]
// Copyright 2014 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 "components/search_engines/util.h"
#include <stddef.h>
#include <stdint.h>
#include <algorithm>
#include <map>
#include <set>
#include <string>
#include <vector>
#include "base/logging.h"
#include "base/memory/ptr_util.h"
#include "base/time/time.h"
#include "components/prefs/pref_service.h"
#include "components/search_engines/template_url.h"
#include "components/search_engines/template_url_prepopulate_data.h"
#include "components/search_engines/template_url_service.h"
base::string16 GetDefaultSearchEngineName(TemplateURLService* service) {
const TemplateURL* const default_provider =
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
return base::string16();
return default_provider->short_name();
GURL GetDefaultSearchURLForSearchTerms(TemplateURLService* service,
const base::string16& terms) {
const TemplateURL* default_provider = service->GetDefaultSearchProvider();
if (!default_provider)
return GURL();
const TemplateURLRef& search_url = default_provider->url_ref();
TemplateURLRef::SearchTermsArgs search_terms_args(terms);
search_terms_args.append_extra_query_params = true;
return GURL(search_url.ReplaceSearchTerms(search_terms_args,
void RemoveDuplicatePrepopulateIDs(
KeywordWebDataService* service,
const std::vector<std::unique_ptr<TemplateURLData>>& prepopulated_urls,
TemplateURL* default_search_provider,
TemplateURLService::OwnedTemplateURLVector* template_urls,
const SearchTermsData& search_terms_data,
std::set<std::string>* removed_keyword_guids) {
// For convenience construct an ID->TemplateURL* map from |prepopulated_urls|.
std::map<int, TemplateURLData*> prepopulated_url_map;
for (const auto& url : prepopulated_urls)
prepopulated_url_map[url->prepopulate_id] = url.get();
// Separate |template_urls| into prepopulated and non-prepopulated groups.
std::multimap<int, std::unique_ptr<TemplateURL>> unchecked_urls;
TemplateURLService::OwnedTemplateURLVector checked_urls;
for (auto& turl : *template_urls) {
int prepopulate_id = turl->prepopulate_id();
if (prepopulate_id)
unchecked_urls.insert(std::make_pair(prepopulate_id, std::move(turl)));
// For each group of prepopulated URLs with one ID, find the best URL to use
// and add it to the (initially all non-prepopulated) URLs we've already OKed.
// Delete the others from the service and from memory.
while (!unchecked_urls.empty()) {
// Find the best URL.
int prepopulate_id = unchecked_urls.begin()->first;
auto prepopulated_url = prepopulated_url_map.find(prepopulate_id);
auto end = unchecked_urls.upper_bound(prepopulate_id);
auto best = unchecked_urls.begin();
bool matched_keyword = false;
for (auto i = unchecked_urls.begin(); i != end; ++i) {
// If the user-selected DSE is a prepopulated engine its properties will
// either come from the prepopulation origin or from the user preferences
// file (see DefaultSearchManager). Those properties will end up
// overwriting whatever we load now anyway. If we are eliminating
// duplicates, then, we err on the side of keeping the thing that looks
// more like the value we will end up with in the end.
if (default_search_provider &&
(default_search_provider->prepopulate_id() ==
i->second->prepopulate_id()) &&
search_terms_data)) {
best = i;
// Otherwise, a URL is best if it matches the prepopulated data's keyword;
// if none match, just fall back to using the one with the lowest ID.
if (matched_keyword)
if ((prepopulated_url != prepopulated_url_map.end()) &&
search_terms_data)) {
best = i;
matched_keyword = true;
} else if (i->second->id() < best->second->id()) {
best = i;
// Add the best URL to the checked group and delete the rest.
for (auto i = unchecked_urls.begin(); i != end; ++i) {
if (i == best)
if (service) {
if (removed_keyword_guids)
// Done with this group.
unchecked_urls.erase(unchecked_urls.begin(), end);
// Return the checked URLs.
// Returns the TemplateURL with id specified from the list of TemplateURLs.
// If not found, returns NULL.
TemplateURL* GetTemplateURLByID(
const TemplateURLService::TemplateURLVector& template_urls,
int64_t id) {
for (TemplateURLService::TemplateURLVector::const_iterator i(
template_urls.begin()); i != template_urls.end(); ++i) {
if ((*i)->id() == id) {
return *i;
return NULL;
TemplateURL* FindURLByPrepopulateID(
const TemplateURLService::TemplateURLVector& template_urls,
int prepopulate_id) {
for (std::vector<TemplateURL*>::const_iterator i = template_urls.begin();
i < template_urls.end(); ++i) {
if ((*i)->prepopulate_id() == prepopulate_id)
return *i;
return NULL;
void MergeIntoPrepopulatedEngineData(const TemplateURL* original_turl,
TemplateURLData* prepopulated_url) {
DCHECK_EQ(original_turl->prepopulate_id(), prepopulated_url->prepopulate_id);
if (!original_turl->safe_for_autoreplace()) {
prepopulated_url->safe_for_autoreplace = false;
prepopulated_url->id = original_turl->id();
prepopulated_url->sync_guid = original_turl->sync_guid();
prepopulated_url->date_created = original_turl->date_created();
prepopulated_url->last_modified = original_turl->last_modified();
ActionsFromPrepopulateData::ActionsFromPrepopulateData() {}
const ActionsFromPrepopulateData& other) = default;
ActionsFromPrepopulateData::~ActionsFromPrepopulateData() {}
// This is invoked when the version of the prepopulate data changes.
// If |removed_keyword_guids| is not NULL, the Sync GUID of each item removed
// from the DB will be added to it. Note that this function will take
// ownership of |prepopulated_urls| and will clear the vector.
void MergeEnginesFromPrepopulateData(
KeywordWebDataService* service,
std::vector<std::unique_ptr<TemplateURLData>>* prepopulated_urls,
size_t default_search_index,
TemplateURLService::OwnedTemplateURLVector* template_urls,
TemplateURL* default_search_provider,
std::set<std::string>* removed_keyword_guids) {
ActionsFromPrepopulateData actions(CreateActionsFromCurrentPrepopulateData(
prepopulated_urls, *template_urls, default_search_provider));
// Remove items.
for (const auto* removed_engine : actions.removed_engines) {
auto j = FindTemplateURL(template_urls, removed_engine);
DCHECK(j != template_urls->end());
DCHECK(!default_search_provider ||
(*j)->prepopulate_id() != default_search_provider->prepopulate_id());
std::unique_ptr<TemplateURL> template_url = std::move(*j);
if (service) {
if (removed_keyword_guids)
// Edit items.
for (const auto& edited_engine : actions.edited_engines) {
const TemplateURLData& data = edited_engine.second;
if (service)
// Replace the entry in |template_urls| with the updated one.
auto j = FindTemplateURL(template_urls, edited_engine.first);
*j = base::MakeUnique<TemplateURL>(data);
// Add items.
for (const auto& added_engine : actions.added_engines)
ActionsFromPrepopulateData CreateActionsFromCurrentPrepopulateData(
std::vector<std::unique_ptr<TemplateURLData>>* prepopulated_urls,
const TemplateURLService::OwnedTemplateURLVector& existing_urls,
const TemplateURL* default_search_provider) {
// Create a map to hold all provided |template_urls| that originally came from
// prepopulate data (i.e. have a non-zero prepopulate_id()).
std::map<int, TemplateURL*> id_to_turl;
for (auto& turl : existing_urls) {
int prepopulate_id = turl->prepopulate_id();
if (prepopulate_id > 0)
id_to_turl[prepopulate_id] = turl.get();
// For each current prepopulated URL, check whether |template_urls| contained
// a matching prepopulated URL. If so, update the passed-in URL to match the
// current data. (If the passed-in URL was user-edited, we persist the user's
// name and keyword.) If not, add the prepopulated URL.
ActionsFromPrepopulateData actions;
for (auto& prepopulated_url : *prepopulated_urls) {
const int prepopulated_id = prepopulated_url->prepopulate_id;
DCHECK_NE(0, prepopulated_id);
auto existing_url_iter = id_to_turl.find(prepopulated_id);
if (existing_url_iter != id_to_turl.end()) {
// Update the data store with the new prepopulated data. Preserve user
// edits to the name and keyword.
TemplateURL* existing_url(existing_url_iter->second);
MergeIntoPrepopulatedEngineData(existing_url, prepopulated_url.get());
// Update last_modified to ensure that if this entry is later merged with
// entries from Sync, the conflict resolution logic knows that this was
// updated and propagates the new values to the server.
prepopulated_url->last_modified = base::Time::Now();
std::make_pair(existing_url, *prepopulated_url));
} else {
// The block above removed all the URLs from the |id_to_turl| map that were
// found in the prepopulate data. Any remaining URLs that haven't been
// user-edited or made default can be removed from the data store.
// We assume that this entry is equivalent to the DSE if its prepopulate ID
// and keyword both match. If the prepopulate ID _does_ match all properties
// will be replaced with those from |default_search_provider| anyway.
for (auto i = id_to_turl.begin(); i != id_to_turl.end(); ++i) {
TemplateURL* template_url = i->second;
if ((template_url->safe_for_autoreplace()) &&
(!default_search_provider ||
(template_url->prepopulate_id() !=
default_search_provider->prepopulate_id()) ||
(template_url->keyword() != default_search_provider->keyword())))
return actions;
void GetSearchProvidersUsingKeywordResult(
const WDTypedResult& result,
KeywordWebDataService* service,
PrefService* prefs,
TemplateURLService::OwnedTemplateURLVector* template_urls,
TemplateURL* default_search_provider,
const SearchTermsData& search_terms_data,
int* new_resource_keyword_version,
std::set<std::string>* removed_keyword_guids) {
WDKeywordsResult keyword_result = reinterpret_cast<
const WDResult<WDKeywordsResult>*>(&result)->GetValue();
for (auto& keyword : keyword_result.keywords) {
// Fix any duplicate encodings in the local database. Note that we don't
// adjust the last_modified time of this keyword; this way, we won't later
// overwrite any changes on the sync server that happened to this keyword
// since the last time we synced. Instead, we also run a de-duping pass on
// the server-provided data in
// TemplateURLService::CreateTemplateURLFromTemplateURLAndSyncData() and
// update the server with the merged, de-duped results at that time. We
// still fix here, though, to correct problems in clients that have disabled
// search engine sync, since in that case that code will never be reached.
if (DeDupeEncodings(&keyword.input_encodings) && service)
*new_resource_keyword_version = keyword_result.builtin_keyword_version;
GetSearchProvidersUsingLoadedEngines(service, prefs, template_urls,
void GetSearchProvidersUsingLoadedEngines(
KeywordWebDataService* service,
PrefService* prefs,
TemplateURLService::OwnedTemplateURLVector* template_urls,
TemplateURL* default_search_provider,
const SearchTermsData& search_terms_data,
int* resource_keyword_version,
std::set<std::string>* removed_keyword_guids) {
size_t default_search_index;
std::vector<std::unique_ptr<TemplateURLData>> prepopulated_urls =
RemoveDuplicatePrepopulateIDs(service, prepopulated_urls,
default_search_provider, template_urls,
search_terms_data, removed_keyword_guids);
const int prepopulate_resource_keyword_version =
if (*resource_keyword_version < prepopulate_resource_keyword_version) {
service, &prepopulated_urls, default_search_index, template_urls,
default_search_provider, removed_keyword_guids);
*resource_keyword_version = prepopulate_resource_keyword_version;
} else {
*resource_keyword_version = 0;
bool DeDupeEncodings(std::vector<std::string>* encodings) {
std::vector<std::string> deduped_encodings;
std::set<std::string> encoding_set;
for (std::vector<std::string>::const_iterator i(encodings->begin());
i != encodings->end(); ++i) {
if (encoding_set.insert(*i).second)
return encodings->size() != deduped_encodings.size();
TemplateURLService::OwnedTemplateURLVector::iterator FindTemplateURL(
TemplateURLService::OwnedTemplateURLVector* urls,
const TemplateURL* url) {
return std::find_if(urls->begin(), urls->end(),
[url](const std::unique_ptr<TemplateURL>& ptr) {
return ptr.get() == url;