// Copyright 2018 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include <utility>
#include "chromeos/ash/services/multidevice_setup/multidevice_setup_initializer.h"
#include "base/check.h"
#include "base/memory/ptr_util.h"
#include "chromeos/ash/components/multidevice/logging/logging.h"
#include "chromeos/ash/services/multidevice_setup/multidevice_setup_impl.h"
#include "chromeos/ash/services/multidevice_setup/public/cpp/android_sms_app_helper_delegate.h"
#include "chromeos/ash/services/multidevice_setup/public/cpp/android_sms_pairing_state_tracker.h"
namespace ash {
namespace multidevice_setup {
// static
MultiDeviceSetupInitializer::Factory*
MultiDeviceSetupInitializer::Factory::test_factory_ = nullptr;
// static
std::unique_ptr<MultiDeviceSetupBase>
MultiDeviceSetupInitializer::Factory::Create(
PrefService* pref_service,
device_sync::DeviceSyncClient* device_sync_client,
AuthTokenValidator* auth_token_validator,
OobeCompletionTracker* oobe_completion_tracker,
AndroidSmsAppHelperDelegate* android_sms_app_helper_delegate,
AndroidSmsPairingStateTracker* android_sms_pairing_state_tracker,
const device_sync::GcmDeviceInfoProvider* gcm_device_info_provider,
bool is_secondary_user) {
if (test_factory_) {
return test_factory_->CreateInstance(
pref_service, device_sync_client, auth_token_validator,
oobe_completion_tracker, android_sms_app_helper_delegate,
android_sms_pairing_state_tracker, gcm_device_info_provider,
is_secondary_user);
}
return base::WrapUnique(new MultiDeviceSetupInitializer(
pref_service, device_sync_client, auth_token_validator,
oobe_completion_tracker, android_sms_app_helper_delegate,
android_sms_pairing_state_tracker, gcm_device_info_provider,
is_secondary_user));
}
// static
void MultiDeviceSetupInitializer::Factory::SetFactoryForTesting(
Factory* test_factory) {
test_factory_ = test_factory;
}
MultiDeviceSetupInitializer::Factory::~Factory() = default;
MultiDeviceSetupInitializer::SetHostDeviceArgs::SetHostDeviceArgs(
const std::string& host_instance_id_or_legacy_device_id,
const std::string& auth_token,
SetHostDeviceCallback callback)
: host_instance_id_or_legacy_device_id(
host_instance_id_or_legacy_device_id),
auth_token(auth_token),
callback(std::move(callback)) {}
MultiDeviceSetupInitializer::SetHostDeviceArgs::SetHostDeviceArgs(
const std::string& host_instance_id_or_legacy_device_id,
mojom::PrivilegedHostDeviceSetter::SetHostDeviceCallback callback)
: host_instance_id_or_legacy_device_id(
host_instance_id_or_legacy_device_id),
callback(std::move(callback)) {}
MultiDeviceSetupInitializer::SetHostDeviceArgs::~SetHostDeviceArgs() = default;
MultiDeviceSetupInitializer::MultiDeviceSetupInitializer(
PrefService* pref_service,
device_sync::DeviceSyncClient* device_sync_client,
AuthTokenValidator* auth_token_validator,
OobeCompletionTracker* oobe_completion_tracker,
AndroidSmsAppHelperDelegate* android_sms_app_helper_delegate,
AndroidSmsPairingStateTracker* android_sms_pairing_state_tracker,
const device_sync::GcmDeviceInfoProvider* gcm_device_info_provider,
bool is_secondary_user)
: pref_service_(pref_service),
device_sync_client_(device_sync_client),
auth_token_validator_(auth_token_validator),
oobe_completion_tracker_(oobe_completion_tracker),
android_sms_app_helper_delegate_(android_sms_app_helper_delegate),
android_sms_pairing_state_tracker_(android_sms_pairing_state_tracker),
gcm_device_info_provider_(gcm_device_info_provider),
is_secondary_user_(is_secondary_user) {
// If |device_sync_client_| is null, this interface cannot perform its tasks.
if (!device_sync_client_)
return;
if (device_sync_client_->is_ready()) {
InitializeImplementation();
return;
}
device_sync_client_->AddObserver(this);
}
MultiDeviceSetupInitializer::~MultiDeviceSetupInitializer() = default;
void MultiDeviceSetupInitializer::SetAccountStatusChangeDelegate(
mojo::PendingRemote<mojom::AccountStatusChangeDelegate> delegate) {
if (multidevice_setup_impl_) {
multidevice_setup_impl_->SetAccountStatusChangeDelegate(
std::move(delegate));
return;
}
pending_delegate_ = std::move(delegate);
}
void MultiDeviceSetupInitializer::AddHostStatusObserver(
mojo::PendingRemote<mojom::HostStatusObserver> observer) {
if (multidevice_setup_impl_) {
multidevice_setup_impl_->AddHostStatusObserver(std::move(observer));
return;
}
pending_host_status_observers_.push_back(std::move(observer));
}
void MultiDeviceSetupInitializer::AddFeatureStateObserver(
mojo::PendingRemote<mojom::FeatureStateObserver> observer) {
if (multidevice_setup_impl_) {
multidevice_setup_impl_->AddFeatureStateObserver(std::move(observer));
return;
}
pending_feature_state_observers_.push_back(std::move(observer));
}
void MultiDeviceSetupInitializer::GetEligibleHostDevices(
GetEligibleHostDevicesCallback callback) {
if (multidevice_setup_impl_) {
multidevice_setup_impl_->GetEligibleHostDevices(std::move(callback));
return;
}
pending_get_eligible_hosts_args_.push_back(std::move(callback));
}
void MultiDeviceSetupInitializer::GetEligibleActiveHostDevices(
GetEligibleActiveHostDevicesCallback callback) {
if (multidevice_setup_impl_) {
multidevice_setup_impl_->GetEligibleActiveHostDevices(std::move(callback));
return;
}
pending_get_eligible_active_hosts_args_.push_back(std::move(callback));
}
void MultiDeviceSetupInitializer::SetHostDevice(
const std::string& host_instance_id_or_legacy_device_id,
const std::string& auth_token,
SetHostDeviceCallback callback) {
if (multidevice_setup_impl_) {
multidevice_setup_impl_->SetHostDevice(host_instance_id_or_legacy_device_id,
auth_token, std::move(callback));
return;
}
// If a pending request to set another device exists, invoke its callback. It
// is stale, since now an updated request has been made to set the host.
if (pending_set_host_args_)
std::move(pending_set_host_args_->callback).Run(false /* success */);
// If a pending request to remove the current device exists, cancel it.
pending_should_remove_host_device_ = false;
pending_set_host_args_.emplace(host_instance_id_or_legacy_device_id,
auth_token, std::move(callback));
}
void MultiDeviceSetupInitializer::RemoveHostDevice() {
if (multidevice_setup_impl_) {
multidevice_setup_impl_->RemoveHostDevice();
return;
}
// If a pending request to set another device exists, invoke its callback. It
// is stale, since now a request has been made to remove the host.
if (pending_set_host_args_) {
std::move(pending_set_host_args_->callback).Run(false /* success */);
pending_set_host_args_.reset();
}
pending_should_remove_host_device_ = true;
}
void MultiDeviceSetupInitializer::GetHostStatus(
GetHostStatusCallback callback) {
if (multidevice_setup_impl_) {
multidevice_setup_impl_->GetHostStatus(std::move(callback));
return;
}
pending_get_host_args_.push_back(std::move(callback));
}
void MultiDeviceSetupInitializer::SetFeatureEnabledState(
mojom::Feature feature,
bool enabled,
const std::optional<std::string>& auth_token,
SetFeatureEnabledStateCallback callback) {
if (multidevice_setup_impl_) {
multidevice_setup_impl_->SetFeatureEnabledState(
feature, enabled, auth_token, std::move(callback));
return;
}
pending_set_feature_enabled_args_.emplace_back(feature, enabled, auth_token,
std::move(callback));
}
void MultiDeviceSetupInitializer::GetFeatureStates(
GetFeatureStatesCallback callback) {
if (multidevice_setup_impl_) {
multidevice_setup_impl_->GetFeatureStates(std::move(callback));
return;
}
pending_get_feature_states_args_.emplace_back(std::move(callback));
}
void MultiDeviceSetupInitializer::RetrySetHostNow(
RetrySetHostNowCallback callback) {
if (multidevice_setup_impl_) {
multidevice_setup_impl_->RetrySetHostNow(std::move(callback));
return;
}
pending_retry_set_host_args_.push_back(std::move(callback));
}
void MultiDeviceSetupInitializer::TriggerEventForDebugging(
mojom::EventTypeForDebugging type,
TriggerEventForDebuggingCallback callback) {
if (multidevice_setup_impl_) {
multidevice_setup_impl_->TriggerEventForDebugging(type,
std::move(callback));
return;
}
// If initialization is not complete, no debug event is sent.
std::move(callback).Run(false /* success */);
}
void MultiDeviceSetupInitializer::SetQuickStartPhoneInstanceID(
const std::string& qs_phone_instance_id) {
if (multidevice_setup_impl_) {
multidevice_setup_impl_->SetQuickStartPhoneInstanceID(qs_phone_instance_id);
return;
}
pending_set_qs_phone_instance_id_args_.push_back(qs_phone_instance_id);
}
void MultiDeviceSetupInitializer::GetQuickStartPhoneInstanceID(
GetQuickStartPhoneInstanceIDCallback callback) {
if (multidevice_setup_impl_) {
multidevice_setup_impl_->GetQuickStartPhoneInstanceID(std::move(callback));
return;
}
pending_get_qs_phone_instance_id_args_.push_back(std::move(callback));
}
void MultiDeviceSetupInitializer::SetHostDeviceWithoutAuthToken(
const std::string& host_instance_id_or_legacy_device_id,
mojom::PrivilegedHostDeviceSetter::SetHostDeviceCallback callback) {
if (multidevice_setup_impl_) {
multidevice_setup_impl_->SetHostDeviceWithoutAuthToken(
host_instance_id_or_legacy_device_id, std::move(callback));
return;
}
// If a pending request to set another device exists, invoke its callback. It
// is stale, since now an updated request has been made to set the host.
if (pending_set_host_args_) {
std::move(pending_set_host_args_->callback).Run(false /* success */);
pending_set_host_args_.reset();
}
// If a pending request to remove the current device exists, cancel it.
pending_should_remove_host_device_ = false;
pending_set_host_args_.emplace(host_instance_id_or_legacy_device_id,
std::move(callback));
}
void MultiDeviceSetupInitializer::OnReady() {
device_sync_client_->RemoveObserver(this);
InitializeImplementation();
}
void MultiDeviceSetupInitializer::InitializeImplementation() {
DCHECK(!multidevice_setup_impl_);
multidevice_setup_impl_ = MultiDeviceSetupImpl::Factory::Create(
pref_service_, device_sync_client_, auth_token_validator_,
oobe_completion_tracker_, android_sms_app_helper_delegate_,
android_sms_pairing_state_tracker_, gcm_device_info_provider_,
is_secondary_user_);
if (pending_delegate_) {
multidevice_setup_impl_->SetAccountStatusChangeDelegate(
std::move(pending_delegate_));
}
for (auto& observer : pending_host_status_observers_)
multidevice_setup_impl_->AddHostStatusObserver(std::move(observer));
pending_host_status_observers_.clear();
for (auto& observer : pending_feature_state_observers_)
multidevice_setup_impl_->AddFeatureStateObserver(std::move(observer));
pending_feature_state_observers_.clear();
if (pending_set_host_args_) {
DCHECK(!pending_should_remove_host_device_);
if (pending_set_host_args_->auth_token) {
multidevice_setup_impl_->SetHostDevice(
pending_set_host_args_->host_instance_id_or_legacy_device_id,
*pending_set_host_args_->auth_token,
std::move(pending_set_host_args_->callback));
} else {
multidevice_setup_impl_->SetHostDeviceWithoutAuthToken(
pending_set_host_args_->host_instance_id_or_legacy_device_id,
std::move(pending_set_host_args_->callback));
}
pending_set_host_args_.reset();
}
if (pending_should_remove_host_device_)
multidevice_setup_impl_->RemoveHostDevice();
pending_should_remove_host_device_ = false;
for (auto& set_feature_enabled_args : pending_set_feature_enabled_args_) {
multidevice_setup_impl_->SetFeatureEnabledState(
std::get<0>(set_feature_enabled_args),
std::get<1>(set_feature_enabled_args),
std::get<2>(set_feature_enabled_args),
std::move(std::get<3>(set_feature_enabled_args)));
}
pending_set_feature_enabled_args_.clear();
for (auto& get_feature_states_callback : pending_get_feature_states_args_) {
multidevice_setup_impl_->GetFeatureStates(
std::move(get_feature_states_callback));
}
pending_get_feature_states_args_.clear();
for (auto& retry_callback : pending_retry_set_host_args_)
multidevice_setup_impl_->RetrySetHostNow(std::move(retry_callback));
pending_retry_set_host_args_.clear();
for (auto& get_eligible_callback : pending_get_eligible_hosts_args_) {
multidevice_setup_impl_->GetEligibleHostDevices(
std::move(get_eligible_callback));
}
pending_get_eligible_hosts_args_.clear();
for (auto& get_eligible_callback : pending_get_eligible_active_hosts_args_) {
multidevice_setup_impl_->GetEligibleActiveHostDevices(
std::move(get_eligible_callback));
}
pending_get_eligible_active_hosts_args_.clear();
for (auto& get_host_callback : pending_get_host_args_)
multidevice_setup_impl_->GetHostStatus(std::move(get_host_callback));
pending_get_host_args_.clear();
for (auto& qs_phone_instance_id : pending_set_qs_phone_instance_id_args_) {
multidevice_setup_impl_->SetQuickStartPhoneInstanceID(qs_phone_instance_id);
}
pending_set_qs_phone_instance_id_args_.clear();
for (auto& get_qs_phone_instance_id_callback :
pending_get_qs_phone_instance_id_args_) {
multidevice_setup_impl_->GetQuickStartPhoneInstanceID(
std::move(get_qs_phone_instance_id_callback));
}
pending_get_qs_phone_instance_id_args_.clear();
}
} // namespace multidevice_setup
} // namespace ash