diff options
Diffstat (limited to 'chrome/browser/autocomplete/autocomplete_unittest.cc')
-rw-r--r-- | chrome/browser/autocomplete/autocomplete_unittest.cc | 361 |
1 files changed, 361 insertions, 0 deletions
diff --git a/chrome/browser/autocomplete/autocomplete_unittest.cc b/chrome/browser/autocomplete/autocomplete_unittest.cc new file mode 100644 index 0000000..e2fc2f5 --- /dev/null +++ b/chrome/browser/autocomplete/autocomplete_unittest.cc @@ -0,0 +1,361 @@ +// Copyright 2008, Google Inc. +// All rights reserved. +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +#include "base/message_loop.h" +#include "base/ref_counted.h" +#include "chrome/browser/autocomplete/autocomplete.h" +#include "testing/gtest/include/gtest/gtest.h" + +// identifiers for known autocomplete providers +#define HISTORY_IDENTIFIER L"Chrome:History" +#define SEARCH_IDENTIFIER L"google.com/websearch/en" + +namespace { + +const int num_results_per_provider = 3; + +// Autocomplete provider that provides known results. Note that this is +// refcounted so that it can also be a task on the message loop. +class TestProvider : public AutocompleteProvider { + public: + TestProvider(int relevance, const std::wstring& prefix) + : AutocompleteProvider(NULL, NULL, ""), + relevance_(relevance), + prefix_(prefix) { + } + + virtual void Start(const AutocompleteInput& input, + bool minimal_changes, + bool synchronous_only); + + void set_listener(ACProviderListener* listener) { + listener_ = listener; + } + + private: + void Run(); + + void AddResults(int start_at, int num); + + int relevance_; + const std::wstring prefix_; +}; + +void TestProvider::Start(const AutocompleteInput& input, + bool minimal_changes, + bool synchronous_only) { + if (minimal_changes) + return; + + matches_.clear(); + + // Generate one result synchronously, the rest later. + AddResults(0, 1); + + if (!synchronous_only) { + done_ = false; + MessageLoop::current()->PostTask(FROM_HERE, NewRunnableMethod( + this, &TestProvider::Run)); + } +} + +void TestProvider::Run() { + DCHECK(num_results_per_provider > 0); + AddResults(1, num_results_per_provider); + done_ = true; + DCHECK(listener_); + listener_->OnProviderUpdate(true); +} + +void TestProvider::AddResults(int start_at, int num) { + for (int i = start_at; i < num; i++) { + AutocompleteMatch match(this, relevance_ - i, false); + + wchar_t str[16]; + swprintf_s(str, L"%d", i); + match.fill_into_edit = prefix_ + str; + match.destination_url = match.fill_into_edit; + + match.contents = match.destination_url; + match.contents_class.push_back( + ACMatchClassification(0, ACMatchClassification::NONE)); + match.description = match.destination_url; + match.description_class.push_back( + ACMatchClassification(0, ACMatchClassification::NONE)); + + matches_.push_back(match); + } +} + +class AutocompleteProviderTest : public testing::Test, + public ACControllerListener { + public: + // ACControllerListener + virtual void OnAutocompleteUpdate(bool updated_result, bool query_complete); + + protected: + // testing::Test + virtual void SetUp(); + + void ResetController(bool same_destinations); + + // Runs a query on the input "a", and makes sure both providers' input is + // properly collected. + void RunTest(); + + // These providers are owned by the controller once it's created. + ACProviders providers_; + + AutocompleteResult result_; + + private: + scoped_ptr<AutocompleteController> controller_; +}; + +void AutocompleteProviderTest::SetUp() { + ResetController(false); +} + +void AutocompleteProviderTest::ResetController(bool same_destinations) { + // Forget about any existing providers. The controller owns them and will + // Release() them below, when we delete it during the call to reset(). + providers_.clear(); + + // Construct two new providers, with either the same or different prefixes. + TestProvider* providerA = new TestProvider(num_results_per_provider, L"a"); + providerA->AddRef(); + providers_.push_back(providerA); + + TestProvider* providerB = new TestProvider(num_results_per_provider * 2, + same_destinations ? L"a" : L"b"); + providerB->AddRef(); + providers_.push_back(providerB); + + // Reset the controller to contain our new providers. + AutocompleteController* controller = + new AutocompleteController(this, providers_); + controller_.reset(controller); + providerA->set_listener(controller); + providerB->set_listener(controller); +} + +void AutocompleteProviderTest::OnAutocompleteUpdate(bool updated_result, + bool query_complete) { + controller_->GetResult(&result_); + if (query_complete) + MessageLoop::current()->Quit(); +} + +void AutocompleteProviderTest::RunTest() { + result_.Reset(); + const AutocompleteInput input(L"a", std::wstring(), true); + EXPECT_FALSE(controller_->Start(input, false, false)); + + // The message loop will terminate when all autocomplete input has been + // collected. + MessageLoop::current()->Run(); +} + +} // namespace + +std::ostream& operator<<(std::ostream& os, + const AutocompleteResult::const_iterator& iter) { + return os << static_cast<const AutocompleteMatch*>(&(*iter)); +} + +// Tests that the default selection is set properly when updating results. +TEST_F(AutocompleteProviderTest, Query) { + RunTest(); + + // Make sure the default match gets set to the highest relevance match when + // we have no preference. The highest relevance matches should come from + // the second provider. + AutocompleteResult::Selection selection; + result_.SetDefaultMatch(selection); + EXPECT_EQ(num_results_per_provider * 2, result_.size()); // two providers + ASSERT_NE(result_.end(), result_.default_match()); + EXPECT_EQ(providers_[1], result_.default_match()->provider); + + // Change provider affinity. + selection.provider_affinity = providers_[0]; + result_.SetDefaultMatch(selection); + ASSERT_NE(result_.end(), result_.default_match()); + EXPECT_EQ(providers_[0], result_.default_match()->provider); +} + +TEST_F(AutocompleteProviderTest, UpdateSelection) { + RunTest(); + + // An empty selection should simply result in the default match overall. + AutocompleteResult::Selection selection; + result_.SetDefaultMatch(selection); + ASSERT_NE(result_.end(), result_.default_match()); + EXPECT_EQ(providers_[1], result_.default_match()->provider); + + // ...As should specifying a provider that didn't return results. + scoped_refptr<TestProvider> test_provider = new TestProvider(0, L""); + selection.provider_affinity = test_provider.get(); + result_.SetDefaultMatch(selection); + ASSERT_NE(result_.end(), result_.default_match()); + EXPECT_EQ(providers_[1], result_.default_match()->provider); + selection.provider_affinity = NULL; + test_provider = NULL; + + // ...As should specifying a destination URL that doesn't exist, and no + // provider. + selection.destination_url = L"garbage"; + selection.provider_affinity = NULL; + result_.SetDefaultMatch(selection); + ASSERT_NE(result_.end(), result_.default_match()); + EXPECT_EQ(providers_[1], result_.default_match()->provider); + delete selection.provider_affinity; + + // Specifying a valid provider should result in the default match from that + // provider. + selection.destination_url.clear(); + selection.provider_affinity = providers_[0]; + result_.SetDefaultMatch(selection); + ASSERT_NE(result_.end(), result_.default_match()); + EXPECT_EQ(providers_[0], result_.default_match()->provider); + + // ...And nothing should change if we specify a destination that doesn't + // exist. + selection.destination_url = L"garbage"; + result_.SetDefaultMatch(selection); + ASSERT_NE(result_.end(), result_.default_match()); + EXPECT_EQ(providers_[0], result_.default_match()->provider); + + // Specifying a particular URL should match that URL. + std::wstring non_default_url_from_provider_0(L"a2"); + selection.destination_url = non_default_url_from_provider_0; + selection.provider_affinity = providers_[0]; + result_.SetDefaultMatch(selection); + ASSERT_NE(result_.end(), result_.default_match()); + EXPECT_EQ(non_default_url_from_provider_0, + result_.default_match()->destination_url); + + // ...Even when we ask for a different provider. + selection.provider_affinity = providers_[1]; + result_.SetDefaultMatch(selection); + ASSERT_NE(result_.end(), result_.default_match()); + EXPECT_EQ(non_default_url_from_provider_0, + result_.default_match()->destination_url); + + // ...Or when we don't ask for a provider at all. + selection.provider_affinity = NULL; + result_.SetDefaultMatch(selection); + ASSERT_NE(result_.end(), result_.default_match()); + EXPECT_EQ(non_default_url_from_provider_0, + result_.default_match()->destination_url); +} + +TEST_F(AutocompleteProviderTest, RemoveDuplicates) { + // Set up the providers to provide duplicate results. + ResetController(true); + + RunTest(); + + // Make sure all the first provider's results were eliminated by the second + // provider's. + EXPECT_EQ(num_results_per_provider, result_.size()); + for (AutocompleteResult::const_iterator i(result_.begin()); + i != result_.end(); ++i) + EXPECT_EQ(providers_[1], i->provider); + + // Set things back to the default for the benefit of any tests that run after + // us. + ResetController(false); +} + +TEST(AutocompleteTest, InputType) { + struct test_data { + const wchar_t* input; + const AutocompleteInput::Type type; + } input_cases[] = { + { L"", AutocompleteInput::INVALID }, + { L"?", AutocompleteInput::FORCED_QUERY }, + { L"?foo", AutocompleteInput::FORCED_QUERY }, + { L"?foo bar", AutocompleteInput::FORCED_QUERY }, + { L"?http://foo.com/bar", AutocompleteInput::FORCED_QUERY }, + { L"foo", AutocompleteInput::UNKNOWN }, + { L"foo.com", AutocompleteInput::URL }, + { L"foo/bar", AutocompleteInput::URL }, + { L"foo/bar baz", AutocompleteInput::UNKNOWN }, + { L"http://foo/bar baz", AutocompleteInput::URL }, + { L"foo bar", AutocompleteInput::QUERY }, + { L"link:foo.com", AutocompleteInput::UNKNOWN }, + { L"www.foo.com:81", AutocompleteInput::URL }, + { L"localhost:8080", AutocompleteInput::URL }, + { L"en.wikipedia.org/wiki/James Bond", AutocompleteInput::URL }, + // In Chrome itself, mailto: will get handled by ShellExecute, but in + // unittest mode, we don't have the data loaded in the external protocol + // handler to know this. + // { L"mailto:abuse@foo.com", AutocompleteInput::URL }, + { L"view-source:http://www.foo.com/", AutocompleteInput::URL }, + { L"javascript:alert(\"Hey there!\");", AutocompleteInput::URL }, + { L"C:\\Program Files", AutocompleteInput::URL }, + { L"\\\\Server\\Folder\\File", AutocompleteInput::URL }, + { L"http://foo.com/", AutocompleteInput::URL }, + { L"127.0.0.1", AutocompleteInput::URL }, + { L"browser.tabs.closeButtons", AutocompleteInput::UNKNOWN }, + }; + + for (int i = 0; i < arraysize(input_cases); ++i) { + AutocompleteInput input(input_cases[i].input, std::wstring(), true); + EXPECT_EQ(input_cases[i].type, input.type()) << "Input: " << + input_cases[i].input; + } +} + +// Test that we can properly compare matches' relevance when at least one is +// negative. +TEST(AutocompleteMatch, MoreRelevant) { + struct RelevantCases { + int r1; + int r2; + bool expected_result; + } cases[] = { + { 10, 0, true }, + { 10, -5, true }, + { -5, 10, false }, + { 0, 10, false }, + { -10, -5, true }, + { -5, -10, false }, + }; + + AutocompleteMatch m1; + AutocompleteMatch m2; + + for (int i = 0; i < arraysize(cases); ++i) { + m1.relevance = cases[i].r1; + m2.relevance = cases[i].r2; + EXPECT_EQ(cases[i].expected_result, + AutocompleteMatch::MoreRelevant(m1, m2)); + } +} |