• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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.
4 
5 #include "google_apis/gaia/ubertoken_fetcher.h"
6 
7 #include <vector>
8 
9 #include "base/logging.h"
10 #include "base/rand_util.h"
11 #include "base/time/time.h"
12 #include "google_apis/gaia/gaia_auth_fetcher.h"
13 #include "google_apis/gaia/gaia_constants.h"
14 #include "google_apis/gaia/google_service_auth_error.h"
15 #include "google_apis/gaia/oauth2_token_service.h"
16 
17 const int UbertokenFetcher::kMaxRetries = 3;
18 
UbertokenFetcher(OAuth2TokenService * token_service,UbertokenConsumer * consumer,net::URLRequestContextGetter * request_context)19 UbertokenFetcher::UbertokenFetcher(
20     OAuth2TokenService* token_service,
21     UbertokenConsumer* consumer,
22     net::URLRequestContextGetter* request_context)
23     : OAuth2TokenService::Consumer("uber_token_fetcher"),
24       token_service_(token_service),
25       consumer_(consumer),
26       request_context_(request_context),
27       retry_number_(0),
28       second_access_token_request_(false) {
29   DCHECK(token_service);
30   DCHECK(consumer);
31   DCHECK(request_context);
32 }
33 
~UbertokenFetcher()34 UbertokenFetcher::~UbertokenFetcher() {
35 }
36 
StartFetchingToken(const std::string & account_id)37 void UbertokenFetcher::StartFetchingToken(const std::string& account_id) {
38   DCHECK(!account_id.empty());
39   account_id_ = account_id;
40   second_access_token_request_ = false;
41   RequestAccessToken();
42 }
43 
OnUberAuthTokenSuccess(const std::string & token)44 void UbertokenFetcher::OnUberAuthTokenSuccess(const std::string& token) {
45   consumer_->OnUbertokenSuccess(token);
46 }
47 
OnUberAuthTokenFailure(const GoogleServiceAuthError & error)48 void UbertokenFetcher::OnUberAuthTokenFailure(
49     const GoogleServiceAuthError& error) {
50   // Retry only transient errors.
51   bool should_retry =
52       error.state() == GoogleServiceAuthError::CONNECTION_FAILED ||
53       error.state() == GoogleServiceAuthError::SERVICE_UNAVAILABLE;
54   if (should_retry) {
55     if (retry_number_ < kMaxRetries) {
56       // Calculate an exponential backoff with randomness of less than 1 sec.
57       double backoff = base::RandDouble() + (1 << retry_number_);
58       ++retry_number_;
59       retry_timer_.Stop();
60       retry_timer_.Start(FROM_HERE,
61                          base::TimeDelta::FromSecondsD(backoff),
62                          this,
63                          &UbertokenFetcher::ExchangeTokens);
64       return;
65     }
66   } else {
67     // The access token is invalid.  Tell the token service.
68     OAuth2TokenService::ScopeSet scopes;
69     scopes.insert(GaiaConstants::kOAuth1LoginScope);
70     token_service_->InvalidateToken(account_id_, scopes, access_token_);
71 
72     // In case the access was just stale, try one more time.
73     if (!second_access_token_request_) {
74       second_access_token_request_ = true;
75       RequestAccessToken();
76       return;
77     }
78   }
79 
80   consumer_->OnUbertokenFailure(error);
81 }
82 
OnGetTokenSuccess(const OAuth2TokenService::Request * request,const std::string & access_token,const base::Time & expiration_time)83 void UbertokenFetcher::OnGetTokenSuccess(
84     const OAuth2TokenService::Request* request,
85     const std::string& access_token,
86     const base::Time& expiration_time) {
87   DCHECK(!access_token.empty());
88   access_token_ = access_token;
89   access_token_request_.reset();
90   ExchangeTokens();
91 }
92 
OnGetTokenFailure(const OAuth2TokenService::Request * request,const GoogleServiceAuthError & error)93 void UbertokenFetcher::OnGetTokenFailure(
94     const OAuth2TokenService::Request* request,
95     const GoogleServiceAuthError& error) {
96   access_token_request_.reset();
97   consumer_->OnUbertokenFailure(error);
98 }
99 
RequestAccessToken()100 void UbertokenFetcher::RequestAccessToken() {
101   retry_number_ = 0;
102   gaia_auth_fetcher_.reset();
103   retry_timer_.Stop();
104 
105   OAuth2TokenService::ScopeSet scopes;
106   scopes.insert(GaiaConstants::kOAuth1LoginScope);
107   access_token_request_ =
108       token_service_->StartRequest(account_id_, scopes, this);
109 }
110 
ExchangeTokens()111 void UbertokenFetcher::ExchangeTokens() {
112   gaia_auth_fetcher_.reset(new GaiaAuthFetcher(this,
113                                                GaiaConstants::kChromeSource,
114                                                request_context_));
115   gaia_auth_fetcher_->StartTokenFetchForUberAuthExchange(access_token_);
116 }
117