blob: 1e28f87f032519b20261bef1f55a4f5803884404 [file] [log] [blame]
// Copyright 2014 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "components/omnibox/browser/suggestion_answer.h"
#include <algorithm>
#include <memory>
#include "base/json/json_reader.h"
#include "base/strings/utf_string_conversions.h"
#include "base/test/scoped_feature_list.h"
#include "base/values.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/base/ui_base_features.h"
namespace {
bool ParseAnswer(const std::string& answer_json, SuggestionAnswer* answer) {
base::Optional<base::Value> value = base::JSONReader::Read(answer_json);
if (!value || !value->is_dict())
return false;
// ParseAnswer previously did not change the default answer type of -1, so
// here we keep the same behavior by explicitly supplying default value.
return SuggestionAnswer::ParseAnswer(*value, base::UTF8ToUTF16("-1"), answer);
}
} // namespace
TEST(SuggestionAnswerTest, DefaultAreEqual) {
SuggestionAnswer answer1;
SuggestionAnswer answer2;
EXPECT_TRUE(answer1.Equals(answer2));
}
TEST(SuggestionAnswerTest, CopiesAreEqual) {
SuggestionAnswer answer1;
EXPECT_TRUE(answer1.Equals(SuggestionAnswer(answer1)));
auto answer2 = std::make_unique<SuggestionAnswer>();
answer2->set_type(832345);
EXPECT_TRUE(answer2->Equals(SuggestionAnswer(*answer2)));
std::string json =
"{ \"l\": ["
" { \"il\": { \"t\": [{ \"t\": \"text\", \"tt\": 8 }] } }, "
" { \"il\": { \"t\": [{ \"t\": \"other text\", \"tt\": 5 }] } } "
"] }";
SuggestionAnswer answer3;
ASSERT_TRUE(ParseAnswer(json, &answer3));
EXPECT_TRUE(answer3.Equals(SuggestionAnswer(answer3)));
}
TEST(SuggestionAnswerTest, DifferentValuesAreUnequal) {
std::string json =
"{ \"l\": ["
" { \"il\": { \"t\": [{ \"t\": \"text\", \"tt\": 8 }, "
" { \"t\": \"moar text\", \"tt\": 0 }], "
" \"i\": { \"d\": \"//example.com/foo.jpg\" } } }, "
" { \"il\": { \"t\": [{ \"t\": \"other text\", \"tt\": 5 }], "
" \"at\": { \"t\": \"slatfatf\", \"tt\": 42 }, "
" \"st\": { \"t\": \"oh hi, Mark\", \"tt\": 729347 } } } "
"] }";
SuggestionAnswer answer1;
ASSERT_TRUE(ParseAnswer(json, &answer1));
// Same but with a different answer type.
SuggestionAnswer answer2 = answer1;
EXPECT_TRUE(answer1.Equals(answer2));
answer2.set_type(44);
EXPECT_FALSE(answer1.Equals(answer2));
// Same but with a different type for one of the text fields.
answer2 = answer1;
EXPECT_TRUE(answer1.Equals(answer2));
answer2.first_line_.text_fields_[1].type_ = 1;
EXPECT_FALSE(answer1.Equals(answer2));
// Same but with different text for one of the text fields.
answer2 = answer1;
EXPECT_TRUE(answer1.Equals(answer2));
answer2.first_line_.text_fields_[0].text_ = base::UTF8ToUTF16("some text");
EXPECT_FALSE(answer1.Equals(answer2));
// Same but with a new URL on the second line.
answer2 = answer1;
EXPECT_TRUE(answer1.Equals(answer2));
answer2.second_line_.image_url_ = GURL("http://foo.com/bar.png");
EXPECT_FALSE(answer1.Equals(answer2));
// Same but with the additional text removed from the second line.
answer2 = answer1;
EXPECT_TRUE(answer1.Equals(answer2));
answer2.second_line_.additional_text_.reset();
EXPECT_FALSE(answer1.Equals(answer2));
// Same but with the status text removed from the second line.
answer2 = answer1;
EXPECT_TRUE(answer1.Equals(answer2));
answer2.second_line_.status_text_.reset();
EXPECT_FALSE(answer1.Equals(answer2));
// Same but with the status text removed from the second line of the first
// answer.
answer2 = answer1;
EXPECT_TRUE(answer1.Equals(answer2));
answer1.second_line_.status_text_.reset();
EXPECT_FALSE(answer1.Equals(answer2));
// Same but with the additional text removed from the second line of the first
// answer.
answer2 = answer1;
EXPECT_TRUE(answer1.Equals(answer2));
answer1.second_line_.additional_text_.reset();
EXPECT_FALSE(answer1.Equals(answer2));
}
TEST(SuggestionAnswerTest, EmptyJsonIsInvalid) {
SuggestionAnswer answer;
ASSERT_FALSE(ParseAnswer("", &answer));
}
TEST(SuggestionAnswerTest, MalformedJsonIsInvalid) {
SuggestionAnswer answer;
ASSERT_FALSE(ParseAnswer("} malformed json {", &answer));
}
TEST(SuggestionAnswerTest, TextFieldsRequireBothTextAndType) {
SuggestionAnswer answer;
std::string json =
"{ \"l\": ["
" { \"il\": { \"t\": [{ \"t\": \"text\" }] } }, "
"] }";
ASSERT_FALSE(ParseAnswer(json, &answer));
json =
"{ \"l\": ["
" { \"il\": { \"t\": [{ \"tt\": 8 }] } }, "
"] }";
ASSERT_FALSE(ParseAnswer(json, &answer));
}
TEST(SuggestionAnswerTest, ImageLinesMustContainAtLeastOneTextField) {
SuggestionAnswer answer;
std::string json =
"{ \"l\": ["
" { \"il\": { \"t\": [{ \"t\": \"text\", \"tt\": 8 }, "
" { \"t\": \"moar text\", \"tt\": 0 }], "
" \"i\": { \"d\": \"//example.com/foo.jpg\" } } }, "
" { \"il\": { \"t\": [], "
" \"at\": { \"t\": \"slatfatf\", \"tt\": 42 }, "
" \"st\": { \"t\": \"oh hi, Mark\", \"tt\": 729347 } } } "
"] }";
ASSERT_FALSE(ParseAnswer(json, &answer));
}
TEST(SuggestionAnswerTest, ExactlyTwoLinesRequired) {
SuggestionAnswer answer;
std::string json =
"{ \"l\": ["
" { \"il\": { \"t\": [{ \"t\": \"text\", \"tt\": 8 }] } }, "
"] }";
ASSERT_FALSE(ParseAnswer(json, &answer));
json =
"{ \"l\": ["
" { \"il\": { \"t\": [{ \"t\": \"text\", \"tt\": 8 }] } }, "
" { \"il\": { \"t\": [{ \"t\": \"other text\", \"tt\": 5 }] } } "
"] }";
ASSERT_TRUE(ParseAnswer(json, &answer));
json =
"{ \"l\": ["
" { \"il\": { \"t\": [{ \"t\": \"text\", \"tt\": 8 }] } }, "
" { \"il\": { \"t\": [{ \"t\": \"other text\", \"tt\": 5 }] } } "
" { \"il\": { \"t\": [{ \"t\": \"yet more text\", \"tt\": 13 }] } } "
"] }";
ASSERT_FALSE(ParseAnswer(json, &answer));
}
TEST(SuggestionAnswerTest, URLPresent) {
SuggestionAnswer answer;
std::string json =
"{ \"l\": ["
" { \"il\": { \"t\": [{ \"t\": \"text\", \"tt\": 8 }] } }, "
" { \"il\": { \"t\": [{ \"t\": \"other text\", \"tt\": 5 }], "
" \"i\": { \"d\": \"\" } } } "
"] }";
ASSERT_FALSE(ParseAnswer(json, &answer));
json =
"{ \"l\": ["
" { \"il\": { \"t\": [{ \"t\": \"text\", \"tt\": 8 }] } }, "
" { \"il\": { \"t\": [{ \"t\": \"other text\", \"tt\": 5 }], "
" \"i\": { \"d\": \"https://example.com/foo.jpg\" } } } "
"] }";
ASSERT_TRUE(ParseAnswer(json, &answer));
json =
"{ \"l\": ["
" { \"il\": { \"t\": [{ \"t\": \"text\", \"tt\": 8 }] } }, "
" { \"il\": { \"t\": [{ \"t\": \"other text\", \"tt\": 5 }], "
" \"i\": { \"d\": \"//example.com/foo.jpg\" } } } "
"] }";
ASSERT_TRUE(ParseAnswer(json, &answer));
}
TEST(SuggestionAnswerTest, ValidPropertyValues) {
SuggestionAnswer answer;
std::string json =
"{ \"l\": ["
" { \"il\": { \"t\": [{ \"t\": \"text\", \"tt\": 8 }, "
" { \"t\": \"moar text\", \"tt\": 0 }], "
" \"i\": { \"d\": \"//example.com/foo.jpg\" } } }, "
" { \"il\": { \"t\": [{ \"t\": \"other text\", \"tt\": 5, \"ln\": 3 }], "
" \"at\": { \"t\": \"slatfatf\", \"tt\": 42 }, "
" \"st\": { \"t\": \"oh hi, Mark\", \"tt\": 729347 } } } "
"] }";
ASSERT_TRUE(ParseAnswer(json, &answer));
answer.set_type(420527);
EXPECT_EQ(420527, answer.type());
const SuggestionAnswer::ImageLine& first_line = answer.first_line();
EXPECT_EQ(2U, first_line.text_fields().size());
EXPECT_EQ(base::UTF8ToUTF16("text"), first_line.text_fields()[0].text());
EXPECT_EQ(8, first_line.text_fields()[0].type());
EXPECT_EQ(base::UTF8ToUTF16("moar text"), first_line.text_fields()[1].text());
EXPECT_EQ(0, first_line.text_fields()[1].type());
EXPECT_FALSE(first_line.text_fields()[1].has_num_lines());
EXPECT_EQ(1, first_line.num_text_lines());
EXPECT_FALSE(first_line.additional_text());
EXPECT_FALSE(first_line.status_text());
EXPECT_TRUE(first_line.image_url().is_valid());
EXPECT_EQ(GURL("https://example.com/foo.jpg"), first_line.image_url());
const SuggestionAnswer::ImageLine& second_line = answer.second_line();
EXPECT_EQ(1U, second_line.text_fields().size());
EXPECT_EQ(
base::UTF8ToUTF16("other text"), second_line.text_fields()[0].text());
EXPECT_EQ(5, second_line.text_fields()[0].type());
EXPECT_TRUE(second_line.text_fields()[0].has_num_lines());
EXPECT_EQ(3, second_line.text_fields()[0].num_lines());
EXPECT_EQ(3, second_line.num_text_lines());
EXPECT_TRUE(second_line.additional_text());
EXPECT_EQ(
base::UTF8ToUTF16("slatfatf"), second_line.additional_text()->text());
EXPECT_EQ(42, second_line.additional_text()->type());
EXPECT_TRUE(second_line.status_text());
EXPECT_EQ(
base::UTF8ToUTF16("oh hi, Mark"), second_line.status_text()->text());
EXPECT_EQ(729347, second_line.status_text()->type());
EXPECT_FALSE(second_line.image_url().is_valid());
}
TEST(SuggestionAnswerTest, AddImageURLsTo) {
SuggestionAnswer::URLs urls;
SuggestionAnswer answer;
std::string json =
"{ \"l\": ["
" { \"il\": { \"t\": [{ \"t\": \"text\", \"tt\": 8 }] } }, "
" { \"il\": { \"t\": [{ \"t\": \"other text\", \"tt\": 5 }] } }] }";
ASSERT_TRUE(ParseAnswer(json, &answer));
answer.AddImageURLsTo(&urls);
ASSERT_EQ(0U, urls.size());
{
// Test with the image URL supplied by the "i" (image) param.
base::test::ScopedFeatureList feature_list;
feature_list.InitAndEnableFeature(features::kExperimentalUi);
json =
"{ \"i\": { \"d\": \"https://gstatic.com/foo.png\", \"t\": 3 },"
" \"l\" : ["
" { \"il\": { \"t\": [{ \"t\": \"some text\", \"tt\": 5 }] } },"
" { \"il\": { \"t\": [{ \"t\": \"other text\", \"tt\": 8 }] } }"
" ]}";
ASSERT_TRUE(ParseAnswer(json, &answer));
answer.AddImageURLsTo(&urls);
ASSERT_EQ(1U, urls.size());
EXPECT_EQ(GURL("https://gstatic.com/foo.png"), urls[0]);
urls.clear();
}
// Test with the image URL supplied by the "il" (image line) param.
json =
"{ \"l\" : ["
" { \"il\": { \"t\": [{ \"t\": \"some text\", \"tt\": 5 }] } },"
" { \"il\": { \"t\": [{ \"t\": \"other text\", \"tt\": 8 }],"
" \"i\": { \"d\": \"//gstatic.com/foo.png\", \"t\": 3 }}}]}";
ASSERT_TRUE(ParseAnswer(json, &answer));
answer.AddImageURLsTo(&urls);
ASSERT_EQ(1U, urls.size());
EXPECT_EQ(GURL("https://gstatic.com/foo.png"), urls[0]);
urls.clear();
// Test with image URLs supplied by both the "i" and "il" params. In this
// case, prefer the URL provided by the "i" param because the new answer code
// uses this.
json =
"{ \"i\": { \"d\": \"https://gstatic.com/foo.png\", \"t\": 3 },"
" \"l\" : ["
" { \"il\": { \"t\": [{ \"t\": \"some text\", \"tt\": 5 }] } },"
" { \"il\": { \"t\": [{ \"t\": \"other text\", \"tt\": 8 }],"
" \"i\": { \"d\": \"//gstatic.com/bar.png\", \"t\": 3 }}}"
" ]}";
ASSERT_TRUE(ParseAnswer(json, &answer));
answer.AddImageURLsTo(&urls);
ASSERT_EQ(1U, urls.size());
EXPECT_EQ(GURL("https://gstatic.com/foo.png"), urls[0]);
urls.clear();
// Test with the image URL supplied by both "il" params. In this case, prefer
// the URL in the second line as the first is currently not used.
json =
"{ \"l\" : ["
" { \"il\": { \"t\": [{ \"t\": \"some text\", \"tt\": 5 }],"
" \"i\": { \"d\": \"//gstatic.com/foo.png\" } } }, "
" { \"il\": { \"t\": [{ \"t\": \"other text\", \"tt\": 8 }],"
" \"i\": { \"d\": \"//gstatic.com/bar.jpg\", \"t\": 3 }}}]}";
ASSERT_TRUE(ParseAnswer(json, &answer));
answer.AddImageURLsTo(&urls);
ASSERT_EQ(1U, urls.size());
EXPECT_EQ(GURL("https://gstatic.com/bar.jpg"), urls[0]);
}