blob: 4a17c37cad030e45da05dc5e248229d5c0ab6ef4 [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.
#ifndef COMPONENTS_SIGNIN_CORE_BROWSER_ACCOUNT_RECONCILOR_H_
#define COMPONENTS_SIGNIN_CORE_BROWSER_ACCOUNT_RECONCILOR_H_
#include <memory>
#include <string>
#include <vector>
#include "base/callback_forward.h"
#include "base/compiler_specific.h"
#include "base/feature_list.h"
#include "base/gtest_prod_util.h"
#include "base/macros.h"
#include "base/memory/weak_ptr.h"
#include "base/observer_list.h"
#include "base/threading/thread_checker.h"
#include "base/time/time.h"
#include "base/timer/timer.h"
#include "build/build_config.h"
#include "components/content_settings/core/browser/content_settings_observer.h"
#include "components/content_settings/core/common/content_settings_pattern.h"
#include "components/keyed_service/core/keyed_service.h"
#include "components/signin/core/browser/account_reconcilor_delegate.h"
#include "components/signin/core/browser/signin_header_helper.h"
#include "components/signin/public/base/signin_client.h"
#include "components/signin/public/base/signin_metrics.h"
#include "components/signin/public/identity_manager/identity_manager.h"
#include "google_apis/gaia/google_service_auth_error.h"
// Enables usage of Gaia Auth Multilogin endpoint for identity consistency.
extern const base::Feature kUseMultiloginEndpoint;
namespace signin {
class AccountReconcilorDelegate;
class ConsistencyCookieManagerBase;
enum class SetAccountsInCookieResult;
}
class SigninClient;
class AccountReconcilor : public KeyedService,
public content_settings::Observer,
public signin::IdentityManager::Observer {
public:
// When an instance of this class exists, the account reconcilor is suspended.
// It will automatically restart when all instances of Lock have been
// destroyed.
class Lock final {
public:
explicit Lock(AccountReconcilor* reconcilor);
~Lock();
private:
base::WeakPtr<AccountReconcilor> reconcilor_;
THREAD_CHECKER(thread_checker_);
DISALLOW_COPY_AND_ASSIGN(Lock);
};
// Helper class to indicate that synced data is being deleted. The object
// must be destroyed when the data deletion is complete.
class ScopedSyncedDataDeletion {
public:
~ScopedSyncedDataDeletion();
private:
friend class AccountReconcilor;
explicit ScopedSyncedDataDeletion(AccountReconcilor* reconcilor);
base::WeakPtr<AccountReconcilor> reconcilor_;
DISALLOW_COPY_AND_ASSIGN(ScopedSyncedDataDeletion);
};
class Observer {
public:
virtual ~Observer() {}
// The typical order of events is:
// - When reconcile is blocked:
// 1. current reconcile is aborted with AbortReconcile(),
// 2. OnStateChanged() is called with SCHEDULED.
// 3. OnBlockReconcile() is called.
// - When reconcile is unblocked:
// 1. OnUnblockReconcile() is called,
// 2. reconcile is restarted if needed with StartReconcile(), which
// triggers a call to OnStateChanged() with RUNNING.
// Called whe reconcile starts.
virtual void OnStateChanged(signin_metrics::AccountReconcilorState state) {}
// Called when the AccountReconcilor is blocked.
virtual void OnBlockReconcile() {}
// Called when the AccountReconcilor is unblocked.
virtual void OnUnblockReconcile() {}
};
AccountReconcilor(
signin::IdentityManager* identity_manager,
SigninClient* client,
std::unique_ptr<signin::AccountReconcilorDelegate> delegate);
~AccountReconcilor() override;
// Initializes the account reconcilor. Should be called once after
// construction.
void Initialize(bool start_reconcile_if_tokens_available);
void SetConsistencyCookieManager(
std::unique_ptr<signin::ConsistencyCookieManagerBase>
consistency_cookie_manager);
// Enables and disables the reconciliation.
void EnableReconcile();
void DisableReconcile(bool logout_all_gaia_accounts);
// Signal that an X-Chrome-Manage-Accounts was received from GAIA. Pass the
// ServiceType specified by GAIA in the 204 response.
// Virtual for testing.
virtual void OnReceivedManageAccountsResponse(
signin::GAIAServiceType service_type);
// KeyedService implementation.
void Shutdown() override;
// Determine what the reconcilor is currently doing.
signin_metrics::AccountReconcilorState GetState();
// Adds ands removes observers.
void AddObserver(Observer* observer);
void RemoveObserver(Observer* observer);
// ScopedSyncedDataDeletion can be created when synced data is being removed
// and destroyed when the deletion is complete. It prevents the Sync account
// from being invalidated during the deletion.
std::unique_ptr<ScopedSyncedDataDeletion> GetScopedSyncDataDeletion();
private:
friend class AccountReconcilorTest;
friend class DiceBrowserTest;
friend class BaseAccountReconcilorTestTable;
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorTestForceDiceMigration,
TableRowTestCheckNoOp);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
IdentityManagerRegistration);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest, Reauth);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
ProfileAlreadyConnected);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorTestTable, TableRowTest);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorTestTable,
InconsistencyReasonLogging);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorTestDiceMultilogin, TableRowTest);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorTestMirrorMultilogin, TableRowTest);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorTestMiceMultilogin, TableRowTest);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMiceTest,
AccountReconcilorStateScheduled);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorDiceEndpointParamTest,
DiceTokenServiceRegistration);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorDiceEndpointParamTest,
DiceReconcileWithoutSignin);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorDiceEndpointParamTest,
DiceReconcileNoop);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorDiceEndpointParamTest,
DiceLastKnownFirstAccount);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorDiceEndpointParamTest,
UnverifiedAccountNoop);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorDiceEndpointParamTest,
UnverifiedAccountMerge);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorDiceEndpointParamTest,
HandleSigninDuringReconcile);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorDiceEndpointParamTest,
DiceMigrationAfterNoop);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorDiceEndpointParamTest,
DiceNoMigrationWhenTokensNotReady);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorDiceEndpointParamTest,
DiceNoMigrationAfterReconcile);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorDiceEndpointParamTest,
DiceReconcileReuseGaiaFirstAccount);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorDiceEndpointParamTest,
MigrationClearSecondaryTokens);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorTest, MigrationClearAllTokens);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorTest, DiceDeleteCookie);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorDiceEndpointParamTest,
MigrationClearAllTokens);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
TokensNotLoaded);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
StartReconcileCookiesDisabled);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
StartReconcileContentSettings);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
StartReconcileContentSettingsGaiaUrl);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
StartReconcileContentSettingsNonGaiaUrl);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
StartReconcileContentSettingsInvalidPattern);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
GetAccountsFromCookieSuccess);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
GetAccountsFromCookieFailure);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
ExtraCookieChangeNotification);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
StartReconcileNoop);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
StartReconcileNoopWithDots);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
StartReconcileNoopMultiple);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
StartReconcileAddToCookie);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorTest, AuthErrorTriggersListAccount);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
SignoutAfterErrorDoesNotRecordUma);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
TokenErrorOnPrimary);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
StartReconcileRemoveFromCookie);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
StartReconcileAddToCookieTwice);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
StartReconcileBadPrimary);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
StartReconcileOnlyOnce);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest, Lock);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMethodParamTest,
StartReconcileWithSessionInfoExpiredDefault);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMethodParamTest,
AccountReconcilorStateScheduled);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
AddAccountToCookieCompletedWithBogusAccount);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
NoLoopWithBadPrimary);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
WontMergeAccountsWithError);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorTest, DelegateTimeoutIsCalled);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorMirrorEndpointParamTest,
DelegateTimeoutIsNotCalled);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorTest,
DelegateTimeoutIsNotCalledIfTimeoutIsNotReached);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorTest, MultiloginLogout);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorTestForceDiceMigration,
TableRowTest);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorTestActiveDirectory,
TableRowTestMergeSession);
FRIEND_TEST_ALL_PREFIXES(AccountReconcilorTestActiveDirectory,
TableRowTestMultilogin);
void set_timer_for_testing(std::unique_ptr<base::OneShotTimer> timer);
bool IsRegisteredWithIdentityManager() const {
return registered_with_identity_manager_;
}
// Register and unregister with dependent services.
void RegisterWithAllDependencies();
void UnregisterWithAllDependencies();
void RegisterWithIdentityManager();
void UnregisterWithIdentityManager();
void RegisterWithCookieManagerService();
void UnregisterWithCookieManagerService();
void RegisterWithContentSettings();
void UnregisterWithContentSettings();
// All actions with side effects, only doing meaningful work if account
// consistency is enabled. Virtual so that they can be overridden in tests.
virtual void PerformMergeAction(const CoreAccountId& account_id);
virtual void PerformLogoutAllAccountsAction();
virtual void PerformSetCookiesAction(
const signin::MultiloginParameters& parameters);
// Used during periodic reconciliation.
void StartReconcile();
// |gaia_accounts| are the accounts in the Gaia cookie.
void FinishReconcile(const CoreAccountId& primary_account,
const std::vector<CoreAccountId>& chrome_accounts,
std::vector<gaia::ListedAccount>&& gaia_accounts);
void AbortReconcile();
void CalculateIfReconcileIsDone();
void ScheduleStartReconcileIfChromeAccountsChanged();
// Returns the list of valid accounts from the TokenService.
std::vector<CoreAccountId> LoadValidAccountsFromTokenService() const;
// Note internally that this |account_id| is added to the cookie jar.
bool MarkAccountAsAddedToCookie(const CoreAccountId& account_id);
// The reconcilor only starts when the token service is ready.
bool IsIdentityManagerReady();
// Overridden from content_settings::Observer.
void OnContentSettingChanged(const ContentSettingsPattern& primary_pattern,
const ContentSettingsPattern& secondary_pattern,
ContentSettingsType content_type,
const std::string& resource_identifier) override;
// Overridden from signin::IdentityManager::Observer.
void OnEndBatchOfRefreshTokenStateChanges() override;
void OnRefreshTokensLoaded() override;
void OnErrorStateOfRefreshTokenUpdatedForAccount(
const CoreAccountInfo& account_info,
const GoogleServiceAuthError& error) override;
void OnAccountsInCookieUpdated(
const signin::AccountsInCookieJarInfo& accounts_in_cookie_jar_info,
const GoogleServiceAuthError& error) override;
void OnAccountsCookieDeletedByUserAction() override;
void FinishReconcileWithMultiloginEndpoint(
const CoreAccountId& primary_account,
const std::vector<CoreAccountId>& chrome_accounts,
std::vector<gaia::ListedAccount>&& gaia_accounts);
void OnAddAccountToCookieCompleted(const CoreAccountId& account_id,
const GoogleServiceAuthError& error);
void OnSetAccountsInCookieCompleted(signin::SetAccountsInCookieResult result);
// Lock related methods.
void IncrementLockCount();
void DecrementLockCount();
void BlockReconcile();
void UnblockReconcile();
bool IsReconcileBlocked() const;
void HandleReconcileTimeout();
// Returns true is multilogin endpoint can be enabled.
bool IsMultiloginEndpointEnabled() const;
// Returns true if current array of existing accounts in cookie is different
// from the desired one. If this returns false, the multilogin call would be a
// no-op.
bool CookieNeedsUpdate(
const signin::MultiloginParameters& parameters,
const std::vector<gaia::ListedAccount>& existing_accounts);
// Sets the reconcilor state and calls Observer::OnStateChanged() if needed.
void SetState(signin_metrics::AccountReconcilorState state);
std::unique_ptr<signin::AccountReconcilorDelegate> delegate_;
// The IdentityManager associated with this reconcilor.
signin::IdentityManager* identity_manager_;
// The SigninClient associated with this reconcilor.
SigninClient* client_;
bool registered_with_identity_manager_;
bool registered_with_content_settings_;
// True while the reconcilor is busy checking or managing the accounts in
// this profile.
bool is_reconcile_started_;
base::Time reconcile_start_time_;
// True iff this is the first time the reconcilor is executing.
bool first_execution_;
// 'Most severe' error encountered during the last attempt to reconcile. If
// the last reconciliation attempt was successful, this will be
// |GoogleServiceAuthError::State::NONE|.
// Severity of an error is defined on the basis of
// |GoogleServiceAuthError::IsPersistentError()| only, i.e. any persistent
// error is considered more severe than all non-persistent errors, but
// persistent (or non-persistent) errors do not have an internal severity
// ordering among themselves.
GoogleServiceAuthError error_during_last_reconcile_;
// Used for Dice migration: migration can happen if the accounts are
// consistent, which is indicated by reconcile being a no-op.
bool reconcile_is_noop_;
// Used during reconcile action.
std::vector<CoreAccountId> add_to_cookie_; // Progress of AddAccount calls.
bool set_accounts_in_progress_; // Progress of SetAccounts calls.
bool chrome_accounts_changed_;
// Used for the Lock.
// StartReconcile() is blocked while this is > 0.
int account_reconcilor_lock_count_;
// StartReconcile() should be started when the reconcilor is unblocked.
bool reconcile_on_unblock_;
base::ObserverList<Observer, true>::Unchecked observer_list_;
// A timer to set off reconciliation timeout handlers, if account
// reconciliation does not happen in a given |timeout_| duration.
// Any delegate that wants to use this feature must override
// |AccountReconcilorDelegate::GetReconcileTimeout|.
// Note: This is intended as a safeguard for delegates that want a 'guarantee'
// of reconciliation completing within a finite time. It is technically
// possible for account reconciliation to be running/waiting forever in cases
// such as a network connection not being present.
std::unique_ptr<base::OneShotTimer> timer_;
base::TimeDelta timeout_;
// Greater than 0 when synced data is being deleted, and it is important to
// not invalidate the primary token while this is happening.
int synced_data_deletion_in_progress_count_ = 0;
signin_metrics::AccountReconcilorState state_;
std::unique_ptr<signin::ConsistencyCookieManagerBase>
consistency_cookie_manager_;
base::WeakPtrFactory<AccountReconcilor> weak_factory_{this};
DISALLOW_COPY_AND_ASSIGN(AccountReconcilor);
};
#endif // COMPONENTS_SIGNIN_CORE_BROWSER_ACCOUNT_RECONCILOR_H_