blob: df4fa2bb0666c0ae069b0735d43855a20aff3961 [file] [log] [blame]
// Copyright 2018 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/autofill_assistant/browser/controller.h"
#include <memory>
#include <utility>
#include "base/test/mock_callback.h"
#include "base/test/scoped_feature_list.h"
#include "components/autofill_assistant/browser/features.h"
#include "components/autofill_assistant/browser/mock_run_once_callback.h"
#include "components/autofill_assistant/browser/mock_service.h"
#include "components/autofill_assistant/browser/mock_ui_controller.h"
#include "components/autofill_assistant/browser/mock_web_controller.h"
#include "components/autofill_assistant/browser/service.h"
#include "components/autofill_assistant/browser/trigger_context.h"
#include "content/public/test/navigation_simulator.h"
#include "content/public/test/test_browser_context.h"
#include "content/public/test/test_browser_thread_bundle.h"
#include "content/public/test/test_renderer_host.h"
#include "content/public/test/web_contents_tester.h"
#include "testing/gmock/include/gmock/gmock.h"
namespace autofill_assistant {
using ::testing::_;
using ::testing::AnyNumber;
using ::testing::AtLeast;
using ::testing::Contains;
using ::testing::ElementsAre;
using ::testing::Eq;
using ::testing::Field;
using ::testing::Gt;
using ::testing::InSequence;
using ::testing::Invoke;
using ::testing::IsEmpty;
using ::testing::NiceMock;
using ::testing::Not;
using ::testing::Pair;
using ::testing::Pointee;
using ::testing::Return;
using ::testing::ReturnRef;
using ::testing::SaveArg;
using ::testing::Sequence;
using ::testing::SizeIs;
using ::testing::StrEq;
using ::testing::UnorderedElementsAre;
namespace {
class FakeClient : public Client {
public:
explicit FakeClient(UiController* ui_controller)
: ui_controller_(ui_controller) {}
// Implements Client
std::string GetApiKey() override { return ""; }
AccessTokenFetcher* GetAccessTokenFetcher() override { return nullptr; }
autofill::PersonalDataManager* GetPersonalDataManager() override {
return nullptr;
}
std::string GetServerUrl() override { return ""; }
UiController* GetUiController() override { return ui_controller_; }
std::string GetAccountEmailAddress() override { return ""; }
std::string GetLocale() override { return ""; }
std::string GetCountryCode() override { return ""; }
MOCK_METHOD1(Shutdown, void(Metrics::DropOutReason reason));
MOCK_METHOD0(ShowUI, void());
MOCK_METHOD0(DestroyUI, void());
private:
UiController* ui_controller_;
};
} // namespace
class ControllerTest : public content::RenderViewHostTestHarness {
public:
ControllerTest()
: RenderViewHostTestHarness(
base::test::ScopedTaskEnvironment::MainThreadType::UI_MOCK_TIME),
fake_client_(&mock_ui_controller_) {}
~ControllerTest() override {}
void SetUp() override {
RenderViewHostTestHarness::SetUp();
scoped_feature_list_.InitAndEnableFeature(
features::kAutofillAssistantChromeEntry);
auto web_controller = std::make_unique<NiceMock<MockWebController>>();
mock_web_controller_ = web_controller.get();
auto service = std::make_unique<NiceMock<MockService>>();
mock_service_ = service.get();
controller_ = std::make_unique<Controller>(
web_contents(), &fake_client_, thread_bundle()->GetMockTickClock());
controller_->SetWebControllerAndServiceForTest(std::move(web_controller),
std::move(service));
// Fetching scripts succeeds for all URLs, but return nothing.
ON_CALL(*mock_service_, OnGetScriptsForUrl(_, _, _))
.WillByDefault(RunOnceCallback<2>(true, ""));
// Scripts run, but have no actions.
ON_CALL(*mock_service_, OnGetActions(_, _, _, _, _, _))
.WillByDefault(RunOnceCallback<5>(true, ""));
ON_CALL(*mock_service_, OnGetNextActions(_, _, _, _, _))
.WillByDefault(RunOnceCallback<4>(true, ""));
ON_CALL(mock_ui_controller_, OnStateChanged(_))
.WillByDefault(Invoke([this](AutofillAssistantState state) {
states_.emplace_back(state);
}));
ON_CALL(*mock_web_controller_, OnElementCheck(_, _))
.WillByDefault(RunOnceCallback<1>(false));
}
protected:
static SupportedScriptProto* AddRunnableScript(
SupportsScriptResponseProto* response,
const std::string& name_and_path) {
SupportedScriptProto* script = response->add_scripts();
script->set_path(name_and_path);
script->mutable_presentation()->set_name(name_and_path);
return script;
}
static void RunOnce(SupportedScriptProto* proto) {
auto* run_once = proto->mutable_presentation()
->mutable_precondition()
->add_script_status_match();
run_once->set_script(proto->path());
run_once->set_status(SCRIPT_STATUS_NOT_RUN);
}
void SetupScripts(SupportsScriptResponseProto scripts) {
std::string scripts_str;
scripts.SerializeToString(&scripts_str);
EXPECT_CALL(*mock_service_, OnGetScriptsForUrl(_, _, _))
.WillOnce(RunOnceCallback<2>(true, scripts_str));
}
void SetupActionsForScript(const std::string& path,
ActionsResponseProto actions_response) {
std::string actions_response_str;
actions_response.SerializeToString(&actions_response_str);
EXPECT_CALL(*mock_service_, OnGetActions(StrEq("script"), _, _, _, _, _))
.WillOnce(RunOnceCallback<5>(true, actions_response_str));
}
void Start() { Start("http://initialurl.com"); }
void Start(const std::string& url) {
controller_->Start(GURL(url), std::make_unique<TriggerContext>());
}
void SetLastCommittedUrl(const GURL& url) {
content::WebContentsTester::For(web_contents())->SetLastCommittedURL(url);
}
void SimulateNavigateToUrl(const GURL& url) {
content::NavigationSimulator::NavigateAndCommitFromDocument(
url, web_contents()->GetMainFrame());
content::WebContentsTester::For(web_contents())->TestSetIsLoading(false);
}
void SimulateWebContentsFocused() {
controller_->OnWebContentsFocused(nullptr);
}
// Sets up the next call to the service for scripts to return |response|.
void SetNextScriptResponse(const SupportsScriptResponseProto& response) {
std::string response_str;
response.SerializeToString(&response_str);
EXPECT_CALL(*mock_service_, OnGetScriptsForUrl(_, _, _))
.WillOnce(RunOnceCallback<2>(true, response_str));
}
// Sets up all calls to the service for scripts to return |response|.
void SetRepeatedScriptResponse(const SupportsScriptResponseProto& response) {
std::string response_str;
response.SerializeToString(&response_str);
EXPECT_CALL(*mock_service_, OnGetScriptsForUrl(_, _, _))
.WillRepeatedly(RunOnceCallback<2>(true, response_str));
}
UiDelegate* GetUiDelegate() { return controller_.get(); }
// |thread_bundle_| must be the first field, to make sure that everything runs
// in the same task environment.
base::test::ScopedFeatureList scoped_feature_list_;
base::TimeTicks now_;
std::vector<AutofillAssistantState> states_;
MockService* mock_service_;
MockWebController* mock_web_controller_;
NiceMock<FakeClient> fake_client_;
NiceMock<MockUiController> mock_ui_controller_;
std::unique_ptr<Controller> controller_;
};
struct NavigationState {
bool navigating = false;
bool has_errors = false;
bool operator==(const NavigationState& other) const {
return navigating == other.navigating && has_errors == other.has_errors;
}
};
std::ostream& operator<<(std::ostream& out, const NavigationState& state) {
out << "{navigating=" << state.navigating << ","
<< "has_errors=" << state.has_errors << "}";
return out;
}
// A Listener that keeps track of the reported state of the delegate captured
// from OnNavigationStateChanged.
class NavigationStateChangeListener : public ScriptExecutorDelegate::Listener {
public:
explicit NavigationStateChangeListener(ScriptExecutorDelegate* delegate)
: delegate_(delegate) {}
~NavigationStateChangeListener() = default;
void OnNavigationStateChanged() override;
std::vector<NavigationState> events;
private:
ScriptExecutorDelegate* const delegate_;
};
void NavigationStateChangeListener::OnNavigationStateChanged() {
NavigationState state;
state.navigating = delegate_->IsNavigatingToNewDocument();
state.has_errors = delegate_->HasNavigationError();
events.emplace_back(state);
}
TEST_F(ControllerTest, FetchAndRunScripts) {
SupportsScriptResponseProto script_response;
AddRunnableScript(&script_response, "script1");
auto* script2 = AddRunnableScript(&script_response, "script2");
RunOnce(script2);
SetNextScriptResponse(script_response);
testing::InSequence seq;
Start("http://a.example.com/path");
// Offering the choices: script1 and script2
EXPECT_EQ(AutofillAssistantState::AUTOSTART_FALLBACK_PROMPT,
controller_->GetState());
EXPECT_THAT(controller_->GetSuggestions(),
UnorderedElementsAre(Field(&Chip::text, StrEq("script1")),
Field(&Chip::text, StrEq("script2"))));
// Choose script2 and run it successfully.
EXPECT_CALL(*mock_service_, OnGetActions(StrEq("script2"), _, _, _, _, _))
.WillOnce(RunOnceCallback<5>(true, ""));
controller_->SelectSuggestion(1);
// Offering the remaining choice: script1 as script2 can only run once.
EXPECT_EQ(AutofillAssistantState::PROMPT, controller_->GetState());
EXPECT_THAT(controller_->GetSuggestions(),
ElementsAre(Field(&Chip::text, StrEq("script1"))));
}
TEST_F(ControllerTest, NoScripts) {
SupportsScriptResponseProto empty;
SetNextScriptResponse(empty);
Start("http://a.example.com/path");
EXPECT_EQ(AutofillAssistantState::STOPPED, controller_->GetState());
}
TEST_F(ControllerTest, NoRelevantScripts) {
SupportsScriptResponseProto script_response;
AddRunnableScript(&script_response, "no_match")
->mutable_presentation()
->mutable_precondition()
->add_domain("http://otherdomain.com");
SetNextScriptResponse(script_response);
Start("http://a.example.com/path");
EXPECT_EQ(AutofillAssistantState::STOPPED, controller_->GetState());
}
TEST_F(ControllerTest, NoRelevantScriptYet) {
SupportsScriptResponseProto script_response;
AddRunnableScript(&script_response, "no_match_yet")
->mutable_presentation()
->mutable_precondition()
->add_elements_exist()
->add_selectors("#element");
SetNextScriptResponse(script_response);
Start("http://a.example.com/path");
EXPECT_EQ(AutofillAssistantState::STARTING, controller_->GetState());
}
TEST_F(ControllerTest, ReportPromptAndSuggestionsChanged) {
SupportsScriptResponseProto script_response;
AddRunnableScript(&script_response, "script1");
AddRunnableScript(&script_response, "script2");
SetNextScriptResponse(script_response);
EXPECT_CALL(mock_ui_controller_, OnSuggestionsChanged(SizeIs(2)));
Start("http://a.example.com/path");
EXPECT_EQ(AutofillAssistantState::AUTOSTART_FALLBACK_PROMPT,
controller_->GetState());
}
TEST_F(ControllerTest, ClearChipsWhenRunning) {
SupportsScriptResponseProto script_response;
AddRunnableScript(&script_response, "script1");
AddRunnableScript(&script_response, "script2");
SetNextScriptResponse(script_response);
// Discover 2 scripts, one is selected and run (with no chips shown), then the
// same chips are shown.
{
testing::InSequence seq;
// Discover 2 scripts, script1 and script2.
EXPECT_CALL(mock_ui_controller_, OnSuggestionsChanged(SizeIs(2)));
// Set of chips is cleared while running script1.
EXPECT_CALL(mock_ui_controller_, OnSuggestionsChanged(SizeIs(0)));
// This test doesn't specify what happens after that.
EXPECT_CALL(mock_ui_controller_, OnSuggestionsChanged(_))
.Times(AnyNumber());
}
Start("http://a.example.com/path");
controller_->SelectSuggestion(0);
}
TEST_F(ControllerTest, ShowFirstInitialStatusMessage) {
SupportsScriptResponseProto script_response;
AddRunnableScript(&script_response, "script1");
SupportedScriptProto* script2 =
AddRunnableScript(&script_response, "script2");
script2->mutable_presentation()->set_initial_prompt("script2 prompt");
script2->mutable_presentation()->set_priority(10);
SupportedScriptProto* script3 =
AddRunnableScript(&script_response, "script3");
script3->mutable_presentation()->set_initial_prompt("script3 prompt");
script3->mutable_presentation()->set_priority(5);
SupportedScriptProto* script4 =
AddRunnableScript(&script_response, "script4");
script4->mutable_presentation()->set_initial_prompt("script4 prompt");
script4->mutable_presentation()->set_priority(8);
SetNextScriptResponse(script_response);
Start("http://a.example.com/path");
EXPECT_THAT(controller_->GetSuggestions(), SizeIs(4));
// Script3, with higher priority (lower number), wins.
EXPECT_EQ("script3 prompt", controller_->GetStatusMessage());
}
TEST_F(ControllerTest, Stop) {
SupportsScriptResponseProto script_response;
AddRunnableScript(&script_response, "stop");
SetNextScriptResponse(script_response);
ActionsResponseProto actions_response;
actions_response.add_actions()->mutable_stop();
std::string actions_response_str;
actions_response.SerializeToString(&actions_response_str);
EXPECT_CALL(*mock_service_, OnGetActions(StrEq("stop"), _, _, _, _, _))
.WillOnce(RunOnceCallback<5>(true, actions_response_str));
Start();
ASSERT_THAT(controller_->GetSuggestions(), SizeIs(1));
testing::InSequence seq;
EXPECT_CALL(fake_client_, Shutdown(Metrics::SCRIPT_SHUTDOWN));
controller_->SelectSuggestion(0);
// Simulates Client::Shutdown(SCRIPT_SHUTDOWN)
EXPECT_CALL(mock_ui_controller_, WillShutdown(Metrics::SCRIPT_SHUTDOWN));
controller_->WillShutdown(Metrics::SCRIPT_SHUTDOWN);
}
TEST_F(ControllerTest, Reset) {
// 1. Fetch scripts for URL, which in contains a single "reset" script.
SupportsScriptResponseProto script_response;
auto* reset_script = AddRunnableScript(&script_response, "reset");
RunOnce(reset_script);
std::string script_response_str;
script_response.SerializeToString(&script_response_str);
EXPECT_CALL(*mock_service_, OnGetScriptsForUrl(_, _, _))
.WillRepeatedly(RunOnceCallback<2>(true, script_response_str));
Start("http://a.example.com/path");
EXPECT_THAT(controller_->GetSuggestions(),
ElementsAre(Field(&Chip::text, StrEq("reset"))));
// 2. Execute the "reset" script, which contains a reset action.
ActionsResponseProto actions_response;
actions_response.add_actions()->mutable_reset();
std::string actions_response_str;
actions_response.SerializeToString(&actions_response_str);
EXPECT_CALL(*mock_service_, OnGetActions(StrEq("reset"), _, _, _, _, _))
.WillOnce(RunOnceCallback<5>(true, actions_response_str));
controller_->GetClientMemory()->set_selected_card(
std::make_unique<autofill::CreditCard>());
EXPECT_TRUE(controller_->GetClientMemory()->has_selected_card());
controller_->SelectSuggestion(0);
// Resetting should have cleared the client memory
EXPECT_FALSE(controller_->GetClientMemory()->has_selected_card());
// The reset script should be available again, even though it's marked
// RunOnce, as the script state should have been cleared as well.
EXPECT_THAT(controller_->GetSuggestions(),
ElementsAre(Field(&Chip::text, StrEq("reset"))));
}
TEST_F(ControllerTest, RefreshScriptWhenDomainChanges) {
EXPECT_CALL(*mock_service_,
OnGetScriptsForUrl(Eq(GURL("http://a.example.com/path1")), _, _))
.WillOnce(RunOnceCallback<2>(true, ""));
EXPECT_CALL(*mock_service_,
OnGetScriptsForUrl(Eq(GURL("http://b.example.com/path1")), _, _))
.WillOnce(RunOnceCallback<2>(true, ""));
Start("http://a.example.com/path1");
SimulateNavigateToUrl(GURL("http://a.example.com/path2"));
SimulateNavigateToUrl(GURL("http://b.example.com/path1"));
SimulateNavigateToUrl(GURL("http://b.example.com/path2"));
}
TEST_F(ControllerTest, ForwardParameters) {
EXPECT_CALL(*mock_service_,
OnGetScriptsForUrl(_,
Field(&TriggerContext::script_parameters,
Contains(Pair("a", "b"))),
_))
.WillOnce(RunOnceCallback<2>(true, ""));
GURL initialUrl("http://example.com/");
std::unique_ptr<TriggerContext> context(new TriggerContext);
context->script_parameters["a"] = "b";
controller_->Start(initialUrl, std::move(context));
}
TEST_F(ControllerTest, Autostart) {
SupportsScriptResponseProto script_response;
AddRunnableScript(&script_response, "runnable");
AddRunnableScript(&script_response, "autostart")
->mutable_presentation()
->set_autostart(true);
AddRunnableScript(&script_response, "alsorunnable");
SetNextScriptResponse(script_response);
EXPECT_CALL(*mock_service_, OnGetActions(StrEq("autostart"), _, _, _, _, _))
.WillOnce(RunOnceCallback<5>(true, ""));
Start("http://a.example.com/path");
}
TEST_F(ControllerTest, AutostartFirstInterrupt) {
SupportsScriptResponseProto script_response;
AddRunnableScript(&script_response, "runnable");
auto* interrupt1 =
AddRunnableScript(&script_response, "autostart interrupt 1");
interrupt1->mutable_presentation()->set_interrupt(true);
interrupt1->mutable_presentation()->set_priority(1);
interrupt1->mutable_presentation()->set_autostart(true);
auto* interrupt2 =
AddRunnableScript(&script_response, "autostart interrupt 2");
interrupt2->mutable_presentation()->set_interrupt(true);
interrupt2->mutable_presentation()->set_priority(2);
interrupt2->mutable_presentation()->set_autostart(true);
SetNextScriptResponse(script_response);
EXPECT_CALL(*mock_service_,
OnGetActions(StrEq("autostart interrupt 1"), _, _, _, _, _))
.WillOnce(RunOnceCallback<5>(false, ""));
// The script fails, ending the flow. What matters is that the correct
// expectation is met.
Start("http://a.example.com/path");
}
TEST_F(ControllerTest, InterruptThenAutostart) {
SupportsScriptResponseProto script_response;
AddRunnableScript(&script_response, "runnable");
auto* interrupt = AddRunnableScript(&script_response, "autostart interrupt");
interrupt->mutable_presentation()->set_interrupt(true);
interrupt->mutable_presentation()->set_autostart(true);
RunOnce(interrupt);
auto* autostart = AddRunnableScript(&script_response, "autostart");
autostart->mutable_presentation()->set_autostart(true);
RunOnce(autostart);
SetRepeatedScriptResponse(script_response);
{
testing::InSequence seq;
EXPECT_CALL(*mock_service_,
OnGetActions(StrEq("autostart interrupt"), _, _, _, _, _));
EXPECT_CALL(*mock_service_,
OnGetActions(StrEq("autostart"), _, _, _, _, _));
}
Start("http://a.example.com/path");
}
TEST_F(ControllerTest, AutostartIsNotPassedToTheUi) {
SupportsScriptResponseProto script_response;
auto* autostart = AddRunnableScript(&script_response, "runnable");
autostart->mutable_presentation()->set_autostart(true);
RunOnce(autostart);
SetRepeatedScriptResponse(script_response);
EXPECT_CALL(mock_ui_controller_, OnSuggestionsChanged(SizeIs(0u)))
.Times(AnyNumber());
EXPECT_CALL(mock_ui_controller_, OnSuggestionsChanged(SizeIs(Gt(0u))))
.Times(0);
Start("http://a.example.com/path");
EXPECT_THAT(controller_->GetSuggestions(), SizeIs(0));
}
TEST_F(ControllerTest, InitialUrlLoads) {
GURL initialUrl("http://a.example.com/path");
EXPECT_CALL(*mock_service_, OnGetScriptsForUrl(Eq(initialUrl), _, _))
.WillOnce(RunOnceCallback<2>(true, ""));
controller_->Start(initialUrl, std::make_unique<TriggerContext>());
}
TEST_F(ControllerTest, CookieExperimentEnabled) {
GURL initialUrl("http://a.example.com/path");
// TODO(crbug.com/806868): Extend this test once the cookie information is
// passed to the initial request. Currently the public controller API does not
// yet allow proper testing.
EXPECT_CALL(*mock_service_, OnGetScriptsForUrl(Eq(initialUrl), _, _))
.WillOnce(RunOnceCallback<2>(true, ""));
std::unique_ptr<TriggerContext> trigger_context(new TriggerContext);
trigger_context->script_parameters.insert(std::make_pair("EXP_COOKIE", "1"));
controller_->Start(initialUrl, std::move(trigger_context));
// TODO(crbug.com): Make IsCookieExperimentEnabled private and remove this
// test when we pass the cookie data along in the initial request so that it
// can be tested.
EXPECT_TRUE(controller_->IsCookieExperimentEnabled());
}
TEST_F(ControllerTest, ProgressIncreasesAtStart) {
EXPECT_EQ(0, controller_->GetProgress());
EXPECT_CALL(mock_ui_controller_, OnProgressChanged(10));
Start();
EXPECT_EQ(10, controller_->GetProgress());
}
TEST_F(ControllerTest, SetProgress) {
Start();
EXPECT_CALL(mock_ui_controller_, OnProgressChanged(20));
controller_->SetProgress(20);
EXPECT_EQ(20, controller_->GetProgress());
}
TEST_F(ControllerTest, IgnoreProgressDecreases) {
Start();
EXPECT_CALL(mock_ui_controller_, OnProgressChanged(Not(15)))
.Times(AnyNumber());
controller_->SetProgress(20);
controller_->SetProgress(15);
EXPECT_EQ(20, controller_->GetProgress());
}
TEST_F(ControllerTest, StateChanges) {
EXPECT_EQ(AutofillAssistantState::INACTIVE, GetUiDelegate()->GetState());
SupportsScriptResponseProto script_response;
auto* script1 = AddRunnableScript(&script_response, "script1");
RunOnce(script1);
auto* script2 = AddRunnableScript(&script_response, "script2");
RunOnce(script2);
SetNextScriptResponse(script_response);
Start("http://a.example.com/path");
EXPECT_THAT(states_,
ElementsAre(AutofillAssistantState::STARTING,
AutofillAssistantState::AUTOSTART_FALLBACK_PROMPT));
// Run script1: State should become RUNNING, as there's another script, then
// go back to prompt to propose that script.
states_.clear();
ASSERT_THAT(controller_->GetSuggestions(), SizeIs(2));
controller_->SelectSuggestion(0);
EXPECT_EQ(AutofillAssistantState::PROMPT, GetUiDelegate()->GetState());
EXPECT_THAT(states_, ElementsAre(AutofillAssistantState::RUNNING,
AutofillAssistantState::PROMPT));
// Run script2: State should become STOPPED, as there are no more runnable
// scripts.
states_.clear();
ASSERT_THAT(controller_->GetSuggestions(), SizeIs(1));
controller_->SelectSuggestion(0);
EXPECT_EQ(AutofillAssistantState::STOPPED, GetUiDelegate()->GetState());
EXPECT_THAT(states_, ElementsAre(AutofillAssistantState::RUNNING,
AutofillAssistantState::PROMPT,
AutofillAssistantState::STOPPED));
// The cancel button is removed.
EXPECT_TRUE(controller_->GetActions().empty());
}
TEST_F(ControllerTest, ShowUIWhenStarting) {
EXPECT_CALL(fake_client_, ShowUI());
Start();
}
TEST_F(ControllerTest, ShowUIWhenContentsFocused) {
SimulateWebContentsFocused(); // must not call ShowUI
testing::InSequence seq;
EXPECT_CALL(fake_client_, ShowUI());
SupportsScriptResponseProto script_response;
AddRunnableScript(&script_response, "script1");
SetNextScriptResponse(script_response);
Start(); // must call ShowUI
EXPECT_CALL(fake_client_, ShowUI());
SimulateWebContentsFocused(); // must call ShowUI
controller_->OnFatalError("test", Metrics::TAB_CHANGED);
SimulateWebContentsFocused(); // must not call ShowUI
}
TEST_F(ControllerTest, KeepCheckingForElement) {
SupportsScriptResponseProto script_response;
AddRunnableScript(&script_response, "no_match_yet")
->mutable_presentation()
->mutable_precondition()
->add_elements_exist()
->add_selectors("#element");
SetNextScriptResponse(script_response);
Start("http://a.example.com/path");
// No scripts yet; the element doesn't exit.
EXPECT_EQ(AutofillAssistantState::STARTING, controller_->GetState());
for (int i = 0; i < 3; i++) {
thread_bundle()->FastForwardBy(base::TimeDelta::FromSeconds(1));
EXPECT_EQ(AutofillAssistantState::STARTING, controller_->GetState());
}
EXPECT_CALL(*mock_web_controller_, OnElementCheck(_, _))
.WillRepeatedly(RunOnceCallback<1>(true));
thread_bundle()->FastForwardBy(base::TimeDelta::FromSeconds(1));
EXPECT_EQ(AutofillAssistantState::AUTOSTART_FALLBACK_PROMPT,
controller_->GetState());
}
TEST_F(ControllerTest, ScriptTimeoutError) {
// Wait for #element to show up for will_never_match. After 25s, execute the
// script on_timeout_error.
SupportsScriptResponseProto script_response;
AddRunnableScript(&script_response, "will_never_match")
->mutable_presentation()
->mutable_precondition()
->add_elements_exist()
->add_selectors("#element");
script_response.mutable_script_timeout_error()->set_timeout_ms(30000);
script_response.mutable_script_timeout_error()->set_script_path(
"on_timeout_error");
SetNextScriptResponse(script_response);
// on_timeout_error stops everything with a custom error message.
ActionsResponseProto on_timeout_error;
on_timeout_error.add_actions()->mutable_tell()->set_message("I give up");
on_timeout_error.add_actions()->mutable_stop();
std::string on_timeout_error_str;
on_timeout_error.SerializeToString(&on_timeout_error_str);
EXPECT_CALL(*mock_service_,
OnGetActions(StrEq("on_timeout_error"), _, _, _, _, _))
.WillOnce(RunOnceCallback<5>(true, on_timeout_error_str));
Start("http://a.example.com/path");
for (int i = 0; i < 30; i++) {
EXPECT_EQ(AutofillAssistantState::STARTING, controller_->GetState());
thread_bundle()->FastForwardBy(base::TimeDelta::FromSeconds(1));
}
EXPECT_EQ(AutofillAssistantState::STOPPED, controller_->GetState());
EXPECT_EQ("I give up", controller_->GetStatusMessage());
}
TEST_F(ControllerTest, ScriptTimeoutWarning) {
// Wait for #element to show up for will_never_match. After 10s, execute the
// script on_timeout_error.
SupportsScriptResponseProto script_response;
AddRunnableScript(&script_response, "will_never_match")
->mutable_presentation()
->mutable_precondition()
->add_elements_exist()
->add_selectors("#element");
script_response.mutable_script_timeout_error()->set_timeout_ms(4000);
script_response.mutable_script_timeout_error()->set_script_path(
"on_timeout_error");
SetNextScriptResponse(script_response);
// on_timeout_error displays an error message and terminates
ActionsResponseProto on_timeout_error;
on_timeout_error.add_actions()->mutable_tell()->set_message("This is slow");
std::string on_timeout_error_str;
on_timeout_error.SerializeToString(&on_timeout_error_str);
EXPECT_CALL(*mock_service_,
OnGetActions(StrEq("on_timeout_error"), _, _, _, _, _))
.WillOnce(RunOnceCallback<5>(true, on_timeout_error_str));
Start("http://a.example.com/path");
// Warning after 4s, script succeeds and the client continues to wait.
for (int i = 0; i < 4; i++) {
EXPECT_EQ(AutofillAssistantState::STARTING, controller_->GetState());
thread_bundle()->FastForwardBy(base::TimeDelta::FromSeconds(1));
}
EXPECT_EQ(AutofillAssistantState::STARTING, controller_->GetState());
EXPECT_EQ("This is slow", controller_->GetStatusMessage());
for (int i = 0; i < 10; i++) {
EXPECT_EQ(AutofillAssistantState::STARTING, controller_->GetState());
thread_bundle()->FastForwardBy(base::TimeDelta::FromSeconds(1));
}
}
TEST_F(ControllerTest, SuccessfulNavigation) {
EXPECT_FALSE(controller_->IsNavigatingToNewDocument());
EXPECT_FALSE(controller_->HasNavigationError());
NavigationStateChangeListener listener(controller_.get());
controller_->AddListener(&listener);
content::NavigationSimulator::NavigateAndCommitFromDocument(
GURL("http://initialurl.com"), web_contents()->GetMainFrame());
controller_->RemoveListener(&listener);
EXPECT_FALSE(controller_->IsNavigatingToNewDocument());
EXPECT_FALSE(controller_->HasNavigationError());
EXPECT_THAT(listener.events, ElementsAre(NavigationState{true, false},
NavigationState{false, false}));
}
TEST_F(ControllerTest, FailedNavigation) {
EXPECT_FALSE(controller_->IsNavigatingToNewDocument());
EXPECT_FALSE(controller_->HasNavigationError());
NavigationStateChangeListener listener(controller_.get());
controller_->AddListener(&listener);
content::NavigationSimulator::NavigateAndFailFromDocument(
GURL("http://initialurl.com"), net::ERR_CONNECTION_TIMED_OUT,
web_contents()->GetMainFrame());
controller_->RemoveListener(&listener);
EXPECT_FALSE(controller_->IsNavigatingToNewDocument());
EXPECT_TRUE(controller_->HasNavigationError());
EXPECT_THAT(listener.events, ElementsAre(NavigationState{true, false},
NavigationState{false, true}));
}
TEST_F(ControllerTest, NavigationWithRedirects) {
EXPECT_FALSE(controller_->IsNavigatingToNewDocument());
EXPECT_FALSE(controller_->HasNavigationError());
NavigationStateChangeListener listener(controller_.get());
controller_->AddListener(&listener);
std::unique_ptr<content::NavigationSimulator> simulator =
content::NavigationSimulator::CreateRendererInitiated(
GURL("http://original.example.com/"), web_contents()->GetMainFrame());
simulator->SetTransition(ui::PAGE_TRANSITION_LINK);
simulator->Start();
EXPECT_TRUE(controller_->IsNavigatingToNewDocument());
EXPECT_FALSE(controller_->HasNavigationError());
simulator->Redirect(GURL("http://redirect.example.com/"));
EXPECT_TRUE(controller_->IsNavigatingToNewDocument());
EXPECT_FALSE(controller_->HasNavigationError());
simulator->Commit();
EXPECT_FALSE(controller_->IsNavigatingToNewDocument());
EXPECT_FALSE(controller_->HasNavigationError());
controller_->RemoveListener(&listener);
// Redirection should not be reported as a state change.
EXPECT_THAT(listener.events, ElementsAre(NavigationState{true, false},
NavigationState{false, false}));
}
TEST_F(ControllerTest, EventuallySuccessfulNavigation) {
EXPECT_FALSE(controller_->IsNavigatingToNewDocument());
EXPECT_FALSE(controller_->HasNavigationError());
NavigationStateChangeListener listener(controller_.get());
controller_->AddListener(&listener);
content::NavigationSimulator::NavigateAndFailFromDocument(
GURL("http://initialurl.com"), net::ERR_CONNECTION_TIMED_OUT,
web_contents()->GetMainFrame());
content::NavigationSimulator::NavigateAndCommitFromDocument(
GURL("http://initialurl.com"), web_contents()->GetMainFrame());
controller_->RemoveListener(&listener);
EXPECT_FALSE(controller_->IsNavigatingToNewDocument());
EXPECT_FALSE(controller_->HasNavigationError());
EXPECT_THAT(listener.events,
ElementsAre(
// 1st navigation starts
NavigationState{true, false},
// 1st navigation fails
NavigationState{false, true},
// 2nd navigation starts, while in error state
NavigationState{true, true},
// 2nd navigation succeeds
NavigationState{false, false}));
}
TEST_F(ControllerTest, RemoveListener) {
NavigationStateChangeListener listener(controller_.get());
controller_->AddListener(&listener);
content::NavigationSimulator::NavigateAndCommitFromDocument(
GURL("http://initialurl.com"), web_contents()->GetMainFrame());
listener.events.clear();
controller_->RemoveListener(&listener);
content::NavigationSimulator::NavigateAndFailFromDocument(
GURL("http://initialurl.com"), net::ERR_CONNECTION_TIMED_OUT,
web_contents()->GetMainFrame());
content::NavigationSimulator::NavigateAndCommitFromDocument(
GURL("http://initialurl.com"), web_contents()->GetMainFrame());
EXPECT_THAT(listener.events, IsEmpty());
}
TEST_F(ControllerTest, WaitForNavigationActionTimesOut) {
// A single script, with a wait_for_navigation action
SupportsScriptResponseProto script_response;
AddRunnableScript(&script_response, "script");
SetupScripts(script_response);
ActionsResponseProto actions_response;
actions_response.add_actions()->mutable_expect_navigation();
auto* action = actions_response.add_actions()->mutable_wait_for_navigation();
action->set_timeout_ms(1000);
SetupActionsForScript("script", actions_response);
std::vector<ProcessedActionProto> processed_actions_capture;
EXPECT_CALL(*mock_service_, OnGetNextActions(_, _, _, _, _))
.WillOnce(DoAll(SaveArg<3>(&processed_actions_capture),
RunOnceCallback<4>(true, "")));
Start("http://a.example.com/path");
EXPECT_THAT(controller_->GetSuggestions(), SizeIs(1));
// Start script, which waits for some navigation event to happen after the
// expect_navigation action has run..
controller_->SelectSuggestion(0);
// No navigation event happened within the action timeout and the script ends.
EXPECT_THAT(processed_actions_capture, SizeIs(0));
thread_bundle()->FastForwardBy(base::TimeDelta::FromSeconds(1));
ASSERT_THAT(processed_actions_capture, SizeIs(2));
EXPECT_EQ(ACTION_APPLIED, processed_actions_capture[0].status());
EXPECT_EQ(TIMED_OUT, processed_actions_capture[1].status());
}
TEST_F(ControllerTest, WaitForNavigationActionStartWithinTimeout) {
// A single script, with a wait_for_navigation action
SupportsScriptResponseProto script_response;
AddRunnableScript(&script_response, "script");
SetupScripts(script_response);
ActionsResponseProto actions_response;
actions_response.add_actions()->mutable_expect_navigation();
auto* action = actions_response.add_actions()->mutable_wait_for_navigation();
action->set_timeout_ms(1000);
SetupActionsForScript("script", actions_response);
std::vector<ProcessedActionProto> processed_actions_capture;
EXPECT_CALL(*mock_service_, OnGetNextActions(_, _, _, _, _))
.WillOnce(DoAll(SaveArg<3>(&processed_actions_capture),
RunOnceCallback<4>(true, "")));
Start("http://a.example.com/path");
EXPECT_THAT(controller_->GetSuggestions(), SizeIs(1));
// Start script, which waits for some navigation event to happen after the
// expect_navigation action has run..
controller_->SelectSuggestion(0);
// Navigation starts, but does not end, within the timeout.
EXPECT_THAT(processed_actions_capture, SizeIs(0));
std::unique_ptr<content::NavigationSimulator> simulator =
content::NavigationSimulator::CreateRendererInitiated(
GURL("http://a.example.com/path"), web_contents()->GetMainFrame());
simulator->SetTransition(ui::PAGE_TRANSITION_LINK);
simulator->Start();
thread_bundle()->FastForwardBy(base::TimeDelta::FromSeconds(1));
// Navigation finishes and the script ends.
EXPECT_THAT(processed_actions_capture, SizeIs(0));
simulator->Commit();
ASSERT_THAT(processed_actions_capture, SizeIs(2));
EXPECT_EQ(ACTION_APPLIED, processed_actions_capture[0].status());
EXPECT_EQ(ACTION_APPLIED, processed_actions_capture[1].status());
}
} // namespace autofill_assistant