Add per-user preferences support.
[chromium-blink-merge.git] / google_apis / gaia / account_tracker.cc
blobc3653ac14ade99b67ce7835c49bc5c1ce8124f70
1 // Copyright 2014 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "google_apis/gaia/account_tracker.h"
7 #include "base/logging.h"
8 #include "base/stl_util.h"
9 #include "base/trace_event/trace_event.h"
10 #include "net/url_request/url_request_context_getter.h"
12 namespace gaia {
14 AccountTracker::AccountTracker(
15 IdentityProvider* identity_provider,
16 net::URLRequestContextGetter* request_context_getter)
17 : identity_provider_(identity_provider),
18 request_context_getter_(request_context_getter),
19 shutdown_called_(false) {
20 identity_provider_->AddObserver(this);
21 identity_provider_->GetTokenService()->AddObserver(this);
24 AccountTracker::~AccountTracker() {
25 DCHECK(shutdown_called_);
28 void AccountTracker::Shutdown() {
29 shutdown_called_ = true;
30 STLDeleteValues(&user_info_requests_);
31 identity_provider_->GetTokenService()->RemoveObserver(this);
32 identity_provider_->RemoveObserver(this);
35 bool AccountTracker::IsAllUserInfoFetched() const {
36 return user_info_requests_.empty();
39 void AccountTracker::AddObserver(Observer* observer) {
40 observer_list_.AddObserver(observer);
43 void AccountTracker::RemoveObserver(Observer* observer) {
44 observer_list_.RemoveObserver(observer);
47 std::vector<AccountIds> AccountTracker::GetAccounts() const {
48 const std::string active_account_id =
49 identity_provider_->GetActiveAccountId();
50 std::vector<AccountIds> accounts;
52 for (std::map<std::string, AccountState>::const_iterator it =
53 accounts_.begin();
54 it != accounts_.end();
55 ++it) {
56 const AccountState& state = it->second;
57 bool is_visible = state.is_signed_in && !state.ids.gaia.empty();
59 if (it->first == active_account_id) {
60 if (is_visible)
61 accounts.insert(accounts.begin(), state.ids);
62 else
63 return std::vector<AccountIds>();
65 } else if (is_visible) {
66 accounts.push_back(state.ids);
69 return accounts;
72 AccountIds AccountTracker::FindAccountIdsByGaiaId(const std::string& gaia_id) {
73 for (std::map<std::string, AccountState>::const_iterator it =
74 accounts_.begin();
75 it != accounts_.end();
76 ++it) {
77 const AccountState& state = it->second;
78 if (state.ids.gaia == gaia_id) {
79 return state.ids;
83 return AccountIds();
86 void AccountTracker::OnRefreshTokenAvailable(const std::string& account_id) {
87 TRACE_EVENT1("identity",
88 "AccountTracker::OnRefreshTokenAvailable",
89 "account_key",
90 account_id);
92 // Ignore refresh tokens if there is no active account ID at all.
93 if (identity_provider_->GetActiveAccountId().empty())
94 return;
96 DVLOG(1) << "AVAILABLE " << account_id;
97 UpdateSignInState(account_id, true);
100 void AccountTracker::OnRefreshTokenRevoked(const std::string& account_id) {
101 TRACE_EVENT1("identity",
102 "AccountTracker::OnRefreshTokenRevoked",
103 "account_key",
104 account_id);
106 DVLOG(1) << "REVOKED " << account_id;
107 UpdateSignInState(account_id, false);
110 void AccountTracker::OnActiveAccountLogin() {
111 TRACE_EVENT0("identity", "AccountTracker::OnActiveAccountLogin");
113 std::vector<std::string> accounts =
114 identity_provider_->GetTokenService()->GetAccounts();
116 DVLOG(1) << "LOGIN " << accounts.size() << " accounts available.";
118 for (std::vector<std::string>::const_iterator it = accounts.begin();
119 it != accounts.end();
120 ++it) {
121 OnRefreshTokenAvailable(*it);
125 void AccountTracker::OnActiveAccountLogout() {
126 TRACE_EVENT0("identity", "AccountTracker::OnActiveAccountLogout");
127 DVLOG(1) << "LOGOUT";
128 StopTrackingAllAccounts();
131 void AccountTracker::SetAccountStateForTest(AccountIds ids, bool is_signed_in) {
132 accounts_[ids.account_key].ids = ids;
133 accounts_[ids.account_key].is_signed_in = is_signed_in;
135 DVLOG(1) << "SetAccountStateForTest " << ids.account_key << ":"
136 << is_signed_in;
138 if (VLOG_IS_ON(1)) {
139 for (std::map<std::string, AccountState>::const_iterator it =
140 accounts_.begin();
141 it != accounts_.end();
142 ++it) {
143 DVLOG(1) << it->first << ":" << it->second.is_signed_in;
148 void AccountTracker::NotifyAccountAdded(const AccountState& account) {
149 DCHECK(!account.ids.gaia.empty());
150 FOR_EACH_OBSERVER(
151 Observer, observer_list_, OnAccountAdded(account.ids));
154 void AccountTracker::NotifyAccountRemoved(const AccountState& account) {
155 DCHECK(!account.ids.gaia.empty());
156 FOR_EACH_OBSERVER(
157 Observer, observer_list_, OnAccountRemoved(account.ids));
160 void AccountTracker::NotifySignInChanged(const AccountState& account) {
161 DCHECK(!account.ids.gaia.empty());
162 FOR_EACH_OBSERVER(Observer,
163 observer_list_,
164 OnAccountSignInChanged(account.ids, account.is_signed_in));
167 void AccountTracker::UpdateSignInState(const std::string account_key,
168 bool is_signed_in) {
169 StartTrackingAccount(account_key);
170 AccountState& account = accounts_[account_key];
171 bool needs_gaia_id = account.ids.gaia.empty();
172 bool was_signed_in = account.is_signed_in;
173 account.is_signed_in = is_signed_in;
175 if (needs_gaia_id && is_signed_in)
176 StartFetchingUserInfo(account_key);
178 if (!needs_gaia_id && (was_signed_in != is_signed_in))
179 NotifySignInChanged(account);
182 void AccountTracker::StartTrackingAccount(const std::string account_key) {
183 if (!ContainsKey(accounts_, account_key)) {
184 DVLOG(1) << "StartTracking " << account_key;
185 AccountState account_state;
186 account_state.ids.account_key = account_key;
187 account_state.ids.email = account_key;
188 account_state.is_signed_in = false;
189 accounts_.insert(make_pair(account_key, account_state));
193 void AccountTracker::StopTrackingAccount(const std::string account_key) {
194 DVLOG(1) << "StopTracking " << account_key;
195 if (ContainsKey(accounts_, account_key)) {
196 AccountState& account = accounts_[account_key];
197 if (!account.ids.gaia.empty()) {
198 UpdateSignInState(account_key, false);
199 NotifyAccountRemoved(account);
201 accounts_.erase(account_key);
204 if (ContainsKey(user_info_requests_, account_key))
205 DeleteFetcher(user_info_requests_[account_key]);
208 void AccountTracker::StopTrackingAllAccounts() {
209 while (!accounts_.empty())
210 StopTrackingAccount(accounts_.begin()->first);
213 void AccountTracker::StartFetchingUserInfo(const std::string account_key) {
214 if (ContainsKey(user_info_requests_, account_key))
215 DeleteFetcher(user_info_requests_[account_key]);
217 DVLOG(1) << "StartFetching " << account_key;
218 AccountIdFetcher* fetcher =
219 new AccountIdFetcher(identity_provider_->GetTokenService(),
220 request_context_getter_.get(),
221 this,
222 account_key);
223 user_info_requests_[account_key] = fetcher;
224 fetcher->Start();
227 void AccountTracker::OnUserInfoFetchSuccess(AccountIdFetcher* fetcher,
228 const std::string& gaia_id) {
229 const std::string& account_key = fetcher->account_key();
230 DCHECK(ContainsKey(accounts_, account_key));
231 AccountState& account = accounts_[account_key];
233 account.ids.gaia = gaia_id;
234 NotifyAccountAdded(account);
236 if (account.is_signed_in)
237 NotifySignInChanged(account);
239 DeleteFetcher(fetcher);
242 void AccountTracker::OnUserInfoFetchFailure(AccountIdFetcher* fetcher) {
243 LOG(WARNING) << "Failed to get UserInfo for " << fetcher->account_key();
244 std::string key = fetcher->account_key();
245 DeleteFetcher(fetcher);
246 StopTrackingAccount(key);
249 void AccountTracker::DeleteFetcher(AccountIdFetcher* fetcher) {
250 DVLOG(1) << "DeleteFetcher " << fetcher->account_key();
251 const std::string& account_key = fetcher->account_key();
252 DCHECK(ContainsKey(user_info_requests_, account_key));
253 DCHECK_EQ(fetcher, user_info_requests_[account_key]);
254 user_info_requests_.erase(account_key);
255 delete fetcher;
258 AccountIdFetcher::AccountIdFetcher(
259 OAuth2TokenService* token_service,
260 net::URLRequestContextGetter* request_context_getter,
261 AccountTracker* tracker,
262 const std::string& account_key)
263 : OAuth2TokenService::Consumer("gaia_account_tracker"),
264 token_service_(token_service),
265 request_context_getter_(request_context_getter),
266 tracker_(tracker),
267 account_key_(account_key) {
268 TRACE_EVENT_ASYNC_BEGIN1(
269 "identity", "AccountIdFetcher", this, "account_key", account_key);
272 AccountIdFetcher::~AccountIdFetcher() {
273 TRACE_EVENT_ASYNC_END0("identity", "AccountIdFetcher", this);
276 void AccountIdFetcher::Start() {
277 OAuth2TokenService::ScopeSet scopes;
278 scopes.insert("https://www.googleapis.com/auth/userinfo.profile");
279 login_token_request_ = token_service_->StartRequest(
280 account_key_, scopes, this);
283 void AccountIdFetcher::OnGetTokenSuccess(
284 const OAuth2TokenService::Request* request,
285 const std::string& access_token,
286 const base::Time& expiration_time) {
287 TRACE_EVENT_ASYNC_STEP_PAST0(
288 "identity", "AccountIdFetcher", this, "OnGetTokenSuccess");
289 DCHECK_EQ(request, login_token_request_.get());
291 gaia_oauth_client_.reset(new gaia::GaiaOAuthClient(request_context_getter_));
293 const int kMaxGetUserIdRetries = 3;
294 gaia_oauth_client_->GetUserId(access_token, kMaxGetUserIdRetries, this);
297 void AccountIdFetcher::OnGetTokenFailure(
298 const OAuth2TokenService::Request* request,
299 const GoogleServiceAuthError& error) {
300 TRACE_EVENT_ASYNC_STEP_PAST1("identity",
301 "AccountIdFetcher",
302 this,
303 "OnGetTokenFailure",
304 "google_service_auth_error",
305 error.ToString());
306 LOG(ERROR) << "OnGetTokenFailure: " << error.ToString();
307 DCHECK_EQ(request, login_token_request_.get());
308 tracker_->OnUserInfoFetchFailure(this);
311 void AccountIdFetcher::OnGetUserIdResponse(const std::string& gaia_id) {
312 TRACE_EVENT_ASYNC_STEP_PAST1("identity",
313 "AccountIdFetcher",
314 this,
315 "OnGetUserIdResponse",
316 "gaia_id",
317 gaia_id);
318 tracker_->OnUserInfoFetchSuccess(this, gaia_id);
321 void AccountIdFetcher::OnOAuthError() {
322 TRACE_EVENT_ASYNC_STEP_PAST0(
323 "identity", "AccountIdFetcher", this, "OnOAuthError");
324 LOG(ERROR) << "OnOAuthError";
325 tracker_->OnUserInfoFetchFailure(this);
328 void AccountIdFetcher::OnNetworkError(int response_code) {
329 TRACE_EVENT_ASYNC_STEP_PAST1("identity",
330 "AccountIdFetcher",
331 this,
332 "OnNetworkError",
333 "response_code",
334 response_code);
335 LOG(ERROR) << "OnNetworkError " << response_code;
336 tracker_->OnUserInfoFetchFailure(this);
339 } // namespace gaia