blob: ce6224abbc751a81b850cfe680283accebc56cd7 [file] [log] [blame]
// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "components/omnibox/browser/omnibox_view.h"
#include <stddef.h>
#include <array>
#include <string>
#include <utility>
#include "base/functional/callback_helpers.h"
#include "base/memory/raw_ptr.h"
#include "base/strings/string_util.h"
#include "base/strings/utf_string_conversions.h"
#include "base/test/scoped_feature_list.h"
#include "base/test/task_environment.h"
#include "build/build_config.h"
#include "components/bookmarks/browser/bookmark_model.h"
#include "components/bookmarks/test/test_bookmark_client.h"
#include "components/omnibox/browser/autocomplete_match.h"
#include "components/omnibox/browser/omnibox_controller.h"
#include "components/omnibox/browser/test_omnibox_client.h"
#include "components/omnibox/browser/test_omnibox_edit_model.h"
#include "components/omnibox/browser/test_omnibox_view.h"
#include "components/omnibox/common/omnibox_features.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/base/ui_base_features.h"
#include "ui/gfx/color_palette.h"
#include "ui/gfx/favicon_size.h"
#include "ui/gfx/paint_vector_icon.h"
#if !BUILDFLAG(IS_ANDROID) && !BUILDFLAG(IS_IOS)
#include "components/omnibox/browser/vector_icons.h" // nogncheck
#include "components/vector_icons/vector_icons.h" // nogncheck
#endif
using base::ASCIIToUTF16;
using testing::_;
using testing::DoAll;
using testing::Return;
using testing::SaveArg;
namespace {
class OmniboxViewTest : public testing::Test {
public:
OmniboxViewTest()
: bookmark_model_(bookmarks::TestBookmarkClient::CreateModel()) {
auto omnibox_client = std::make_unique<TestOmniboxClient>();
omnibox_client_ = omnibox_client.get();
EXPECT_CALL(*client(), GetBookmarkModel())
.WillRepeatedly(Return(bookmark_model_.get()));
view_ = std::make_unique<TestOmniboxView>(std::move(omnibox_client));
view_->controller()->SetEditModelForTesting(
std::make_unique<TestOmniboxEditModel>(view_->controller(), view_.get(),
/*pref_service=*/nullptr));
}
TestOmniboxView* view() { return view_.get(); }
TestOmniboxEditModel* model() {
return static_cast<TestOmniboxEditModel*>(view_->model());
}
TestOmniboxClient* client() { return omnibox_client_; }
bookmarks::BookmarkModel* bookmark_model() { return bookmark_model_.get(); }
private:
base::test::TaskEnvironment task_environment_;
raw_ptr<TestOmniboxClient, DanglingUntriaged> omnibox_client_;
std::unique_ptr<TestOmniboxView> view_;
std::unique_ptr<bookmarks::BookmarkModel> bookmark_model_;
};
TEST_F(OmniboxViewTest, TestStripSchemasUnsafeForPaste) {
constexpr const auto urls = std::to_array<const char*>({
" \x01 ", // Safe query.
"http://www.google.com?q=javascript:alert(0)", // Safe URL.
"JavaScript", // Safe query.
"javaScript:", // Unsafe JS URL.
" javaScript: ", // Unsafe JS URL.
"javAscript:Javascript:javascript", // Unsafe JS URL.
"javAscript:alert(1)", // Unsafe JS URL.
"javAscript:javascript:alert(2)", // Single strip unsafe.
"jaVascript:\njavaScript:\x01 alert(3) \x01", // Single strip unsafe.
("\x01\x02\x03\x04\x05\x06\x07\x08\x09\x10\x11\x12\x13\x14\x15\x16\x17"
"\x18\x19 JavaScript:alert(4)"), // Leading control chars unsafe.
"\x01\x02javascript:\x03\x04JavaScript:alert(5)", // Embedded control
// characters unsafe.
});
constexpr const auto expecteds = std::to_array<const char*>({
" \x01 ", // Safe query.
"http://www.google.com?q=javascript:alert(0)", // Safe URL.
"JavaScript", // Safe query.
"", // Unsafe JS URL.
"", // Unsafe JS URL.
"javascript", // Unsafe JS URL.
"alert(1)", // Unsafe JS URL.
"alert(2)", // Single strip unsafe.
"alert(3) \x01", // Single strip unsafe.
"alert(4)", // Leading control chars unsafe.
"alert(5)", // Embedded control characters unsafe.
});
for (size_t i = 0; i < std::size(urls); i++) {
EXPECT_EQ(ASCIIToUTF16(expecteds[i]),
OmniboxView::StripJavascriptSchemas(base::UTF8ToUTF16(urls[i])));
}
}
TEST_F(OmniboxViewTest, SanitizeTextForPaste) {
const struct {
std::u16string input;
std::u16string output;
} kTestcases[] = {
// No whitespace: leave unchanged.
{std::u16string(), std::u16string()},
{u"a", u"a"},
{u"abc", u"abc"},
// Leading/trailing whitespace: remove.
{u" abc", u"abc"},
{u" \n abc", u"abc"},
{u"abc ", u"abc"},
{u"abc\t \t", u"abc"},
{u"\nabc\n", u"abc"},
// All whitespace: Convert to single space.
{u" ", u" "},
{u"\n", u" "},
{u" ", u" "},
{u"\n\n\n", u" "},
{u" \n\t", u" "},
// Broken URL has newlines stripped.
{u"http://www.chromium.org/developers/testing/chromium-\n"
u"build-infrastructure/tour-of-the-chromium-buildbot",
u"http://www.chromium.org/developers/testing/"
u"chromium-build-infrastructure/tour-of-the-chromium-buildbot"},
// Multi-line address is converted to a single-line address.
{u"1600 Amphitheatre Parkway\nMountain View, CA",
u"1600 Amphitheatre Parkway Mountain View, CA"},
// Line-breaking the JavaScript scheme with no other whitespace results in
// a
// dangerous URL that is sanitized by dropping the scheme.
{u"java\x0d\x0ascript:alert(0)", u"alert(0)"},
// Line-breaking the JavaScript scheme with whitespace elsewhere in the
// string results in a safe string with a space replacing the line break.
{u"java\x0d\x0ascript: alert(0)", u"java script: alert(0)"},
// Unusual URL with multiple internal spaces is preserved as-is.
{u"http://foo.com/a. b", u"http://foo.com/a. b"},
// URL with unicode whitespace is also preserved as-is.
{u"http://foo.com/a\x3000"
u"b",
u"http://foo.com/a\x3000"
u"b"},
};
for (const auto& testcase : kTestcases) {
EXPECT_EQ(testcase.output,
OmniboxView::SanitizeTextForPaste(testcase.input));
}
}
#if !BUILDFLAG(IS_ANDROID) && !BUILDFLAG(IS_IOS)
// Tests GetIcon returns the default search icon when the match is a search
// query.
TEST_F(OmniboxViewTest, DISABLED_GetIcon_Default) {
ui::ImageModel expected_icon =
ui::ImageModel::FromVectorIcon(vector_icons::kSearchChromeRefreshIcon,
gfx::kPlaceholderColor, gfx::kFaviconSize);
ui::ImageModel icon = view()->GetIcon(
gfx::kFaviconSize, gfx::kPlaceholderColor, gfx::kPlaceholderColor,
gfx::kPlaceholderColor, gfx::kPlaceholderColor, base::DoNothing(), false);
EXPECT_EQ(expected_icon, icon);
}
// Tests GetIcon returns the bookmark icon when the match is bookmarked.
TEST_F(OmniboxViewTest, DISABLED_GetIcon_BookmarkIcon) {
const GURL kUrl("https://bookmarks.com");
AutocompleteMatch match;
match.destination_url = kUrl;
model()->SetCurrentMatchForTest(match);
bookmark_model()->AddURL(bookmark_model()->bookmark_bar_node(), 0,
u"a bookmark", kUrl);
ui::ImageModel expected_icon =
ui::ImageModel::FromVectorIcon(omnibox::kBookmarkChromeRefreshIcon,
gfx::kPlaceholderColor, gfx::kFaviconSize);
ui::ImageModel icon = view()->GetIcon(
gfx::kFaviconSize, gfx::kPlaceholderColor, gfx::kPlaceholderColor,
gfx::kPlaceholderColor, gfx::kPlaceholderColor, base::DoNothing(), false);
EXPECT_EQ(expected_icon, icon);
}
// Tests GetIcon returns the website's favicon when the match is a website.
TEST_F(OmniboxViewTest, GetIcon_Favicon) {
const GURL kUrl("https://woahDude.com");
GURL page_url;
EXPECT_CALL(*client(), GetFaviconForPageUrl(_, _))
.WillOnce(DoAll(SaveArg<0>(&page_url), Return(gfx::Image())));
AutocompleteMatch match;
match.type = AutocompleteMatchType::URL_WHAT_YOU_TYPED;
match.destination_url = kUrl;
model()->SetCurrentMatchForTest(match);
view()->GetIcon(gfx::kFaviconSize, gfx::kPlaceholderColor,
gfx::kPlaceholderColor, gfx::kPlaceholderColor,
gfx::kPlaceholderColor, base::DoNothing(), false);
EXPECT_EQ(page_url, kUrl);
}
#endif // !BUILDFLAG(IS_ANDROID) && !BUILDFLAG(IS_IOS)
// Tests GetStateChanges correctly determines if text was deleted.
TEST_F(OmniboxViewTest, GetStateChanges_DeletedText) {
{
// Continuing autocompletion
auto state_before =
TestOmniboxView::CreateState("google.com", 10, 3, 0); // goo[gle.com]
auto state_after = TestOmniboxView::CreateState("goog", 4, 4, 0); // goog|
auto state_changes = view()->GetStateChanges(state_before, state_after);
EXPECT_FALSE(state_changes.just_deleted_text);
}
{
// Typing not the autocompletion
auto state_before =
TestOmniboxView::CreateState("google.com", 1, 10, 0); // g[oogle.com]
auto state_after = TestOmniboxView::CreateState("gi", 2, 2, 0); // gi|
auto state_changes = view()->GetStateChanges(state_before, state_after);
EXPECT_FALSE(state_changes.just_deleted_text);
}
{
// Deleting autocompletion
auto state_before =
TestOmniboxView::CreateState("google.com", 1, 10, 0); // g[oogle.com]
auto state_after = TestOmniboxView::CreateState("g", 1, 1, 0); // g|
auto state_changes = view()->GetStateChanges(state_before, state_after);
EXPECT_TRUE(state_changes.just_deleted_text);
}
{
// Inserting
auto state_before =
TestOmniboxView::CreateState("goole.com", 3, 3, 0); // goo|le.com
auto state_after =
TestOmniboxView::CreateState("google.com", 4, 4, 0); // goog|le.com
auto state_changes = view()->GetStateChanges(state_before, state_after);
EXPECT_FALSE(state_changes.just_deleted_text);
}
{
// Deleting
auto state_before =
TestOmniboxView::CreateState("googgle.com", 5, 5, 0); // googg|le.com
auto state_after =
TestOmniboxView::CreateState("google.com", 4, 4, 0); // goog|le.com
auto state_changes = view()->GetStateChanges(state_before, state_after);
EXPECT_TRUE(state_changes.just_deleted_text);
}
{
// Replacing
auto state_before =
TestOmniboxView::CreateState("goojle.com", 3, 4, 0); // goo[j]le.com
auto state_after =
TestOmniboxView::CreateState("google.com", 4, 4, 0); // goog|le.com
auto state_changes = view()->GetStateChanges(state_before, state_after);
EXPECT_FALSE(state_changes.just_deleted_text);
}
}
// Tests GetStateChanges correctly determines if text was deleted.
TEST_F(OmniboxViewTest, GetStateChanges_DeletedText_RichAutocompletion) {
base::test::ScopedFeatureList feature_list;
feature_list.InitAndEnableFeatureWithParameters(
omnibox::kRichAutocompletion,
{{OmniboxFieldTrial::kRichAutocompletionAutocompleteNonPrefixAll.name,
"true"}});
// Cases with single selection
{
// Continuing autocompletion
auto state_before =
TestOmniboxView::CreateState("google.com", 10, 3, 7); // goo[gle.com]
auto state_after = TestOmniboxView::CreateState("goog", 4, 4, 0); // goog|
auto state_changes = view()->GetStateChanges(state_before, state_after);
EXPECT_FALSE(state_changes.just_deleted_text);
}
{
// Typing not the autocompletion
auto state_before =
TestOmniboxView::CreateState("google.com", 1, 10, 9); // g[oogle.com]
auto state_after = TestOmniboxView::CreateState("gi", 2, 2, 0); // gi|
auto state_changes = view()->GetStateChanges(state_before, state_after);
EXPECT_FALSE(state_changes.just_deleted_text);
}
{
// Deleting autocompletion
auto state_before =
TestOmniboxView::CreateState("google.com", 1, 10, 9); // g[oogle.com]
auto state_after = TestOmniboxView::CreateState("g", 1, 1, 0); // g|
auto state_changes = view()->GetStateChanges(state_before, state_after);
EXPECT_TRUE(state_changes.just_deleted_text);
}
{
// Inserting
auto state_before =
TestOmniboxView::CreateState("goole.com", 3, 3, 6); // goo|le.com
auto state_after =
TestOmniboxView::CreateState("google.com", 4, 4, 0); // goog|le.com
auto state_changes = view()->GetStateChanges(state_before, state_after);
EXPECT_FALSE(state_changes.just_deleted_text);
}
{
// Deleting
auto state_before =
TestOmniboxView::CreateState("googgle.com", 5, 5, 0); // googg|le.com
auto state_after =
TestOmniboxView::CreateState("google.com", 4, 4, 0); // goog|le.com
auto state_changes = view()->GetStateChanges(state_before, state_after);
EXPECT_TRUE(state_changes.just_deleted_text);
}
{
// Replacing
auto state_before =
TestOmniboxView::CreateState("goojle.com", 3, 4, 1); // goo[j]le.com
auto state_after =
TestOmniboxView::CreateState("google.com", 4, 4, 0); // goog|le.com
auto state_changes = view()->GetStateChanges(state_before, state_after);
EXPECT_FALSE(state_changes.just_deleted_text);
}
// Cases with multiselection
{
// Continuing autocompletion with multiselection
auto state_before =
TestOmniboxView::CreateState("google.com", 4, 10, 7); // [g]oog[le.com]
auto state_after = TestOmniboxView::CreateState("oogl", 4, 4, 0); // oogl|
auto state_changes = view()->GetStateChanges(state_before, state_after);
EXPECT_FALSE(state_changes.just_deleted_text);
}
{
// Typing not the autocompletion with multiselection
auto state_before =
TestOmniboxView::CreateState("google.com", 4, 10, 7); // [g]oog[le.com]
auto state_after = TestOmniboxView::CreateState("oogm", 4, 4, 0); // oogm|
auto state_changes = view()->GetStateChanges(state_before, state_after);
EXPECT_FALSE(state_changes.just_deleted_text);
}
{
// Deleting autocompletion with multiselection
auto state_before =
TestOmniboxView::CreateState("google.com", 4, 10, 7); // [g]oog[le.com]
auto state_after = TestOmniboxView::CreateState("oog", 3, 3, 0); // oog|
auto state_changes = view()->GetStateChanges(state_before, state_after);
EXPECT_TRUE(state_changes.just_deleted_text);
}
}
} // namespace