| // 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/script_executor.h" |
| |
| #include <string> |
| #include <utility> |
| |
| #include "base/bind.h" |
| #include "base/callback.h" |
| #include "base/single_thread_task_runner.h" |
| #include "base/threading/thread_task_runner_handle.h" |
| #include "components/autofill_assistant/browser/protocol_utils.h" |
| #include "components/autofill_assistant/browser/service.h" |
| #include "components/autofill_assistant/browser/ui_controller.h" |
| #include "components/autofill_assistant/browser/web_controller.h" |
| |
| namespace autofill_assistant { |
| |
| ScriptExecutor::ScriptExecutor(Script* script, ScriptExecutorDelegate* delegate) |
| : script_(script), delegate_(delegate), weak_ptr_factory_(this) { |
| DCHECK(script_); |
| DCHECK(delegate_); |
| } |
| ScriptExecutor::~ScriptExecutor() {} |
| |
| void ScriptExecutor::Run(RunScriptCallback callback) { |
| callback_ = std::move(callback); |
| DCHECK(delegate_->GetService()); |
| |
| delegate_->GetService()->GetActions( |
| script_->path, base::BindOnce(&ScriptExecutor::OnGetActions, |
| weak_ptr_factory_.GetWeakPtr())); |
| } |
| |
| void ScriptExecutor::ShowStatusMessage(const std::string& message) { |
| delegate_->GetUiController()->ShowStatusMessage(message); |
| } |
| |
| void ScriptExecutor::ClickElement(const std::vector<std::string>& selectors, |
| base::OnceCallback<void(bool)> callback) { |
| delegate_->GetWebController()->ClickElement(selectors, std::move(callback)); |
| } |
| |
| void ScriptExecutor::ElementExists(const std::vector<std::string>& selectors, |
| base::OnceCallback<void(bool)> callback) { |
| delegate_->GetWebController()->ElementExists(selectors, std::move(callback)); |
| } |
| |
| void ScriptExecutor::ChooseAddress( |
| base::OnceCallback<void(const std::string&)> callback) { |
| delegate_->GetUiController()->ChooseAddress(std::move(callback)); |
| } |
| |
| void ScriptExecutor::FillAddressForm(const std::string& guid, |
| const std::vector<std::string>& selectors, |
| base::OnceCallback<void(bool)> callback) { |
| delegate_->GetWebController()->FillAddressForm(guid, selectors, |
| std::move(callback)); |
| } |
| |
| void ScriptExecutor::ChooseCard( |
| base::OnceCallback<void(const std::string&)> callback) { |
| delegate_->GetUiController()->ChooseCard(std::move(callback)); |
| } |
| |
| void ScriptExecutor::FillCardForm(const std::string& guid, |
| const std::vector<std::string>& selectors, |
| base::OnceCallback<void(bool)> callback) { |
| delegate_->GetWebController()->FillCardForm(guid, selectors, |
| std::move(callback)); |
| } |
| |
| ClientMemory* ScriptExecutor::GetClientMemory() { |
| return delegate_->GetClientMemory(); |
| } |
| |
| void ScriptExecutor::OnGetActions(bool result, const std::string& response) { |
| if (!result) { |
| std::move(callback_).Run(false); |
| return; |
| } |
| processed_actions_.clear(); |
| actions_.clear(); |
| |
| bool parse_result = |
| ProtocolUtils::ParseActions(response, &last_server_payload_, &actions_); |
| if (!parse_result) { |
| std::move(callback_).Run(false); |
| return; |
| } |
| |
| if (actions_.empty()) { |
| // Finished executing the script if there are no more actions. |
| std::move(callback_).Run(true); |
| return; |
| } |
| ProcessNextAction(); |
| } |
| |
| void ScriptExecutor::ProcessNextAction() { |
| if (actions_.empty()) { |
| // Request more actions to execute. |
| GetNextActions(); |
| return; |
| } |
| |
| std::unique_ptr<Action> action = std::move(actions_.front()); |
| actions_.pop_front(); |
| int delay_ms = action->proto().action_delay_ms(); |
| if (delay_ms > 0) { |
| base::ThreadTaskRunnerHandle::Get()->PostDelayedTask( |
| FROM_HERE, |
| base::BindOnce(&ScriptExecutor::ProcessAction, |
| weak_ptr_factory_.GetWeakPtr(), std::move(action)), |
| base::TimeDelta::FromMilliseconds(delay_ms)); |
| } else { |
| ProcessAction(std::move(action)); |
| } |
| } |
| |
| void ScriptExecutor::ProcessAction(std::unique_ptr<Action> action) { |
| action->ProcessAction( |
| this, base::BindOnce(&ScriptExecutor::OnProcessedAction, |
| weak_ptr_factory_.GetWeakPtr(), std::move(action))); |
| } |
| |
| void ScriptExecutor::GetNextActions() { |
| delegate_->GetService()->GetNextActions( |
| last_server_payload_, processed_actions_, |
| base::BindOnce(&ScriptExecutor::OnGetActions, |
| weak_ptr_factory_.GetWeakPtr())); |
| } |
| |
| void ScriptExecutor::OnProcessedAction(std::unique_ptr<Action> action, |
| bool success) { |
| processed_actions_.emplace_back(); |
| ProcessedActionProto* proto = &processed_actions_.back(); |
| proto->mutable_action()->MergeFrom(action->proto()); |
| proto->set_status(success ? ProcessedActionStatus::ACTION_APPLIED |
| : ProcessedActionStatus::OTHER_ACTION_STATUS); |
| if (!success) { |
| // Report error immediately, interrupting action processing. |
| GetNextActions(); |
| return; |
| } |
| ProcessNextAction(); |
| } |
| |
| } // namespace autofill_assistant |