// 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 "base/bind.h" #include "base/bind_helpers.h" #include "base/callback.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 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 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 clock(new base::SimpleTestClock); clock->SetNow(now); model()->set_clock(clock.Pass()); // 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 clock(new base::SimpleTestClock); clock->SetNow(now); model()->set_clock(clock.Pass()); // 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 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 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 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 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 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 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 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 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 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 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 extension_info( new TemplateURL::AssociatedExtensionInfo( TemplateURL::NORMAL_CONTROLLED_BY_EXTENSION, "ext")); extension_info->wants_to_be_default_engine = true; model()->AddExtensionControlledTURL(ext_dse, extension_info.Pass()); 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 extension_info( new TemplateURL::AssociatedExtensionInfo( TemplateURL::NORMAL_CONTROLLED_BY_EXTENSION, "ext1")); extension_info->wants_to_be_default_engine = false; model()->AddExtensionControlledTURL(ext_dse, extension_info.Pass()); 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, extension_info.Pass()); 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 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 extension_info( new TemplateURL::AssociatedExtensionInfo( TemplateURL::NORMAL_CONTROLLED_BY_EXTENSION, "ext1")); extension_info->wants_to_be_default_engine = true; model()->AddExtensionControlledTURL(ext_dse, extension_info.Pass()); 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); }