|  | // 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 "components/search_engines/template_url_service.h" | 
|  |  | 
|  | #include <stddef.h> | 
|  | #include <utility> | 
|  |  | 
|  | #include "base/bind.h" | 
|  | #include "base/bind_helpers.h" | 
|  | #include "base/callback.h" | 
|  | #include "base/macros.h" | 
|  | #include "base/memory/ref_counted.h" | 
|  | #include "base/memory/scoped_ptr.h" | 
|  | #include "base/run_loop.h" | 
|  | #include "base/strings/string_split.h" | 
|  | #include "base/strings/string_util.h" | 
|  | #include "base/strings/utf_string_conversions.h" | 
|  | #include "base/task/cancelable_task_tracker.h" | 
|  | #include "base/test/simple_test_clock.h" | 
|  | #include "base/threading/thread.h" | 
|  | #include "base/time/time.h" | 
|  | #include "chrome/browser/history/history_service_factory.h" | 
|  | #include "chrome/browser/search_engines/template_url_service_test_util.h" | 
|  | #include "chrome/test/base/testing_profile.h" | 
|  | #include "components/history/core/browser/history_service.h" | 
|  | #include "components/search_engines/keyword_web_data_service.h" | 
|  | #include "components/search_engines/search_host_to_urls_map.h" | 
|  | #include "components/search_engines/search_terms_data.h" | 
|  | #include "components/search_engines/template_url.h" | 
|  | #include "components/search_engines/template_url_prepopulate_data.h" | 
|  | #include "content/public/test/test_browser_thread_bundle.h" | 
|  | #include "testing/gtest/include/gtest/gtest.h" | 
|  |  | 
|  | using base::ASCIIToUTF16; | 
|  | using base::Time; | 
|  | using base::TimeDelta; | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | // QueryHistoryCallbackImpl --------------------------------------------------- | 
|  |  | 
|  | struct QueryHistoryCallbackImpl { | 
|  | QueryHistoryCallbackImpl() : success(false) {} | 
|  |  | 
|  | void Callback(bool success, | 
|  | const history::URLRow& row, | 
|  | const history::VisitVector& visits) { | 
|  | this->success = success; | 
|  | if (success) { | 
|  | this->row = row; | 
|  | this->visits = visits; | 
|  | } | 
|  | } | 
|  |  | 
|  | bool success; | 
|  | history::URLRow row; | 
|  | history::VisitVector visits; | 
|  | }; | 
|  |  | 
|  | TemplateURL* CreateKeywordWithDate( | 
|  | TemplateURLService* model, | 
|  | const std::string& short_name, | 
|  | const std::string& keyword, | 
|  | const std::string& url, | 
|  | const std::string& suggest_url, | 
|  | const std::string& alternate_url, | 
|  | const std::string& favicon_url, | 
|  | bool safe_for_autoreplace, | 
|  | bool show_in_default_list, | 
|  | const std::string& encodings, | 
|  | Time date_created, | 
|  | Time last_modified) { | 
|  | TemplateURLData data; | 
|  | data.SetShortName(base::UTF8ToUTF16(short_name)); | 
|  | data.SetKeyword(base::UTF8ToUTF16(keyword)); | 
|  | data.SetURL(url); | 
|  | data.suggestions_url = suggest_url; | 
|  | if (!alternate_url.empty()) | 
|  | data.alternate_urls.push_back(alternate_url); | 
|  | data.favicon_url = GURL(favicon_url); | 
|  | data.safe_for_autoreplace = safe_for_autoreplace; | 
|  | data.show_in_default_list = show_in_default_list; | 
|  | data.input_encodings = base::SplitString( | 
|  | encodings, ";", base::TRIM_WHITESPACE, base::SPLIT_WANT_ALL); | 
|  | data.date_created = date_created; | 
|  | data.last_modified = last_modified; | 
|  | return new TemplateURL(data); | 
|  | } | 
|  |  | 
|  | TemplateURL* AddKeywordWithDate( | 
|  | TemplateURLService* model, | 
|  | const std::string& short_name, | 
|  | const std::string& keyword, | 
|  | const std::string& url, | 
|  | const std::string& suggest_url, | 
|  | const std::string& alternate_url, | 
|  | const std::string& favicon_url, | 
|  | bool safe_for_autoreplace, | 
|  | const std::string& encodings, | 
|  | Time date_created, | 
|  | Time last_modified) { | 
|  | TemplateURL* t_url = CreateKeywordWithDate( | 
|  | model, short_name, keyword, url, suggest_url, alternate_url,favicon_url, | 
|  | safe_for_autoreplace, false, encodings, date_created, last_modified); | 
|  | model->Add(t_url); | 
|  | EXPECT_NE(0, t_url->id()); | 
|  | return t_url; | 
|  | } | 
|  |  | 
|  | // Checks that the two TemplateURLs are similar. It does not check the id, the | 
|  | // date_created or the last_modified time.  Neither pointer should be NULL. | 
|  | void ExpectSimilar(const TemplateURL* expected, const TemplateURL* actual) { | 
|  | ASSERT_TRUE(expected != NULL); | 
|  | ASSERT_TRUE(actual != NULL); | 
|  | EXPECT_EQ(expected->short_name(), actual->short_name()); | 
|  | EXPECT_EQ(expected->keyword(), actual->keyword()); | 
|  | EXPECT_EQ(expected->url(), actual->url()); | 
|  | EXPECT_EQ(expected->suggestions_url(), actual->suggestions_url()); | 
|  | EXPECT_EQ(expected->favicon_url(), actual->favicon_url()); | 
|  | EXPECT_EQ(expected->alternate_urls(), actual->alternate_urls()); | 
|  | EXPECT_EQ(expected->show_in_default_list(), actual->show_in_default_list()); | 
|  | EXPECT_EQ(expected->safe_for_autoreplace(), actual->safe_for_autoreplace()); | 
|  | EXPECT_EQ(expected->input_encodings(), actual->input_encodings()); | 
|  | EXPECT_EQ(expected->search_terms_replacement_key(), | 
|  | actual->search_terms_replacement_key()); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  |  | 
|  | // TemplateURLServiceTest ----------------------------------------------------- | 
|  |  | 
|  | class TemplateURLServiceTest : public testing::Test { | 
|  | public: | 
|  | TemplateURLServiceTest(); | 
|  |  | 
|  | // testing::Test: | 
|  | void SetUp() override; | 
|  | void TearDown() override; | 
|  |  | 
|  | TemplateURL* AddKeywordWithDate(const std::string& short_name, | 
|  | const std::string& keyword, | 
|  | const std::string& url, | 
|  | const std::string& suggest_url, | 
|  | const std::string& alternate_url, | 
|  | const std::string& favicon_url, | 
|  | bool safe_for_autoreplace, | 
|  | const std::string& encodings, | 
|  | Time date_created, | 
|  | Time last_modified); | 
|  |  | 
|  | // Verifies the two TemplateURLs are equal. | 
|  | void AssertEquals(const TemplateURL& expected, const TemplateURL& actual); | 
|  |  | 
|  | // Verifies the two timestamps are equal, within the expected degree of | 
|  | // precision. | 
|  | void AssertTimesEqual(const base::Time& expected, const base::Time& actual); | 
|  |  | 
|  | // Create an URL that appears to have been prepopulated, but won't be in the | 
|  | // current data. The caller owns the returned TemplateURL*. | 
|  | TemplateURL* CreatePreloadedTemplateURL(bool safe_for_autoreplace, | 
|  | int prepopulate_id); | 
|  |  | 
|  | // Helper methods to make calling TemplateURLServiceTestUtil methods less | 
|  | // visually noisy in the test code. | 
|  | void VerifyObserverCount(int expected_changed_count); | 
|  | void VerifyObserverFired(); | 
|  | TemplateURLServiceTestUtil* test_util() { return test_util_.get(); } | 
|  | TemplateURLService* model() { return test_util_->model(); } | 
|  | const SearchTermsData& search_terms_data() { | 
|  | return model()->search_terms_data(); | 
|  | } | 
|  |  | 
|  | private: | 
|  | content::TestBrowserThreadBundle thread_bundle_;  // To set up BrowserThreads. | 
|  | scoped_ptr<TemplateURLServiceTestUtil> test_util_; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(TemplateURLServiceTest); | 
|  | }; | 
|  |  | 
|  | class TemplateURLServiceWithoutFallbackTest : public TemplateURLServiceTest { | 
|  | public: | 
|  | TemplateURLServiceWithoutFallbackTest() : TemplateURLServiceTest() {} | 
|  |  | 
|  | void SetUp() override { | 
|  | DefaultSearchManager::SetFallbackSearchEnginesDisabledForTesting(true); | 
|  | TemplateURLServiceTest::SetUp(); | 
|  | } | 
|  |  | 
|  | void TearDown() override { | 
|  | TemplateURLServiceTest::TearDown(); | 
|  | DefaultSearchManager::SetFallbackSearchEnginesDisabledForTesting(false); | 
|  | } | 
|  | }; | 
|  |  | 
|  | TemplateURLServiceTest::TemplateURLServiceTest() { | 
|  | } | 
|  |  | 
|  | void TemplateURLServiceTest::SetUp() { | 
|  | test_util_.reset(new TemplateURLServiceTestUtil); | 
|  | } | 
|  |  | 
|  | void TemplateURLServiceTest::TearDown() { | 
|  | test_util_.reset(); | 
|  | } | 
|  |  | 
|  | TemplateURL* TemplateURLServiceTest::AddKeywordWithDate( | 
|  | const std::string& short_name, | 
|  | const std::string& keyword, | 
|  | const std::string& url, | 
|  | const std::string& suggest_url, | 
|  | const std::string& alternate_url, | 
|  | const std::string& favicon_url, | 
|  | bool safe_for_autoreplace, | 
|  | const std::string& encodings, | 
|  | Time date_created, | 
|  | Time last_modified) { | 
|  | return ::AddKeywordWithDate(model(), short_name, keyword, url, suggest_url, | 
|  | alternate_url, favicon_url, safe_for_autoreplace, | 
|  | encodings, date_created, last_modified); | 
|  | } | 
|  |  | 
|  | void TemplateURLServiceTest::AssertEquals(const TemplateURL& expected, | 
|  | const TemplateURL& actual) { | 
|  | ASSERT_EQ(expected.short_name(), actual.short_name()); | 
|  | ASSERT_EQ(expected.keyword(), actual.keyword()); | 
|  | ASSERT_EQ(expected.url(), actual.url()); | 
|  | ASSERT_EQ(expected.suggestions_url(), actual.suggestions_url()); | 
|  | ASSERT_EQ(expected.favicon_url(), actual.favicon_url()); | 
|  | ASSERT_EQ(expected.alternate_urls(), actual.alternate_urls()); | 
|  | ASSERT_EQ(expected.show_in_default_list(), actual.show_in_default_list()); | 
|  | ASSERT_EQ(expected.safe_for_autoreplace(), actual.safe_for_autoreplace()); | 
|  | ASSERT_EQ(expected.input_encodings(), actual.input_encodings()); | 
|  | ASSERT_EQ(expected.id(), actual.id()); | 
|  | ASSERT_EQ(expected.date_created(), actual.date_created()); | 
|  | AssertTimesEqual(expected.last_modified(), actual.last_modified()); | 
|  | ASSERT_EQ(expected.sync_guid(), actual.sync_guid()); | 
|  | ASSERT_EQ(expected.search_terms_replacement_key(), | 
|  | actual.search_terms_replacement_key()); | 
|  | } | 
|  |  | 
|  | void TemplateURLServiceTest::AssertTimesEqual(const base::Time& expected, | 
|  | const base::Time& actual) { | 
|  | // Because times are stored with a granularity of one second, there is a loss | 
|  | // of precision when serializing and deserializing the timestamps. Hence, only | 
|  | // expect timestamps to be equal to within one second of one another. | 
|  | ASSERT_LT((expected - actual).magnitude(), base::TimeDelta::FromSeconds(1)); | 
|  | } | 
|  |  | 
|  | TemplateURL* TemplateURLServiceTest::CreatePreloadedTemplateURL( | 
|  | bool safe_for_autoreplace, | 
|  | int prepopulate_id) { | 
|  | TemplateURLData data; | 
|  | data.SetShortName(ASCIIToUTF16("unittest")); | 
|  | data.SetKeyword(ASCIIToUTF16("unittest")); | 
|  | data.SetURL("http://www.unittest.com/{searchTerms}"); | 
|  | data.favicon_url = GURL("http://favicon.url"); | 
|  | data.show_in_default_list = true; | 
|  | data.safe_for_autoreplace = safe_for_autoreplace; | 
|  | data.input_encodings.push_back("UTF-8"); | 
|  | data.date_created = Time::FromTimeT(100); | 
|  | data.last_modified = Time::FromTimeT(100); | 
|  | data.prepopulate_id = prepopulate_id; | 
|  | return new TemplateURL(data); | 
|  | } | 
|  |  | 
|  | void TemplateURLServiceTest::VerifyObserverCount(int expected_changed_count) { | 
|  | EXPECT_EQ(expected_changed_count, test_util_->GetObserverCount()); | 
|  | test_util_->ResetObserverCount(); | 
|  | } | 
|  |  | 
|  | void TemplateURLServiceTest::VerifyObserverFired() { | 
|  | EXPECT_LE(1, test_util_->GetObserverCount()); | 
|  | test_util_->ResetObserverCount(); | 
|  | } | 
|  |  | 
|  |  | 
|  | // Actual tests --------------------------------------------------------------- | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, Load) { | 
|  | test_util()->VerifyLoad(); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, AddUpdateRemove) { | 
|  | // Add a new TemplateURL. | 
|  | test_util()->VerifyLoad(); | 
|  | const size_t initial_count = model()->GetTemplateURLs().size(); | 
|  |  | 
|  | TemplateURLData data; | 
|  | data.SetShortName(ASCIIToUTF16("google")); | 
|  | data.SetKeyword(ASCIIToUTF16("keyword")); | 
|  | data.SetURL("http://www.google.com/foo/bar"); | 
|  | data.favicon_url = GURL("http://favicon.url"); | 
|  | data.safe_for_autoreplace = true; | 
|  | data.date_created = Time::FromTimeT(100); | 
|  | data.last_modified = Time::FromTimeT(100); | 
|  | data.sync_guid = "00000000-0000-0000-0000-000000000001"; | 
|  | TemplateURL* t_url = new TemplateURL(data); | 
|  | model()->Add(t_url); | 
|  | ASSERT_TRUE(model()->CanAddAutogeneratedKeyword(ASCIIToUTF16("keyword"), | 
|  | GURL(), NULL)); | 
|  | VerifyObserverCount(1); | 
|  | base::RunLoop().RunUntilIdle(); | 
|  | ASSERT_EQ(initial_count + 1, model()->GetTemplateURLs().size()); | 
|  | ASSERT_EQ(t_url, model()->GetTemplateURLForKeyword(t_url->keyword())); | 
|  | // We need to make a second copy as the model takes ownership of |t_url| and | 
|  | // will delete it.  We have to do this after calling Add() since that gives | 
|  | // |t_url| its ID. | 
|  | scoped_ptr<TemplateURL> cloned_url(new TemplateURL(t_url->data())); | 
|  |  | 
|  | // Reload the model to verify it was actually saved to the database. | 
|  | test_util()->ResetModel(true); | 
|  | ASSERT_EQ(initial_count + 1, model()->GetTemplateURLs().size()); | 
|  | TemplateURL* loaded_url = | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("keyword")); | 
|  | ASSERT_TRUE(loaded_url != NULL); | 
|  | AssertEquals(*cloned_url, *loaded_url); | 
|  | ASSERT_TRUE(model()->CanAddAutogeneratedKeyword(ASCIIToUTF16("keyword"), | 
|  | GURL(), NULL)); | 
|  |  | 
|  | // We expect the last_modified time to be updated to the present time on an | 
|  | // explicit reset. | 
|  | base::Time now = base::Time::Now(); | 
|  | scoped_ptr<base::SimpleTestClock> clock(new base::SimpleTestClock); | 
|  | clock->SetNow(now); | 
|  | model()->set_clock(std::move(clock)); | 
|  |  | 
|  | // Mutate an element and verify it succeeded. | 
|  | model()->ResetTemplateURL(loaded_url, ASCIIToUTF16("a"), ASCIIToUTF16("b"), | 
|  | "c"); | 
|  | ASSERT_EQ(ASCIIToUTF16("a"), loaded_url->short_name()); | 
|  | ASSERT_EQ(ASCIIToUTF16("b"), loaded_url->keyword()); | 
|  | ASSERT_EQ("c", loaded_url->url()); | 
|  | ASSERT_FALSE(loaded_url->safe_for_autoreplace()); | 
|  | ASSERT_TRUE(model()->CanAddAutogeneratedKeyword(ASCIIToUTF16("keyword"), | 
|  | GURL(), NULL)); | 
|  | ASSERT_FALSE(model()->CanAddAutogeneratedKeyword(ASCIIToUTF16("b"), GURL(), | 
|  | NULL)); | 
|  | cloned_url.reset(new TemplateURL(loaded_url->data())); | 
|  | base::RunLoop().RunUntilIdle(); | 
|  | test_util()->ResetModel(true); | 
|  | ASSERT_EQ(initial_count + 1, model()->GetTemplateURLs().size()); | 
|  | loaded_url = model()->GetTemplateURLForKeyword(ASCIIToUTF16("b")); | 
|  | ASSERT_TRUE(loaded_url != NULL); | 
|  | AssertEquals(*cloned_url, *loaded_url); | 
|  | // We changed a TemplateURL in the service, so ensure that the time was | 
|  | // updated. | 
|  | AssertTimesEqual(now, loaded_url->last_modified()); | 
|  |  | 
|  | // Remove an element and verify it succeeded. | 
|  | model()->Remove(loaded_url); | 
|  | VerifyObserverCount(1); | 
|  | test_util()->ResetModel(true); | 
|  | ASSERT_EQ(initial_count, model()->GetTemplateURLs().size()); | 
|  | EXPECT_TRUE(model()->GetTemplateURLForKeyword(ASCIIToUTF16("b")) == NULL); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, AddSameKeyword) { | 
|  | test_util()->VerifyLoad(); | 
|  |  | 
|  | AddKeywordWithDate( | 
|  | "first", "keyword", "http://test1", std::string(), std::string(), | 
|  | std::string(), true, "UTF-8", Time(), Time()); | 
|  | VerifyObserverCount(1); | 
|  |  | 
|  | // Test what happens when we try to add a TemplateURL with the same keyword as | 
|  | // one in the model. | 
|  | TemplateURLData data; | 
|  | data.SetShortName(ASCIIToUTF16("second")); | 
|  | data.SetKeyword(ASCIIToUTF16("keyword")); | 
|  | data.SetURL("http://test2"); | 
|  | data.safe_for_autoreplace = false; | 
|  | TemplateURL* t_url = new TemplateURL(data); | 
|  | model()->Add(t_url); | 
|  |  | 
|  | // Because the old TemplateURL was replaceable and the new one wasn't, the new | 
|  | // one should have replaced the old. | 
|  | VerifyObserverCount(1); | 
|  | EXPECT_EQ(t_url, model()->GetTemplateURLForKeyword(ASCIIToUTF16("keyword"))); | 
|  | EXPECT_EQ(ASCIIToUTF16("second"), t_url->short_name()); | 
|  | EXPECT_EQ(ASCIIToUTF16("keyword"), t_url->keyword()); | 
|  | EXPECT_FALSE(t_url->safe_for_autoreplace()); | 
|  |  | 
|  | // Now try adding a replaceable TemplateURL.  This should just delete the | 
|  | // passed-in URL. | 
|  | data.SetShortName(ASCIIToUTF16("third")); | 
|  | data.SetURL("http://test3"); | 
|  | data.safe_for_autoreplace = true; | 
|  | model()->Add(new TemplateURL(data)); | 
|  | VerifyObserverCount(0); | 
|  | EXPECT_EQ(t_url, model()->GetTemplateURLForKeyword(ASCIIToUTF16("keyword"))); | 
|  | EXPECT_EQ(ASCIIToUTF16("second"), t_url->short_name()); | 
|  | EXPECT_EQ(ASCIIToUTF16("keyword"), t_url->keyword()); | 
|  | EXPECT_FALSE(t_url->safe_for_autoreplace()); | 
|  |  | 
|  | // Now try adding a non-replaceable TemplateURL again.  This should uniquify | 
|  | // the existing entry's keyword. | 
|  | data.SetShortName(ASCIIToUTF16("fourth")); | 
|  | data.SetURL("http://test4"); | 
|  | data.safe_for_autoreplace = false; | 
|  | TemplateURL* t_url2 = new TemplateURL(data); | 
|  | model()->Add(t_url2); | 
|  | VerifyObserverCount(1); | 
|  | EXPECT_EQ(t_url2, model()->GetTemplateURLForKeyword(ASCIIToUTF16("keyword"))); | 
|  | EXPECT_EQ(ASCIIToUTF16("fourth"), t_url2->short_name()); | 
|  | EXPECT_EQ(ASCIIToUTF16("keyword"), t_url2->keyword()); | 
|  | EXPECT_EQ(ASCIIToUTF16("second"), t_url->short_name()); | 
|  | EXPECT_EQ(ASCIIToUTF16("test2"), t_url->keyword()); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, AddExtensionKeyword) { | 
|  | test_util()->VerifyLoad(); | 
|  |  | 
|  | AddKeywordWithDate( | 
|  | "replaceable", "keyword1", "http://test1", std::string(), std::string(), | 
|  | std::string(), true, "UTF-8", Time(), Time()); | 
|  | TemplateURL* original2 = AddKeywordWithDate( | 
|  | "nonreplaceable", "keyword2", "http://test2", std::string(), | 
|  | std::string(), std::string(), false, "UTF-8", Time(), Time()); | 
|  | model()->RegisterOmniboxKeyword("test3", "extension", "keyword3", | 
|  | "http://test3"); | 
|  | TemplateURL* original3 = | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("keyword3")); | 
|  | ASSERT_TRUE(original3); | 
|  |  | 
|  | // Extension keywords should override replaceable keywords. | 
|  | model()->RegisterOmniboxKeyword("id1", "test", "keyword1", "http://test4"); | 
|  | TemplateURL* extension1 = model()->FindTemplateURLForExtension( | 
|  | "id1", TemplateURL::OMNIBOX_API_EXTENSION); | 
|  | EXPECT_TRUE(extension1); | 
|  | EXPECT_EQ(extension1, | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("keyword1"))); | 
|  |  | 
|  | // They should not override non-replaceable keywords. | 
|  | model()->RegisterOmniboxKeyword("id2", "test", "keyword2", "http://test5"); | 
|  | TemplateURL* extension2 = model()->FindTemplateURLForExtension( | 
|  | "id2", TemplateURL::OMNIBOX_API_EXTENSION); | 
|  | ASSERT_TRUE(extension2); | 
|  | EXPECT_EQ(original2, | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("keyword2"))); | 
|  |  | 
|  | // They should override extension keywords added earlier. | 
|  | model()->RegisterOmniboxKeyword("id3", "test", "keyword3", "http://test6"); | 
|  | TemplateURL* extension3 = model()->FindTemplateURLForExtension( | 
|  | "id3", TemplateURL::OMNIBOX_API_EXTENSION); | 
|  | ASSERT_TRUE(extension3); | 
|  | EXPECT_EQ(extension3, | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("keyword3"))); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, AddSameKeywordWithExtensionPresent) { | 
|  | test_util()->VerifyLoad(); | 
|  |  | 
|  | // Similar to the AddSameKeyword test, but with an extension keyword masking a | 
|  | // replaceable TemplateURL.  We should still do correct conflict resolution | 
|  | // between the non-template URLs. | 
|  | model()->RegisterOmniboxKeyword("test2", "extension", "keyword", | 
|  | "http://test2"); | 
|  | TemplateURL* extension = | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("keyword")); | 
|  | ASSERT_TRUE(extension); | 
|  | // Adding a keyword that matches the extension. | 
|  | AddKeywordWithDate( | 
|  | "replaceable", "keyword", "http://test1", std::string(),  std::string(), | 
|  | std::string(), true, "UTF-8", Time(), Time()); | 
|  |  | 
|  | // Adding another replaceable keyword should remove the existing one, but | 
|  | // leave the extension as is. | 
|  | TemplateURLData data; | 
|  | data.SetShortName(ASCIIToUTF16("name1")); | 
|  | data.SetKeyword(ASCIIToUTF16("keyword")); | 
|  | data.SetURL("http://test3"); | 
|  | data.safe_for_autoreplace = true; | 
|  | TemplateURL* t_url = new TemplateURL(data); | 
|  | model()->Add(t_url); | 
|  | EXPECT_EQ(extension, | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("keyword"))); | 
|  | EXPECT_EQ(t_url, model()->GetTemplateURLForHost("test3")); | 
|  |  | 
|  | // Adding a nonreplaceable keyword should remove the existing replaceable | 
|  | // keyword and replace the extension as the associated URL for this keyword, | 
|  | // but not evict the extension from the service entirely. | 
|  | data.SetShortName(ASCIIToUTF16("name2")); | 
|  | data.SetURL("http://test4"); | 
|  | data.safe_for_autoreplace = false; | 
|  | TemplateURL* t_url2 = new TemplateURL(data); | 
|  | model()->Add(t_url2); | 
|  | EXPECT_EQ(t_url2, | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("keyword"))); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, NotPersistOmniboxExtensionKeyword) { | 
|  | test_util()->VerifyLoad(); | 
|  |  | 
|  | // Register an omnibox keyword. | 
|  | model()->RegisterOmniboxKeyword("test", "extension", "keyword", | 
|  | "chrome-extension://test"); | 
|  | ASSERT_TRUE(model()->GetTemplateURLForKeyword(ASCIIToUTF16("keyword"))); | 
|  |  | 
|  | // Reload the data. | 
|  | test_util()->ResetModel(true); | 
|  |  | 
|  | // Ensure the omnibox keyword is not persisted. | 
|  | ASSERT_FALSE(model()->GetTemplateURLForKeyword(ASCIIToUTF16("keyword"))); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, ClearBrowsingData_Keywords) { | 
|  | Time now = Time::Now(); | 
|  | TimeDelta one_day = TimeDelta::FromDays(1); | 
|  | Time month_ago = now - TimeDelta::FromDays(30); | 
|  |  | 
|  | // Nothing has been added. | 
|  | EXPECT_EQ(0U, model()->GetTemplateURLs().size()); | 
|  |  | 
|  | // Create one with a 0 time. | 
|  | AddKeywordWithDate("name1", "key1", "http://foo1", "http://suggest1", | 
|  | std::string(), "http://icon1", true, "UTF-8;UTF-16", | 
|  | Time(), Time()); | 
|  | // Create one for now and +/- 1 day. | 
|  | AddKeywordWithDate("name2", "key2", "http://foo2", "http://suggest2", | 
|  | std::string(),  "http://icon2", true, "UTF-8;UTF-16", | 
|  | now - one_day, Time()); | 
|  | AddKeywordWithDate("name3", "key3", "http://foo3", std::string(), | 
|  | std::string(), std::string(), true, std::string(), now, | 
|  | Time()); | 
|  | AddKeywordWithDate("name4", "key4", "http://foo4", std::string(), | 
|  | std::string(), std::string(), true, std::string(), | 
|  | now + one_day, Time()); | 
|  | // Try the other three states. | 
|  | AddKeywordWithDate("name5", "key5", "http://foo5", "http://suggest5", | 
|  | std::string(), "http://icon5", false, "UTF-8;UTF-16", now, | 
|  | Time()); | 
|  | AddKeywordWithDate("name6", "key6", "http://foo6", "http://suggest6", | 
|  | std::string(), "http://icon6", false, "UTF-8;UTF-16", | 
|  | month_ago, Time()); | 
|  |  | 
|  | // We just added a few items, validate them. | 
|  | EXPECT_EQ(6U, model()->GetTemplateURLs().size()); | 
|  |  | 
|  | // Try removing from current timestamp. This should delete the one in the | 
|  | // future and one very recent one. | 
|  | model()->RemoveAutoGeneratedSince(now); | 
|  | EXPECT_EQ(4U, model()->GetTemplateURLs().size()); | 
|  |  | 
|  | // Try removing from two months ago. This should only delete items that are | 
|  | // auto-generated. | 
|  | model()->RemoveAutoGeneratedBetween(now - TimeDelta::FromDays(60), now); | 
|  | EXPECT_EQ(3U, model()->GetTemplateURLs().size()); | 
|  |  | 
|  | // Make sure the right values remain. | 
|  | EXPECT_EQ(ASCIIToUTF16("key1"), model()->GetTemplateURLs()[0]->keyword()); | 
|  | EXPECT_TRUE(model()->GetTemplateURLs()[0]->safe_for_autoreplace()); | 
|  | EXPECT_EQ(0U, | 
|  | model()->GetTemplateURLs()[0]->date_created().ToInternalValue()); | 
|  |  | 
|  | EXPECT_EQ(ASCIIToUTF16("key5"), model()->GetTemplateURLs()[1]->keyword()); | 
|  | EXPECT_FALSE(model()->GetTemplateURLs()[1]->safe_for_autoreplace()); | 
|  | EXPECT_EQ(now.ToInternalValue(), | 
|  | model()->GetTemplateURLs()[1]->date_created().ToInternalValue()); | 
|  |  | 
|  | EXPECT_EQ(ASCIIToUTF16("key6"), model()->GetTemplateURLs()[2]->keyword()); | 
|  | EXPECT_FALSE(model()->GetTemplateURLs()[2]->safe_for_autoreplace()); | 
|  | EXPECT_EQ(month_ago.ToInternalValue(), | 
|  | model()->GetTemplateURLs()[2]->date_created().ToInternalValue()); | 
|  |  | 
|  | // Try removing from Time=0. This should delete one more. | 
|  | model()->RemoveAutoGeneratedSince(Time()); | 
|  | EXPECT_EQ(2U, model()->GetTemplateURLs().size()); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, ClearBrowsingData_KeywordsForOrigin) { | 
|  | Time now = Time::Now(); | 
|  | TimeDelta one_day = TimeDelta::FromDays(1); | 
|  | Time month_ago = now - TimeDelta::FromDays(30); | 
|  |  | 
|  | // Nothing has been added. | 
|  | EXPECT_EQ(0U, model()->GetTemplateURLs().size()); | 
|  |  | 
|  | // Create one for now and +/- 1 day. | 
|  | AddKeywordWithDate("name1", "key1", "http://foo1", "http://suggest1", | 
|  | std::string(), "http://icon2", true, "UTF-8;UTF-16", | 
|  | now - one_day, Time()); | 
|  | AddKeywordWithDate("name2", "key2", "http://foo2", std::string(), | 
|  | std::string(), std::string(), true, std::string(), now, | 
|  | Time()); | 
|  | AddKeywordWithDate("name3", "key3", "http://foo3", std::string(), | 
|  | std::string(), std::string(), true, std::string(), | 
|  | now + one_day, Time()); | 
|  |  | 
|  | // We just added a few items, validate them. | 
|  | EXPECT_EQ(3U, model()->GetTemplateURLs().size()); | 
|  |  | 
|  | // Try removing foo2. This should delete foo2, but leave foo1 and 3 untouched. | 
|  | model()->RemoveAutoGeneratedForOriginBetween(GURL("http://foo2"), month_ago, | 
|  | now + one_day); | 
|  | EXPECT_EQ(2U, model()->GetTemplateURLs().size()); | 
|  | EXPECT_EQ(ASCIIToUTF16("key1"), model()->GetTemplateURLs()[0]->keyword()); | 
|  | EXPECT_TRUE(model()->GetTemplateURLs()[0]->safe_for_autoreplace()); | 
|  | EXPECT_EQ(ASCIIToUTF16("key3"), model()->GetTemplateURLs()[1]->keyword()); | 
|  | EXPECT_TRUE(model()->GetTemplateURLs()[1]->safe_for_autoreplace()); | 
|  |  | 
|  | // Try removing foo1, but outside the range in which it was modified. It | 
|  | // should remain untouched. | 
|  | model()->RemoveAutoGeneratedForOriginBetween(GURL("http://foo1"), now, | 
|  | now + one_day); | 
|  | EXPECT_EQ(2U, model()->GetTemplateURLs().size()); | 
|  | EXPECT_EQ(ASCIIToUTF16("key1"), model()->GetTemplateURLs()[0]->keyword()); | 
|  | EXPECT_TRUE(model()->GetTemplateURLs()[0]->safe_for_autoreplace()); | 
|  | EXPECT_EQ(ASCIIToUTF16("key3"), model()->GetTemplateURLs()[1]->keyword()); | 
|  | EXPECT_TRUE(model()->GetTemplateURLs()[1]->safe_for_autoreplace()); | 
|  |  | 
|  |  | 
|  | // Try removing foo3. This should delete foo3, but leave foo1 untouched. | 
|  | model()->RemoveAutoGeneratedForOriginBetween(GURL("http://foo3"), month_ago, | 
|  | now + one_day + one_day); | 
|  | EXPECT_EQ(1U, model()->GetTemplateURLs().size()); | 
|  | EXPECT_EQ(ASCIIToUTF16("key1"), model()->GetTemplateURLs()[0]->keyword()); | 
|  | EXPECT_TRUE(model()->GetTemplateURLs()[0]->safe_for_autoreplace()); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, Reset) { | 
|  | // Add a new TemplateURL. | 
|  | test_util()->VerifyLoad(); | 
|  | const size_t initial_count = model()->GetTemplateURLs().size(); | 
|  | TemplateURLData data; | 
|  | data.SetShortName(ASCIIToUTF16("google")); | 
|  | data.SetKeyword(ASCIIToUTF16("keyword")); | 
|  | data.SetURL("http://www.google.com/foo/bar"); | 
|  | data.favicon_url = GURL("http://favicon.url"); | 
|  | data.date_created = Time::FromTimeT(100); | 
|  | data.last_modified = Time::FromTimeT(100); | 
|  | TemplateURL* t_url = new TemplateURL(data); | 
|  | model()->Add(t_url); | 
|  |  | 
|  | VerifyObserverCount(1); | 
|  | base::RunLoop().RunUntilIdle(); | 
|  |  | 
|  | base::Time now = base::Time::Now(); | 
|  | scoped_ptr<base::SimpleTestClock> clock(new base::SimpleTestClock); | 
|  | clock->SetNow(now); | 
|  | model()->set_clock(std::move(clock)); | 
|  |  | 
|  | // Reset the short name, keyword, url and make sure it takes. | 
|  | const base::string16 new_short_name(ASCIIToUTF16("a")); | 
|  | const base::string16 new_keyword(ASCIIToUTF16("b")); | 
|  | const std::string new_url("c"); | 
|  | model()->ResetTemplateURL(t_url, new_short_name, new_keyword, new_url); | 
|  | ASSERT_EQ(new_short_name, t_url->short_name()); | 
|  | ASSERT_EQ(new_keyword, t_url->keyword()); | 
|  | ASSERT_EQ(new_url, t_url->url()); | 
|  |  | 
|  | // Make sure the mappings in the model were updated. | 
|  | ASSERT_EQ(t_url, model()->GetTemplateURLForKeyword(new_keyword)); | 
|  | ASSERT_TRUE( | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("keyword")) == NULL); | 
|  |  | 
|  | scoped_ptr<TemplateURL> cloned_url(new TemplateURL(t_url->data())); | 
|  |  | 
|  | // Reload the model from the database and make sure the change took. | 
|  | test_util()->ResetModel(true); | 
|  | EXPECT_EQ(initial_count + 1, model()->GetTemplateURLs().size()); | 
|  | const TemplateURL* read_url = model()->GetTemplateURLForKeyword(new_keyword); | 
|  | ASSERT_TRUE(read_url); | 
|  | AssertEquals(*cloned_url, *read_url); | 
|  | AssertTimesEqual(now, read_url->last_modified()); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, DefaultSearchProvider) { | 
|  | // Add a new TemplateURL. | 
|  | test_util()->VerifyLoad(); | 
|  | const size_t initial_count = model()->GetTemplateURLs().size(); | 
|  | TemplateURL* t_url = AddKeywordWithDate( | 
|  | "name1", "key1", "http://foo1/{searchTerms}", "http://sugg1", | 
|  | std::string(), "http://icon1", true, "UTF-8;UTF-16", Time(), Time()); | 
|  | test_util()->ResetObserverCount(); | 
|  |  | 
|  | model()->SetUserSelectedDefaultSearchProvider(t_url); | 
|  | ASSERT_EQ(t_url, model()->GetDefaultSearchProvider()); | 
|  | ASSERT_TRUE(t_url->safe_for_autoreplace()); | 
|  | ASSERT_TRUE(t_url->show_in_default_list()); | 
|  |  | 
|  | // Setting the default search provider should have caused notification. | 
|  | VerifyObserverCount(1); | 
|  | base::RunLoop().RunUntilIdle(); | 
|  |  | 
|  | scoped_ptr<TemplateURL> cloned_url(new TemplateURL(t_url->data())); | 
|  |  | 
|  | // Make sure when we reload we get a default search provider. | 
|  | test_util()->ResetModel(true); | 
|  | EXPECT_EQ(initial_count + 1, model()->GetTemplateURLs().size()); | 
|  | ASSERT_TRUE(model()->GetDefaultSearchProvider()); | 
|  | AssertEquals(*cloned_url, *model()->GetDefaultSearchProvider()); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, CantReplaceWithSameKeyword) { | 
|  | test_util()->ChangeModelToLoadState(); | 
|  | ASSERT_TRUE(model()->CanAddAutogeneratedKeyword(ASCIIToUTF16("foo"), GURL(), | 
|  | NULL)); | 
|  | TemplateURL* t_url = AddKeywordWithDate( | 
|  | "name1", "foo", "http://foo1", "http://sugg1", std::string(), | 
|  | "http://icon1", true, "UTF-8;UTF-16", Time(), Time()); | 
|  |  | 
|  | // Can still replace, newly added template url is marked safe to replace. | 
|  | ASSERT_TRUE(model()->CanAddAutogeneratedKeyword(ASCIIToUTF16("foo"), | 
|  | GURL("http://foo2"), NULL)); | 
|  |  | 
|  | // ResetTemplateURL marks the TemplateURL as unsafe to replace, so it should | 
|  | // no longer be replaceable. | 
|  | model()->ResetTemplateURL(t_url, t_url->short_name(), t_url->keyword(), | 
|  | t_url->url()); | 
|  |  | 
|  | ASSERT_FALSE(model()->CanAddAutogeneratedKeyword(ASCIIToUTF16("foo"), | 
|  | GURL("http://foo2"), NULL)); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, CantReplaceWithSameHosts) { | 
|  | test_util()->ChangeModelToLoadState(); | 
|  | ASSERT_TRUE(model()->CanAddAutogeneratedKeyword(ASCIIToUTF16("foo"), | 
|  | GURL("http://foo.com"), NULL)); | 
|  | TemplateURL* t_url = AddKeywordWithDate( | 
|  | "name1", "foo", "http://foo.com", "http://sugg1", std::string(), | 
|  | "http://icon1", true, "UTF-8;UTF-16", Time(), Time()); | 
|  |  | 
|  | // Can still replace, newly added template url is marked safe to replace. | 
|  | ASSERT_TRUE(model()->CanAddAutogeneratedKeyword(ASCIIToUTF16("bar"), | 
|  | GURL("http://foo.com"), NULL)); | 
|  |  | 
|  | // ResetTemplateURL marks the TemplateURL as unsafe to replace, so it should | 
|  | // no longer be replaceable. | 
|  | model()->ResetTemplateURL(t_url, t_url->short_name(), t_url->keyword(), | 
|  | t_url->url()); | 
|  |  | 
|  | ASSERT_FALSE(model()->CanAddAutogeneratedKeyword(ASCIIToUTF16("bar"), | 
|  | GURL("http://foo.com"), NULL)); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, HasDefaultSearchProvider) { | 
|  | // We should have a default search provider even if we haven't loaded. | 
|  | ASSERT_TRUE(model()->GetDefaultSearchProvider()); | 
|  |  | 
|  | // Now force the model to load and make sure we still have a default. | 
|  | test_util()->VerifyLoad(); | 
|  |  | 
|  | ASSERT_TRUE(model()->GetDefaultSearchProvider()); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, DefaultSearchProviderLoadedFromPrefs) { | 
|  | test_util()->VerifyLoad(); | 
|  |  | 
|  | TemplateURLData data; | 
|  | data.SetShortName(ASCIIToUTF16("a")); | 
|  | data.safe_for_autoreplace = true; | 
|  | data.SetURL("http://url/{searchTerms}"); | 
|  | data.suggestions_url = "http://url2"; | 
|  | data.instant_url = "http://instant"; | 
|  | data.date_created = Time::FromTimeT(100); | 
|  | data.last_modified = Time::FromTimeT(100); | 
|  | TemplateURL* t_url = new TemplateURL(data); | 
|  | model()->Add(t_url); | 
|  | const TemplateURLID id = t_url->id(); | 
|  |  | 
|  | model()->SetUserSelectedDefaultSearchProvider(t_url); | 
|  | base::RunLoop().RunUntilIdle(); | 
|  | scoped_ptr<TemplateURL> cloned_url(new TemplateURL(t_url->data())); | 
|  |  | 
|  | // Reset the model and don't load it. The template url we set as the default | 
|  | // should be pulled from prefs now. | 
|  | test_util()->ResetModel(false); | 
|  |  | 
|  | // NOTE: This doesn't use AssertEquals as only a subset of the TemplateURLs | 
|  | // value are persisted to prefs. | 
|  | const TemplateURL* default_turl = model()->GetDefaultSearchProvider(); | 
|  | ASSERT_TRUE(default_turl); | 
|  | EXPECT_EQ(ASCIIToUTF16("a"), default_turl->short_name()); | 
|  | EXPECT_EQ("http://url/{searchTerms}", default_turl->url()); | 
|  | EXPECT_EQ("http://url2", default_turl->suggestions_url()); | 
|  | EXPECT_EQ("http://instant", default_turl->instant_url()); | 
|  | EXPECT_EQ(id, default_turl->id()); | 
|  |  | 
|  | // Now do a load and make sure the default search provider really takes. | 
|  | test_util()->VerifyLoad(); | 
|  |  | 
|  | ASSERT_TRUE(model()->GetDefaultSearchProvider()); | 
|  | AssertEquals(*cloned_url, *model()->GetDefaultSearchProvider()); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, RepairPrepopulatedSearchEngines) { | 
|  | test_util()->VerifyLoad(); | 
|  |  | 
|  | // Edit Google search engine. | 
|  | TemplateURL* google = model()->GetTemplateURLForKeyword( | 
|  | ASCIIToUTF16("google.com")); | 
|  | ASSERT_TRUE(google); | 
|  | model()->ResetTemplateURL(google, ASCIIToUTF16("trash"), ASCIIToUTF16("xxx"), | 
|  | "http://www.foo.com/s?q={searchTerms}"); | 
|  | EXPECT_EQ(ASCIIToUTF16("trash"), google->short_name()); | 
|  | EXPECT_EQ(ASCIIToUTF16("xxx"), google->keyword()); | 
|  |  | 
|  | // Add third-party default search engine. | 
|  | TemplateURL* user_dse = AddKeywordWithDate( | 
|  | "malware", "google.com", "http://www.goo.com/s?q={searchTerms}", | 
|  | std::string(), std::string(), std::string(), | 
|  | true, "UTF-8", Time(), Time()); | 
|  | model()->SetUserSelectedDefaultSearchProvider(user_dse); | 
|  | EXPECT_EQ(user_dse, model()->GetDefaultSearchProvider()); | 
|  |  | 
|  | // Remove bing. | 
|  | TemplateURL* bing = model()->GetTemplateURLForKeyword( | 
|  | ASCIIToUTF16("bing.com")); | 
|  | ASSERT_TRUE(bing); | 
|  | model()->Remove(bing); | 
|  | EXPECT_FALSE(model()->GetTemplateURLForKeyword(ASCIIToUTF16("bing.com"))); | 
|  |  | 
|  | // Register an extension with bing keyword. | 
|  | model()->RegisterOmniboxKeyword("abcdefg", "extension_name", "bing.com", | 
|  | "http://abcdefg"); | 
|  | EXPECT_TRUE(model()->GetTemplateURLForKeyword(ASCIIToUTF16("bing.com"))); | 
|  |  | 
|  | model()->RepairPrepopulatedSearchEngines(); | 
|  |  | 
|  | // Google is default. | 
|  | ASSERT_EQ(google, model()->GetDefaultSearchProvider()); | 
|  | // The keyword wasn't reverted. | 
|  | EXPECT_EQ(ASCIIToUTF16("trash"), google->short_name()); | 
|  | EXPECT_EQ("www.google.com", | 
|  | google->GenerateSearchURL(model()->search_terms_data()).host()); | 
|  |  | 
|  | // Bing was repaired. | 
|  | bing = model()->GetTemplateURLForKeyword(ASCIIToUTF16("bing.com")); | 
|  | ASSERT_TRUE(bing); | 
|  | EXPECT_EQ(TemplateURL::NORMAL, bing->GetType()); | 
|  |  | 
|  | // User search engine is preserved. | 
|  | EXPECT_EQ(user_dse, model()->GetTemplateURLForHost("www.goo.com")); | 
|  | EXPECT_EQ(ASCIIToUTF16("google.com"), user_dse->keyword()); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, RepairSearchEnginesWithManagedDefault) { | 
|  | // Set a managed preference that establishes a default search provider. | 
|  | const char kName[] = "test1"; | 
|  | const char kKeyword[] = "test.com"; | 
|  | const char kSearchURL[] = "http://test.com/search?t={searchTerms}"; | 
|  | const char kIconURL[] = "http://test.com/icon.jpg"; | 
|  | const char kEncodings[] = "UTF-16;UTF-32"; | 
|  | const char kAlternateURL[] = "http://test.com/search#t={searchTerms}"; | 
|  | const char kSearchTermsReplacementKey[] = "espv"; | 
|  | test_util()->SetManagedDefaultSearchPreferences(true, kName, kKeyword, | 
|  | kSearchURL, std::string(), | 
|  | kIconURL, kEncodings, | 
|  | kAlternateURL, | 
|  | kSearchTermsReplacementKey); | 
|  | test_util()->VerifyLoad(); | 
|  | // Verify that the default manager we are getting is the managed one. | 
|  | TemplateURLData data; | 
|  | data.SetShortName(ASCIIToUTF16(kName)); | 
|  | data.SetKeyword(ASCIIToUTF16(kKeyword)); | 
|  | data.SetURL(kSearchURL); | 
|  | data.favicon_url = GURL(kIconURL); | 
|  | data.show_in_default_list = true; | 
|  | data.input_encodings = base::SplitString( | 
|  | kEncodings, ";", base::TRIM_WHITESPACE, base::SPLIT_WANT_ALL); | 
|  | data.alternate_urls.push_back(kAlternateURL); | 
|  | data.search_terms_replacement_key = kSearchTermsReplacementKey; | 
|  | scoped_ptr<TemplateURL> expected_managed_default(new TemplateURL(data)); | 
|  | EXPECT_TRUE(model()->is_default_search_managed()); | 
|  | const TemplateURL* actual_managed_default = | 
|  | model()->GetDefaultSearchProvider(); | 
|  | ExpectSimilar(expected_managed_default.get(), actual_managed_default); | 
|  |  | 
|  | // The following call has no effect on the managed search engine. | 
|  | model()->RepairPrepopulatedSearchEngines(); | 
|  |  | 
|  | EXPECT_TRUE(model()->is_default_search_managed()); | 
|  | actual_managed_default = model()->GetDefaultSearchProvider(); | 
|  | ExpectSimilar(expected_managed_default.get(), actual_managed_default); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, UpdateKeywordSearchTermsForURL) { | 
|  | struct TestData { | 
|  | const std::string url; | 
|  | const base::string16 term; | 
|  | } data[] = { | 
|  | { "http://foo/", base::string16() }, | 
|  | { "http://foo/foo?q=xx", base::string16() }, | 
|  | { "http://x/bar?q=xx", base::string16() }, | 
|  | { "http://x/foo?y=xx", base::string16() }, | 
|  | { "http://x/foo?q=xx", ASCIIToUTF16("xx") }, | 
|  | { "http://x/foo?a=b&q=xx", ASCIIToUTF16("xx") }, | 
|  | { "http://x/foo?q=b&q=xx", base::string16() }, | 
|  | { "http://x/foo#query=xx", ASCIIToUTF16("xx") }, | 
|  | { "http://x/foo?q=b#query=xx", ASCIIToUTF16("xx") }, | 
|  | { "http://x/foo?q=b#q=xx", ASCIIToUTF16("b") }, | 
|  | { "http://x/foo?query=b#q=xx", base::string16() }, | 
|  | }; | 
|  |  | 
|  | test_util()->ChangeModelToLoadState(); | 
|  | AddKeywordWithDate("name", "x", "http://x/foo?q={searchTerms}", | 
|  | "http://sugg1", "http://x/foo#query={searchTerms}", | 
|  | "http://icon1", false, "UTF-8;UTF-16", Time(), Time()); | 
|  |  | 
|  | for (size_t i = 0; i < arraysize(data); ++i) { | 
|  | TemplateURLService::URLVisitedDetails details = { | 
|  | GURL(data[i].url), false | 
|  | }; | 
|  | model()->UpdateKeywordSearchTermsForURL(details); | 
|  | EXPECT_EQ(data[i].term, test_util()->GetAndClearSearchTerm()); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, DontUpdateKeywordSearchForNonReplaceable) { | 
|  | struct TestData { | 
|  | const std::string url; | 
|  | } data[] = { | 
|  | { "http://foo/" }, | 
|  | { "http://x/bar?q=xx" }, | 
|  | { "http://x/foo?y=xx" }, | 
|  | }; | 
|  |  | 
|  | test_util()->ChangeModelToLoadState(); | 
|  | AddKeywordWithDate("name", "x", "http://x/foo", "http://sugg1", std::string(), | 
|  | "http://icon1", false, "UTF-8;UTF-16", Time(), Time()); | 
|  |  | 
|  | for (size_t i = 0; i < arraysize(data); ++i) { | 
|  | TemplateURLService::URLVisitedDetails details = { | 
|  | GURL(data[i].url), false | 
|  | }; | 
|  | model()->UpdateKeywordSearchTermsForURL(details); | 
|  | ASSERT_EQ(base::string16(), test_util()->GetAndClearSearchTerm()); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceWithoutFallbackTest, ChangeGoogleBaseValue) { | 
|  | // NOTE: Do not load the prepopulate data, which also has a {google:baseURL} | 
|  | // keyword in it and would confuse this test. | 
|  | test_util()->ChangeModelToLoadState(); | 
|  |  | 
|  | test_util()->SetGoogleBaseURL(GURL("http://google.com/")); | 
|  | const TemplateURL* t_url = AddKeywordWithDate( | 
|  | "name", "google.com", "{google:baseURL}?q={searchTerms}", "http://sugg1", | 
|  | std::string(), "http://icon1", false, "UTF-8;UTF-16", Time(), Time()); | 
|  | ASSERT_EQ(t_url, model()->GetTemplateURLForHost("google.com")); | 
|  | EXPECT_EQ("google.com", t_url->url_ref().GetHost(search_terms_data())); | 
|  | EXPECT_EQ(ASCIIToUTF16("google.com"), t_url->keyword()); | 
|  |  | 
|  | // Change the Google base url. | 
|  | test_util()->ResetObserverCount(); | 
|  | test_util()->SetGoogleBaseURL(GURL("http://google.co.uk/")); | 
|  | VerifyObserverCount(1); | 
|  |  | 
|  | // Make sure the host->TemplateURL map was updated appropriately. | 
|  | ASSERT_EQ(t_url, model()->GetTemplateURLForHost("google.co.uk")); | 
|  | EXPECT_TRUE(model()->GetTemplateURLForHost("google.com") == NULL); | 
|  | EXPECT_EQ("google.co.uk", t_url->url_ref().GetHost(search_terms_data())); | 
|  | EXPECT_EQ(ASCIIToUTF16("google.co.uk"), t_url->keyword()); | 
|  | EXPECT_EQ("http://google.co.uk/?q=x", t_url->url_ref().ReplaceSearchTerms( | 
|  | TemplateURLRef::SearchTermsArgs(ASCIIToUTF16("x")), search_terms_data())); | 
|  |  | 
|  | // Now add a manual entry and then change the Google base URL such that the | 
|  | // autogenerated Google search keyword would conflict. | 
|  | TemplateURL* manual = AddKeywordWithDate( | 
|  | "manual", "google.de", "http://google.de/search?q={searchTerms}", | 
|  | std::string(), std::string(), std::string(), false, "UTF-8", Time(), | 
|  | Time()); | 
|  | test_util()->SetGoogleBaseURL(GURL("http://google.de")); | 
|  |  | 
|  | // Verify that the manual entry is untouched, and the autogenerated keyword | 
|  | // has not changed. | 
|  | ASSERT_EQ(manual, | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("google.de"))); | 
|  | EXPECT_EQ("google.de", manual->url_ref().GetHost(search_terms_data())); | 
|  | ASSERT_EQ(t_url, | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("google.co.uk"))); | 
|  | EXPECT_EQ("google.de", t_url->url_ref().GetHost(search_terms_data())); | 
|  | EXPECT_EQ(ASCIIToUTF16("google.co.uk"), t_url->keyword()); | 
|  |  | 
|  | // Change the base URL again and verify that the autogenerated keyword follows | 
|  | // even though it didn't match the base URL, while the manual entry is still | 
|  | // untouched. | 
|  | test_util()->SetGoogleBaseURL(GURL("http://google.fr/")); | 
|  | ASSERT_EQ(manual, model()->GetTemplateURLForHost("google.de")); | 
|  | EXPECT_EQ("google.de", manual->url_ref().GetHost(search_terms_data())); | 
|  | EXPECT_EQ(ASCIIToUTF16("google.de"), manual->keyword()); | 
|  | ASSERT_EQ(t_url, model()->GetTemplateURLForHost("google.fr")); | 
|  | EXPECT_TRUE(model()->GetTemplateURLForHost("google.co.uk") == NULL); | 
|  | EXPECT_EQ("google.fr", t_url->url_ref().GetHost(search_terms_data())); | 
|  | EXPECT_EQ(ASCIIToUTF16("google.fr"), t_url->keyword()); | 
|  | } | 
|  |  | 
|  | // Make sure TemplateURLService generates a KEYWORD_GENERATED visit for | 
|  | // KEYWORD visits. | 
|  | TEST_F(TemplateURLServiceTest, GenerateVisitOnKeyword) { | 
|  | test_util()->profile()->CreateBookmarkModel(false); | 
|  | ASSERT_TRUE(test_util()->profile()->CreateHistoryService(true, false)); | 
|  | test_util()->ResetModel(true); | 
|  |  | 
|  | // Create a keyword. | 
|  | TemplateURL* t_url = AddKeywordWithDate( | 
|  | "keyword", "keyword", "http://foo.com/foo?query={searchTerms}", | 
|  | "http://sugg1", std::string(), "http://icon1", true, "UTF-8;UTF-16", | 
|  | base::Time::Now(), base::Time::Now()); | 
|  |  | 
|  | // Add a visit that matches the url of the keyword. | 
|  | history::HistoryService* history = HistoryServiceFactory::GetForProfile( | 
|  | test_util()->profile(), ServiceAccessType::EXPLICIT_ACCESS); | 
|  | history->AddPage( | 
|  | GURL(t_url->url_ref().ReplaceSearchTerms( | 
|  | TemplateURLRef::SearchTermsArgs(ASCIIToUTF16("blah")), | 
|  | search_terms_data())), | 
|  | base::Time::Now(), NULL, 0, GURL(), history::RedirectList(), | 
|  | ui::PAGE_TRANSITION_KEYWORD, history::SOURCE_BROWSED, false); | 
|  |  | 
|  | // Wait for history to finish processing the request. | 
|  | test_util()->profile()->BlockUntilHistoryProcessesPendingRequests(); | 
|  |  | 
|  | // Query history for the generated url. | 
|  | base::CancelableTaskTracker tracker; | 
|  | QueryHistoryCallbackImpl callback; | 
|  | history->QueryURL(GURL("http://keyword"), | 
|  | true, | 
|  | base::Bind(&QueryHistoryCallbackImpl::Callback, | 
|  | base::Unretained(&callback)), | 
|  | &tracker); | 
|  |  | 
|  | // Wait for the request to be processed. | 
|  | test_util()->profile()->BlockUntilHistoryProcessesPendingRequests(); | 
|  |  | 
|  | // And make sure the url and visit were added. | 
|  | EXPECT_TRUE(callback.success); | 
|  | EXPECT_NE(0, callback.row.id()); | 
|  | ASSERT_EQ(1U, callback.visits.size()); | 
|  | EXPECT_EQ(ui::PAGE_TRANSITION_KEYWORD_GENERATED, | 
|  | ui::PageTransitionStripQualifier(callback.visits[0].transition)); | 
|  | } | 
|  |  | 
|  | // Make sure that the load routine deletes prepopulated engines that no longer | 
|  | // exist in the prepopulate data. | 
|  | TEST_F(TemplateURLServiceTest, LoadDeletesUnusedProvider) { | 
|  | // Create a preloaded template url. Add it to a loaded model and wait for the | 
|  | // saves to finish. | 
|  | TemplateURL* t_url = CreatePreloadedTemplateURL(true, 999999); | 
|  | test_util()->ChangeModelToLoadState(); | 
|  | model()->Add(t_url); | 
|  | ASSERT_TRUE( | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("unittest")) != NULL); | 
|  | base::RunLoop().RunUntilIdle(); | 
|  |  | 
|  | // Ensure that merging clears this engine. | 
|  | test_util()->ResetModel(true); | 
|  | ASSERT_TRUE( | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("unittest")) == NULL); | 
|  |  | 
|  | // Wait for any saves to finish. | 
|  | base::RunLoop().RunUntilIdle(); | 
|  |  | 
|  | // Reload the model to verify that the database was updated as a result of the | 
|  | // merge. | 
|  | test_util()->ResetModel(true); | 
|  | ASSERT_TRUE( | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("unittest")) == NULL); | 
|  | } | 
|  |  | 
|  | // Make sure that load routine doesn't delete prepopulated engines that no | 
|  | // longer exist in the prepopulate data if it has been modified by the user. | 
|  | TEST_F(TemplateURLServiceTest, LoadRetainsModifiedProvider) { | 
|  | // Create a preloaded template url and add it to a loaded model. | 
|  | TemplateURL* t_url = CreatePreloadedTemplateURL(false, 999999); | 
|  | test_util()->ChangeModelToLoadState(); | 
|  | model()->Add(t_url); | 
|  |  | 
|  | // Do the copy after t_url is added so that the id is set. | 
|  | scoped_ptr<TemplateURL> cloned_url(new TemplateURL(t_url->data())); | 
|  | ASSERT_EQ(t_url, model()->GetTemplateURLForKeyword(ASCIIToUTF16("unittest"))); | 
|  |  | 
|  | // Wait for any saves to finish. | 
|  | base::RunLoop().RunUntilIdle(); | 
|  |  | 
|  | // Ensure that merging won't clear it if the user has edited it. | 
|  | test_util()->ResetModel(true); | 
|  | const TemplateURL* url_for_unittest = | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("unittest")); | 
|  | ASSERT_TRUE(url_for_unittest != NULL); | 
|  | AssertEquals(*cloned_url, *url_for_unittest); | 
|  |  | 
|  | // Wait for any saves to finish. | 
|  | base::RunLoop().RunUntilIdle(); | 
|  |  | 
|  | // Reload the model to verify that save/reload retains the item. | 
|  | test_util()->ResetModel(true); | 
|  | ASSERT_TRUE( | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("unittest")) != NULL); | 
|  | } | 
|  |  | 
|  | // Make sure that load routine doesn't delete | 
|  | // prepopulated engines that no longer exist in the prepopulate data if | 
|  | // it has been modified by the user. | 
|  | TEST_F(TemplateURLServiceTest, LoadSavesPrepopulatedDefaultSearchProvider) { | 
|  | test_util()->VerifyLoad(); | 
|  | // Verify that the default search provider is set to something. | 
|  | TemplateURL* default_search = model()->GetDefaultSearchProvider(); | 
|  | ASSERT_TRUE(default_search != NULL); | 
|  | scoped_ptr<TemplateURL> cloned_url(new TemplateURL(default_search->data())); | 
|  |  | 
|  | // Wait for any saves to finish. | 
|  | base::RunLoop().RunUntilIdle(); | 
|  |  | 
|  | // Reload the model and check that the default search provider | 
|  | // was properly saved. | 
|  | test_util()->ResetModel(true); | 
|  | default_search = model()->GetDefaultSearchProvider(); | 
|  | ASSERT_TRUE(default_search != NULL); | 
|  | AssertEquals(*cloned_url, *default_search); | 
|  | } | 
|  |  | 
|  | // Make sure that the load routine doesn't delete | 
|  | // prepopulated engines that no longer exist in the prepopulate data if | 
|  | // it is the default search provider. | 
|  | TEST_F(TemplateURLServiceTest, LoadRetainsDefaultProvider) { | 
|  | // Set the default search provider to a preloaded template url which | 
|  | // is not in the current set of preloaded template urls and save | 
|  | // the result. | 
|  | TemplateURL* t_url = CreatePreloadedTemplateURL(true, 999999); | 
|  | test_util()->ChangeModelToLoadState(); | 
|  | model()->Add(t_url); | 
|  | model()->SetUserSelectedDefaultSearchProvider(t_url); | 
|  | // Do the copy after t_url is added and set as default so that its | 
|  | // internal state is correct. | 
|  | scoped_ptr<TemplateURL> cloned_url(new TemplateURL(t_url->data())); | 
|  |  | 
|  | ASSERT_EQ(t_url, model()->GetTemplateURLForKeyword(ASCIIToUTF16("unittest"))); | 
|  | ASSERT_EQ(t_url, model()->GetDefaultSearchProvider()); | 
|  | base::RunLoop().RunUntilIdle(); | 
|  |  | 
|  | // Ensure that merging won't clear the prepopulated template url | 
|  | // which is no longer present if it's the default engine. | 
|  | test_util()->ResetModel(true); | 
|  | { | 
|  | const TemplateURL* keyword_url = | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("unittest")); | 
|  | ASSERT_TRUE(keyword_url != NULL); | 
|  | AssertEquals(*cloned_url, *keyword_url); | 
|  | ASSERT_EQ(keyword_url, model()->GetDefaultSearchProvider()); | 
|  | } | 
|  |  | 
|  | // Wait for any saves to finish. | 
|  | base::RunLoop().RunUntilIdle(); | 
|  |  | 
|  | // Reload the model to verify that the update was saved. | 
|  | test_util()->ResetModel(true); | 
|  | { | 
|  | const TemplateURL* keyword_url = | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("unittest")); | 
|  | ASSERT_TRUE(keyword_url != NULL); | 
|  | AssertEquals(*cloned_url, *keyword_url); | 
|  | ASSERT_EQ(keyword_url, model()->GetDefaultSearchProvider()); | 
|  | } | 
|  | } | 
|  |  | 
|  | // Make sure that the load routine sets a default search provider if it was | 
|  | // missing and not managed. | 
|  | TEST_F(TemplateURLServiceTest, LoadEnsuresDefaultSearchProviderExists) { | 
|  | // Force the model to load and make sure we have a default search provider. | 
|  | test_util()->VerifyLoad(); | 
|  | EXPECT_TRUE(model()->GetDefaultSearchProvider()); | 
|  |  | 
|  | EXPECT_TRUE(model()->GetDefaultSearchProvider()->SupportsReplacement( | 
|  | search_terms_data())); | 
|  |  | 
|  | // Make default search provider unusable (no search terms). | 
|  | model()->ResetTemplateURL(model()->GetDefaultSearchProvider(), | 
|  | ASCIIToUTF16("test"), ASCIIToUTF16("test"), | 
|  | "http://example.com/"); | 
|  | base::RunLoop().RunUntilIdle(); | 
|  |  | 
|  | // Reset the model and load it. There should be a usable default search | 
|  | // provider. | 
|  | test_util()->ResetModel(true); | 
|  |  | 
|  | ASSERT_TRUE(model()->GetDefaultSearchProvider()); | 
|  | EXPECT_TRUE(model()->GetDefaultSearchProvider()->SupportsReplacement( | 
|  | search_terms_data())); | 
|  | } | 
|  |  | 
|  | // Simulates failing to load the webdb and makes sure the default search | 
|  | // provider is valid. | 
|  | TEST_F(TemplateURLServiceTest, FailedInit) { | 
|  | test_util()->VerifyLoad(); | 
|  |  | 
|  | test_util()->ClearModel(); | 
|  | test_util()->web_data_service()->ShutdownDatabase(); | 
|  |  | 
|  | test_util()->ResetModel(false); | 
|  | model()->Load(); | 
|  | base::RunLoop().RunUntilIdle(); | 
|  |  | 
|  | ASSERT_TRUE(model()->GetDefaultSearchProvider()); | 
|  | } | 
|  |  | 
|  | // Verifies that if the default search URL preference is managed, we report | 
|  | // the default search as managed.  Also check that we are getting the right | 
|  | // values. | 
|  | TEST_F(TemplateURLServiceTest, TestManagedDefaultSearch) { | 
|  | test_util()->VerifyLoad(); | 
|  | const size_t initial_count = model()->GetTemplateURLs().size(); | 
|  | test_util()->ResetObserverCount(); | 
|  |  | 
|  | // Set a regular default search provider. | 
|  | TemplateURL* regular_default = AddKeywordWithDate( | 
|  | "name1", "key1", "http://foo1/{searchTerms}", "http://sugg1", | 
|  | std::string(), "http://icon1", true, "UTF-8;UTF-16", Time(), Time()); | 
|  | VerifyObserverCount(1); | 
|  | model()->SetUserSelectedDefaultSearchProvider(regular_default); | 
|  | // Adding the URL and setting the default search provider should have caused | 
|  | // notifications. | 
|  | VerifyObserverCount(1); | 
|  | EXPECT_FALSE(model()->is_default_search_managed()); | 
|  | EXPECT_EQ(initial_count + 1, model()->GetTemplateURLs().size()); | 
|  |  | 
|  | // Set a managed preference that establishes a default search provider. | 
|  | const char kName[] = "test1"; | 
|  | const char kKeyword[] = "test.com"; | 
|  | const char kSearchURL[] = "http://test.com/search?t={searchTerms}"; | 
|  | const char kIconURL[] = "http://test.com/icon.jpg"; | 
|  | const char kEncodings[] = "UTF-16;UTF-32"; | 
|  | const char kAlternateURL[] = "http://test.com/search#t={searchTerms}"; | 
|  | const char kSearchTermsReplacementKey[] = "espv"; | 
|  | test_util()->SetManagedDefaultSearchPreferences(true, kName, kKeyword, | 
|  | kSearchURL, std::string(), kIconURL, kEncodings, kAlternateURL, | 
|  | kSearchTermsReplacementKey); | 
|  | VerifyObserverFired(); | 
|  | EXPECT_TRUE(model()->is_default_search_managed()); | 
|  | EXPECT_EQ(initial_count + 2, model()->GetTemplateURLs().size()); | 
|  |  | 
|  | // Verify that the default manager we are getting is the managed one. | 
|  | TemplateURLData data; | 
|  | data.SetShortName(ASCIIToUTF16(kName)); | 
|  | data.SetKeyword(ASCIIToUTF16(kKeyword)); | 
|  | data.SetURL(kSearchURL); | 
|  | data.favicon_url = GURL(kIconURL); | 
|  | data.show_in_default_list = true; | 
|  | data.input_encodings = base::SplitString( | 
|  | kEncodings, ";", base::TRIM_WHITESPACE, base::SPLIT_WANT_ALL); | 
|  | data.alternate_urls.push_back(kAlternateURL); | 
|  | data.search_terms_replacement_key = kSearchTermsReplacementKey; | 
|  | scoped_ptr<TemplateURL> expected_managed_default1(new TemplateURL(data)); | 
|  | const TemplateURL* actual_managed_default = | 
|  | model()->GetDefaultSearchProvider(); | 
|  | ExpectSimilar(expected_managed_default1.get(), actual_managed_default); | 
|  | EXPECT_TRUE(actual_managed_default->show_in_default_list()); | 
|  |  | 
|  | // Update the managed preference and check that the model has changed. | 
|  | const char kNewName[] = "test2"; | 
|  | const char kNewKeyword[] = "other.com"; | 
|  | const char kNewSearchURL[] = "http://other.com/search?t={searchTerms}"; | 
|  | const char kNewSuggestURL[] = "http://other.com/suggest?t={searchTerms}"; | 
|  | test_util()->SetManagedDefaultSearchPreferences(true, kNewName, kNewKeyword, | 
|  | kNewSearchURL, kNewSuggestURL, std::string(), std::string(), | 
|  | std::string(), std::string()); | 
|  | VerifyObserverFired(); | 
|  | EXPECT_TRUE(model()->is_default_search_managed()); | 
|  | EXPECT_EQ(initial_count + 2, model()->GetTemplateURLs().size()); | 
|  |  | 
|  | // Verify that the default manager we are now getting is the correct one. | 
|  | TemplateURLData data2; | 
|  | data2.SetShortName(ASCIIToUTF16(kNewName)); | 
|  | data2.SetKeyword(ASCIIToUTF16(kNewKeyword)); | 
|  | data2.SetURL(kNewSearchURL); | 
|  | data2.suggestions_url = kNewSuggestURL; | 
|  | data2.show_in_default_list = true; | 
|  | scoped_ptr<TemplateURL> expected_managed_default2(new TemplateURL(data2)); | 
|  | actual_managed_default = model()->GetDefaultSearchProvider(); | 
|  | ExpectSimilar(expected_managed_default2.get(), actual_managed_default); | 
|  | EXPECT_EQ(actual_managed_default->show_in_default_list(), true); | 
|  |  | 
|  | // Remove all the managed prefs and check that we are no longer managed. | 
|  | test_util()->RemoveManagedDefaultSearchPreferences(); | 
|  | VerifyObserverFired(); | 
|  | EXPECT_FALSE(model()->is_default_search_managed()); | 
|  | EXPECT_EQ(initial_count + 1, model()->GetTemplateURLs().size()); | 
|  |  | 
|  | // The default should now be the user preference. | 
|  | const TemplateURL* actual_final_managed_default = | 
|  | model()->GetDefaultSearchProvider(); | 
|  | ExpectSimilar(regular_default, actual_final_managed_default); | 
|  | EXPECT_EQ(actual_final_managed_default->show_in_default_list(), true); | 
|  |  | 
|  | // Disable the default search provider through policy. | 
|  | test_util()->SetManagedDefaultSearchPreferences(false, std::string(), | 
|  | std::string(), std::string(), std::string(), std::string(), | 
|  | std::string(), std::string(), std::string()); | 
|  | VerifyObserverFired(); | 
|  | EXPECT_TRUE(model()->is_default_search_managed()); | 
|  | EXPECT_TRUE(NULL == model()->GetDefaultSearchProvider()); | 
|  | EXPECT_EQ(initial_count + 1, model()->GetTemplateURLs().size()); | 
|  |  | 
|  | // Re-enable it. | 
|  | test_util()->SetManagedDefaultSearchPreferences(true, kName, kKeyword, | 
|  | kSearchURL, std::string(), kIconURL, kEncodings, kAlternateURL, | 
|  | kSearchTermsReplacementKey); | 
|  | VerifyObserverFired(); | 
|  | EXPECT_TRUE(model()->is_default_search_managed()); | 
|  | EXPECT_EQ(initial_count + 2, model()->GetTemplateURLs().size()); | 
|  |  | 
|  | // Verify that the default manager we are getting is the managed one. | 
|  | actual_managed_default = model()->GetDefaultSearchProvider(); | 
|  | ExpectSimilar(expected_managed_default1.get(), actual_managed_default); | 
|  | EXPECT_EQ(actual_managed_default->show_in_default_list(), true); | 
|  |  | 
|  | // Clear the model and disable the default search provider through policy. | 
|  | // Verify that there is no default search provider after loading the model. | 
|  | // This checks against regressions of http://crbug.com/67180 | 
|  |  | 
|  | // First, remove the preferences, reset the model, and set a default. | 
|  | test_util()->RemoveManagedDefaultSearchPreferences(); | 
|  | test_util()->ResetModel(true); | 
|  | TemplateURL* new_default = | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("key1")); | 
|  | ASSERT_FALSE(new_default == NULL); | 
|  | model()->SetUserSelectedDefaultSearchProvider(new_default); | 
|  | EXPECT_EQ(new_default, model()->GetDefaultSearchProvider()); | 
|  |  | 
|  | // Now reset the model again but load it after setting the preferences. | 
|  | test_util()->ResetModel(false); | 
|  | test_util()->SetManagedDefaultSearchPreferences(false, std::string(), | 
|  | std::string(), std::string(), std::string(), std::string(), | 
|  | std::string(), std::string(), std::string()); | 
|  | test_util()->VerifyLoad(); | 
|  | EXPECT_TRUE(model()->is_default_search_managed()); | 
|  | EXPECT_TRUE(model()->GetDefaultSearchProvider() == NULL); | 
|  | } | 
|  |  | 
|  | // Test that if we load a TemplateURL with an empty GUID, the load process | 
|  | // assigns it a newly generated GUID. | 
|  | TEST_F(TemplateURLServiceTest, PatchEmptySyncGUID) { | 
|  | // Add a new TemplateURL. | 
|  | test_util()->VerifyLoad(); | 
|  | const size_t initial_count = model()->GetTemplateURLs().size(); | 
|  |  | 
|  | TemplateURLData data; | 
|  | data.SetShortName(ASCIIToUTF16("google")); | 
|  | data.SetKeyword(ASCIIToUTF16("keyword")); | 
|  | data.SetURL("http://www.google.com/foo/bar"); | 
|  | data.sync_guid.clear(); | 
|  | TemplateURL* t_url = new TemplateURL(data); | 
|  | model()->Add(t_url); | 
|  |  | 
|  | VerifyObserverCount(1); | 
|  | base::RunLoop().RunUntilIdle(); | 
|  | ASSERT_EQ(initial_count + 1, model()->GetTemplateURLs().size()); | 
|  |  | 
|  | // Reload the model to verify it was actually saved to the database and | 
|  | // assigned a new GUID when brought back. | 
|  | test_util()->ResetModel(true); | 
|  | ASSERT_EQ(initial_count + 1, model()->GetTemplateURLs().size()); | 
|  | const TemplateURL* loaded_url = | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("keyword")); | 
|  | ASSERT_FALSE(loaded_url == NULL); | 
|  | ASSERT_FALSE(loaded_url->sync_guid().empty()); | 
|  | } | 
|  |  | 
|  | // Test that if we load a TemplateURL with duplicate input encodings, the load | 
|  | // process de-dupes them. | 
|  | TEST_F(TemplateURLServiceTest, DuplicateInputEncodings) { | 
|  | // Add a new TemplateURL. | 
|  | test_util()->VerifyLoad(); | 
|  | const size_t initial_count = model()->GetTemplateURLs().size(); | 
|  |  | 
|  | TemplateURLData data; | 
|  | data.SetShortName(ASCIIToUTF16("google")); | 
|  | data.SetKeyword(ASCIIToUTF16("keyword")); | 
|  | data.SetURL("http://www.google.com/foo/bar"); | 
|  | std::vector<std::string> encodings; | 
|  | data.input_encodings.push_back("UTF-8"); | 
|  | data.input_encodings.push_back("UTF-8"); | 
|  | data.input_encodings.push_back("UTF-16"); | 
|  | data.input_encodings.push_back("UTF-8"); | 
|  | data.input_encodings.push_back("Big5"); | 
|  | data.input_encodings.push_back("UTF-16"); | 
|  | data.input_encodings.push_back("Big5"); | 
|  | data.input_encodings.push_back("Windows-1252"); | 
|  | TemplateURL* t_url = new TemplateURL(data); | 
|  | model()->Add(t_url); | 
|  |  | 
|  | VerifyObserverCount(1); | 
|  | base::RunLoop().RunUntilIdle(); | 
|  | ASSERT_EQ(initial_count + 1, model()->GetTemplateURLs().size()); | 
|  | const TemplateURL* loaded_url = | 
|  | model()->GetTemplateURLForKeyword(ASCIIToUTF16("keyword")); | 
|  | ASSERT_TRUE(loaded_url != NULL); | 
|  | EXPECT_EQ(8U, loaded_url->input_encodings().size()); | 
|  |  | 
|  | // Reload the model to verify it was actually saved to the database and the | 
|  | // duplicate encodings were removed. | 
|  | test_util()->ResetModel(true); | 
|  | ASSERT_EQ(initial_count + 1, model()->GetTemplateURLs().size()); | 
|  | loaded_url = model()->GetTemplateURLForKeyword(ASCIIToUTF16("keyword")); | 
|  | ASSERT_FALSE(loaded_url == NULL); | 
|  | EXPECT_EQ(4U, loaded_url->input_encodings().size()); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, DefaultExtensionEngine) { | 
|  | test_util()->VerifyLoad(); | 
|  | // Add third-party default search engine. | 
|  | TemplateURL* user_dse = AddKeywordWithDate( | 
|  | "user", "user", "http://www.goo.com/s?q={searchTerms}", | 
|  | std::string(), std::string(), std::string(), | 
|  | true, "UTF-8", Time(), Time()); | 
|  | model()->SetUserSelectedDefaultSearchProvider(user_dse); | 
|  | EXPECT_EQ(user_dse, model()->GetDefaultSearchProvider()); | 
|  |  | 
|  | TemplateURL* ext_dse = CreateKeywordWithDate( | 
|  | model(), "ext", "ext", "http://www.search.com/s?q={searchTerms}", | 
|  | std::string(), std::string(), std::string(), | 
|  | true, true, "UTF-8", Time(), Time()); | 
|  | scoped_ptr<TemplateURL::AssociatedExtensionInfo> extension_info( | 
|  | new TemplateURL::AssociatedExtensionInfo( | 
|  | TemplateURL::NORMAL_CONTROLLED_BY_EXTENSION, "ext")); | 
|  | extension_info->wants_to_be_default_engine = true; | 
|  | model()->AddExtensionControlledTURL(ext_dse, std::move(extension_info)); | 
|  | EXPECT_EQ(ext_dse, model()->GetDefaultSearchProvider()); | 
|  |  | 
|  | model()->RemoveExtensionControlledTURL( | 
|  | "ext", TemplateURL::NORMAL_CONTROLLED_BY_EXTENSION); | 
|  | ExpectSimilar(user_dse, model()->GetDefaultSearchProvider()); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, ExtensionEnginesNotPersist) { | 
|  | test_util()->VerifyLoad(); | 
|  | // Add third-party default search engine. | 
|  | TemplateURL* user_dse = AddKeywordWithDate( | 
|  | "user", "user", "http://www.goo.com/s?q={searchTerms}", | 
|  | std::string(), std::string(), std::string(), | 
|  | true, "UTF-8", Time(), Time()); | 
|  | model()->SetUserSelectedDefaultSearchProvider(user_dse); | 
|  | EXPECT_EQ(user_dse, model()->GetDefaultSearchProvider()); | 
|  |  | 
|  | TemplateURL* ext_dse = CreateKeywordWithDate( | 
|  | model(), "ext1", "ext1", "http://www.ext1.com/s?q={searchTerms}", | 
|  | std::string(), std::string(), std::string(), | 
|  | true, false, "UTF-8", Time(), Time()); | 
|  | scoped_ptr<TemplateURL::AssociatedExtensionInfo> extension_info( | 
|  | new TemplateURL::AssociatedExtensionInfo( | 
|  | TemplateURL::NORMAL_CONTROLLED_BY_EXTENSION, "ext1")); | 
|  | extension_info->wants_to_be_default_engine = false; | 
|  | model()->AddExtensionControlledTURL(ext_dse, std::move(extension_info)); | 
|  | EXPECT_EQ(user_dse, model()->GetDefaultSearchProvider()); | 
|  |  | 
|  | ext_dse = CreateKeywordWithDate( | 
|  | model(), "ext2", "ext2", "http://www.ext2.com/s?q={searchTerms}", | 
|  | std::string(), std::string(), std::string(), | 
|  | true, true, "UTF-8", Time(), Time()); | 
|  | extension_info.reset(new TemplateURL::AssociatedExtensionInfo( | 
|  | TemplateURL::NORMAL_CONTROLLED_BY_EXTENSION, "ext2")); | 
|  | extension_info->wants_to_be_default_engine = true; | 
|  | model()->AddExtensionControlledTURL(ext_dse, std::move(extension_info)); | 
|  | EXPECT_EQ(ext_dse, model()->GetDefaultSearchProvider()); | 
|  |  | 
|  | test_util()->ResetModel(true); | 
|  | user_dse = model()->GetTemplateURLForKeyword(ASCIIToUTF16("user")); | 
|  | ExpectSimilar(user_dse, model()->GetDefaultSearchProvider()); | 
|  | EXPECT_FALSE(model()->GetTemplateURLForKeyword(ASCIIToUTF16("ext1"))); | 
|  | EXPECT_FALSE(model()->GetTemplateURLForKeyword(ASCIIToUTF16("ext2"))); | 
|  | } | 
|  |  | 
|  | TEST_F(TemplateURLServiceTest, ExtensionEngineVsPolicy) { | 
|  | // Set a managed preference that establishes a default search provider. | 
|  | const char kName[] = "test"; | 
|  | const char kKeyword[] = "test.com"; | 
|  | const char kSearchURL[] = "http://test.com/search?t={searchTerms}"; | 
|  | const char kIconURL[] = "http://test.com/icon.jpg"; | 
|  | const char kEncodings[] = "UTF-16;UTF-32"; | 
|  | const char kAlternateURL[] = "http://test.com/search#t={searchTerms}"; | 
|  | const char kSearchTermsReplacementKey[] = "espv"; | 
|  | test_util()->SetManagedDefaultSearchPreferences( | 
|  | true, kName, kKeyword, kSearchURL, std::string(), kIconURL, kEncodings, | 
|  | kAlternateURL, kSearchTermsReplacementKey); | 
|  | test_util()->VerifyLoad(); | 
|  | // Verify that the default manager we are getting is the managed one. | 
|  | TemplateURLData data; | 
|  | data.SetShortName(ASCIIToUTF16(kName)); | 
|  | data.SetKeyword(ASCIIToUTF16(kKeyword)); | 
|  | data.SetURL(kSearchURL); | 
|  | data.favicon_url = GURL(kIconURL); | 
|  | data.show_in_default_list = true; | 
|  | data.input_encodings = base::SplitString( | 
|  | kEncodings, ";", base::TRIM_WHITESPACE, base::SPLIT_WANT_ALL); | 
|  | data.alternate_urls.push_back(kAlternateURL); | 
|  | data.search_terms_replacement_key = kSearchTermsReplacementKey; | 
|  | scoped_ptr<TemplateURL> expected_managed_default(new TemplateURL(data)); | 
|  | EXPECT_TRUE(model()->is_default_search_managed()); | 
|  | const TemplateURL* actual_managed_default = | 
|  | model()->GetDefaultSearchProvider(); | 
|  | ExpectSimilar(expected_managed_default.get(), actual_managed_default); | 
|  |  | 
|  | TemplateURL* ext_dse = CreateKeywordWithDate( | 
|  | model(), "ext1", "ext1", "http://www.ext1.com/s?q={searchTerms}", | 
|  | std::string(), std::string(), std::string(), | 
|  | true, true, "UTF-8", Time(), Time()); | 
|  | scoped_ptr<TemplateURL::AssociatedExtensionInfo> extension_info( | 
|  | new TemplateURL::AssociatedExtensionInfo( | 
|  | TemplateURL::NORMAL_CONTROLLED_BY_EXTENSION, "ext1")); | 
|  | extension_info->wants_to_be_default_engine = true; | 
|  | model()->AddExtensionControlledTURL(ext_dse, std::move(extension_info)); | 
|  | EXPECT_EQ(ext_dse, model()->GetTemplateURLForKeyword(ASCIIToUTF16("ext1"))); | 
|  | EXPECT_TRUE(model()->is_default_search_managed()); | 
|  | actual_managed_default = model()->GetDefaultSearchProvider(); | 
|  | ExpectSimilar(expected_managed_default.get(), actual_managed_default); | 
|  | } |