blob: bbbe7a25edaa0d77d5ded426b4e86c497a241182 [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 "chrome/browser/android/autofill_assistant/client_android.h"
#include <utility>
#include <vector>
#include "base/android/jni_android.h"
#include "base/android/jni_array.h"
#include "base/android/jni_string.h"
#include "base/android/locale_utils.h"
#include "base/bind.h"
#include "base/command_line.h"
#include "base/metrics/field_trial_params.h"
#include "base/no_destructor.h"
#include "base/task/post_task.h"
#include "base/time/default_tick_clock.h"
#include "chrome/android/features/autofill_assistant/jni_headers/AutofillAssistantClient_jni.h"
#include "chrome/browser/android/chrome_feature_list.h"
#include "chrome/browser/autofill/android/personal_data_manager_android.h"
#include "chrome/browser/autofill/personal_data_manager_factory.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/profiles/profile_manager.h"
#include "chrome/browser/signin/identity_manager_factory.h"
#include "chrome/common/channel_info.h"
#include "components/autofill_assistant/browser/access_token_fetcher.h"
#include "components/autofill_assistant/browser/controller.h"
#include "components/autofill_assistant/browser/features.h"
#include "components/signin/core/browser/account_info.h"
#include "components/signin/public/identity_manager/identity_manager.h"
#include "components/version_info/channel.h"
#include "content/public/browser/browser_task_traits.h"
#include "content/public/browser/browser_thread.h"
#include "content/public/browser/web_contents.h"
#include "google_apis/google_api_keys.h"
#include "url/gurl.h"
using base::android::AttachCurrentThread;
using base::android::JavaParamRef;
using base::android::JavaRef;
namespace autofill_assistant {
namespace switches {
const char* const kAutofillAssistantServerKey = "autofill-assistant-key";
const char* const kAutofillAssistantUrl = "autofill-assistant-url";
} // namespace switches
namespace {
const char* const kDefaultAutofillAssistantServerUrl =
"https://automate-pa.googleapis.com";
// Fills a map from two Java arrays of strings of the same length.
void FillParametersFromJava(JNIEnv* env,
const JavaRef<jobjectArray>& names,
const JavaRef<jobjectArray>& values,
std::map<std::string, std::string>* parameters) {
std::vector<std::string> names_vector;
base::android::AppendJavaStringArrayToStringVector(env, names, &names_vector);
std::vector<std::string> values_vector;
base::android::AppendJavaStringArrayToStringVector(env, values,
&values_vector);
DCHECK_EQ(names_vector.size(), values_vector.size());
for (size_t i = 0; i < names_vector.size(); ++i) {
parameters->insert(std::make_pair(names_vector[i], values_vector[i]));
}
}
} // namespace
static base::android::ScopedJavaLocalRef<jobject>
JNI_AutofillAssistantClient_FromWebContents(
JNIEnv* env,
const base::android::JavaParamRef<jobject>& jweb_contents) {
auto* web_contents = content::WebContents::FromJavaWebContents(jweb_contents);
ClientAndroid::CreateForWebContents(web_contents);
return ClientAndroid::FromWebContents(web_contents)->GetJavaObject();
}
ClientAndroid::ClientAndroid(content::WebContents* web_contents)
: web_contents_(web_contents),
java_object_(Java_AutofillAssistantClient_create(
AttachCurrentThread(),
reinterpret_cast<intptr_t>(this))),
weak_ptr_factory_(this) {
server_url_ = base::CommandLine::ForCurrentProcess()->GetSwitchValueASCII(
switches::kAutofillAssistantUrl);
if (server_url_.empty()) {
server_url_ = kDefaultAutofillAssistantServerUrl;
}
}
ClientAndroid::~ClientAndroid() {
if (controller_ != nullptr) {
// In the case of an unexpected closing of the activity or tab, controller_
// will not yet have been cleaned up (since that happens when a web
// contents object gets destroyed).
Metrics::RecordDropOut(Metrics::CONTENT_DESTROYED);
}
Java_AutofillAssistantClient_clearNativePtr(AttachCurrentThread(),
java_object_);
}
base::android::ScopedJavaLocalRef<jobject> ClientAndroid::GetJavaObject() {
return base::android::ScopedJavaLocalRef<jobject>(java_object_);
}
void ClientAndroid::Start(JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
const JavaParamRef<jstring>& jinitial_url,
const JavaParamRef<jstring>& jexperiment_ids,
const JavaParamRef<jobjectArray>& parameterNames,
const JavaParamRef<jobjectArray>& parameterValues,
const JavaParamRef<jobject>& joverlay_coordinator) {
CreateController();
// If an overlay is already shown, then show the rest of the UI.
if (joverlay_coordinator) {
CreateUI(joverlay_coordinator);
}
GURL initial_url(base::android::ConvertJavaStringToUTF8(env, jinitial_url));
std::map<std::string, std::string> parameters;
FillParametersFromJava(env, parameterNames, parameterValues, &parameters);
controller_->Start(initial_url, TriggerContext::Create(
std::move(parameters),
base::android::ConvertJavaStringToUTF8(
env, jexperiment_ids)));
}
void ClientAndroid::DestroyUI(
JNIEnv* env,
const base::android::JavaParamRef<jobject>& jcaller) {
DestroyUI();
}
void ClientAndroid::TransferUITo(
JNIEnv* env,
const base::android::JavaParamRef<jobject>& jcaller,
const base::android::JavaParamRef<jobject>& jother_web_contents) {
if (!ui_controller_android_)
return;
auto ui_ptr = std::move(ui_controller_android_);
// From this point on, the UIController, in ui_ptr, is either transferred or
// deleted.
if (!jother_web_contents)
return;
auto* other_web_contents =
content::WebContents::FromJavaWebContents(jother_web_contents);
DCHECK_NE(other_web_contents, web_contents_);
ClientAndroid* other_client =
ClientAndroid::FromWebContents(other_web_contents);
if (!other_client || !other_client->NeedsUI())
return;
other_client->SetUI(std::move(ui_ptr));
}
base::android::ScopedJavaLocalRef<jstring> ClientAndroid::GetPrimaryAccountName(
JNIEnv* env,
const JavaParamRef<jobject>& jcaller) {
CoreAccountInfo account_info =
IdentityManagerFactory::GetForProfile(
Profile::FromBrowserContext(web_contents_->GetBrowserContext()))
->GetPrimaryAccountInfo();
return base::android::ConvertUTF8ToJavaString(env, account_info.email);
}
void ClientAndroid::OnAccessToken(JNIEnv* env,
const JavaParamRef<jobject>& jcaller,
jboolean success,
const JavaParamRef<jstring>& access_token) {
if (fetch_access_token_callback_) {
std::move(fetch_access_token_callback_)
.Run(success, base::android::ConvertJavaStringToUTF8(access_token));
}
}
void ClientAndroid::ShowUI() {
if (!controller_) {
CreateController();
// TODO(crbug.com/806868): allow delaying controller creation, for
// onboarding.
}
if (!ui_controller_android_) {
CreateUI(nullptr);
}
}
void ClientAndroid::CreateUI(
const JavaParamRef<jobject>& joverlay_coordinator) {
std::unique_ptr<UiControllerAndroid> ui_ptr =
UiControllerAndroid::CreateFromWebContents(web_contents_,
joverlay_coordinator);
if (ui_ptr)
SetUI(std::move(ui_ptr));
}
void ClientAndroid::DestroyUI() {
ui_controller_android_.reset();
}
std::string ClientAndroid::GetApiKey() {
std::string api_key;
if (google_apis::IsGoogleChromeAPIKeyUsed()) {
api_key = chrome::GetChannel() == version_info::Channel::STABLE
? google_apis::GetAPIKey()
: google_apis::GetNonStableAPIKey();
}
const auto* command_line = base::CommandLine::ForCurrentProcess();
if (command_line->HasSwitch(switches::kAutofillAssistantServerKey)) {
api_key = command_line->GetSwitchValueASCII(
switches::kAutofillAssistantServerKey);
}
return api_key;
}
std::string ClientAndroid::GetAccountEmailAddress() {
JNIEnv* env = AttachCurrentThread();
return base::android::ConvertJavaStringToUTF8(
Java_AutofillAssistantClient_getAccountEmailAddress(env, java_object_));
}
AccessTokenFetcher* ClientAndroid::GetAccessTokenFetcher() {
return this;
}
autofill::PersonalDataManager* ClientAndroid::GetPersonalDataManager() {
return autofill::PersonalDataManagerFactory::GetForProfile(
ProfileManager::GetLastUsedProfile());
}
std::string ClientAndroid::GetServerUrl() {
return server_url_;
}
UiController* ClientAndroid::GetUiController() {
if (ui_controller_android_)
return ui_controller_android_.get();
static base::NoDestructor<UiController> noop_controller_;
return noop_controller_.get();
}
std::string ClientAndroid::GetLocale() {
return base::android::GetDefaultLocaleString();
}
std::string ClientAndroid::GetCountryCode() {
return base::android::ConvertJavaStringToUTF8(
Java_AutofillAssistantClient_getCountryCode(AttachCurrentThread(),
java_object_));
}
void ClientAndroid::Shutdown(Metrics::DropOutReason reason) {
if (!controller_)
return;
// Lets the controller and the ui controller know shutdown is about to happen.
// TODO(b/128300038): Replace Controller::WillShutdown with a Detach call on
// ui_controller_android_.
controller_->WillShutdown(reason);
Metrics::RecordDropOut(reason);
// Delete the controller in a separate task. This avoids tricky ordering
// issues when Shutdown is called from the controller.
base::PostTaskWithTraits(FROM_HERE, {content::BrowserThread::UI},
base::BindOnce(&ClientAndroid::DestroyController,
weak_ptr_factory_.GetWeakPtr()));
}
void ClientAndroid::FetchAccessToken(
base::OnceCallback<void(bool, const std::string&)> callback) {
DCHECK(!fetch_access_token_callback_);
fetch_access_token_callback_ = std::move(callback);
JNIEnv* env = AttachCurrentThread();
Java_AutofillAssistantClient_fetchAccessToken(env, java_object_);
}
void ClientAndroid::InvalidateAccessToken(const std::string& access_token) {
JNIEnv* env = AttachCurrentThread();
Java_AutofillAssistantClient_invalidateAccessToken(
env, java_object_,
base::android::ConvertUTF8ToJavaString(env, access_token));
}
void ClientAndroid::CreateController() {
if (controller_) {
return;
}
controller_ = std::make_unique<Controller>(
web_contents_, /* client= */ this, base::DefaultTickClock::GetInstance());
}
void ClientAndroid::DestroyController() {
controller_.reset();
}
bool ClientAndroid::NeedsUI() {
return !ui_controller_android_ && controller_ && controller_->NeedsUI();
}
void ClientAndroid::SetUI(
std::unique_ptr<UiControllerAndroid> ui_controller_android) {
ui_controller_android_ = std::move(ui_controller_android);
ui_controller_android_->Attach(web_contents_, this, controller_.get());
}
WEB_CONTENTS_USER_DATA_KEY_IMPL(ClientAndroid)
} // namespace autofill_assistant.