oauth2_api_call_flow_unittest.cc revision a1401311d1ab56c4ed0a474bd38c108f75cb0cd9
1// Copyright (c) 2012 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// A complete set of unit tests for OAuth2MintTokenFlow. 6 7#include <string> 8#include <vector> 9 10#include "base/memory/scoped_ptr.h" 11#include "base/message_loop/message_loop.h" 12#include "base/time/time.h" 13#include "google_apis/gaia/gaia_urls.h" 14#include "google_apis/gaia/google_service_auth_error.h" 15#include "google_apis/gaia/oauth2_access_token_consumer.h" 16#include "google_apis/gaia/oauth2_access_token_fetcher_impl.h" 17#include "google_apis/gaia/oauth2_api_call_flow.h" 18#include "net/http/http_request_headers.h" 19#include "net/http/http_status_code.h" 20#include "net/url_request/test_url_fetcher_factory.h" 21#include "net/url_request/url_fetcher.h" 22#include "net/url_request/url_fetcher_delegate.h" 23#include "net/url_request/url_fetcher_factory.h" 24#include "net/url_request/url_request.h" 25#include "net/url_request/url_request_status.h" 26#include "net/url_request/url_request_test_util.h" 27#include "testing/gmock/include/gmock/gmock.h" 28#include "testing/gtest/include/gtest/gtest.h" 29 30using net::HttpRequestHeaders; 31using net::ScopedURLFetcherFactory; 32using net::TestURLFetcher; 33using net::URLFetcher; 34using net::URLFetcherDelegate; 35using net::URLFetcherFactory; 36using net::URLRequestStatus; 37using testing::_; 38using testing::Return; 39 40namespace { 41 42static std::string CreateBody() { 43 return "some body"; 44} 45 46static GURL CreateApiUrl() { 47 return GURL("https://www.googleapis.com/someapi"); 48} 49 50static std::vector<std::string> CreateTestScopes() { 51 std::vector<std::string> scopes; 52 scopes.push_back("scope1"); 53 scopes.push_back("scope2"); 54 return scopes; 55} 56 57class MockUrlFetcherFactory : public ScopedURLFetcherFactory, 58 public URLFetcherFactory { 59 public: 60 MockUrlFetcherFactory() 61 : ScopedURLFetcherFactory(this) { 62 } 63 virtual ~MockUrlFetcherFactory() {} 64 65 MOCK_METHOD4( 66 CreateURLFetcher, 67 URLFetcher* (int id, 68 const GURL& url, 69 URLFetcher::RequestType request_type, 70 URLFetcherDelegate* d)); 71}; 72 73class MockAccessTokenFetcher : public OAuth2AccessTokenFetcherImpl { 74 public: 75 MockAccessTokenFetcher(OAuth2AccessTokenConsumer* consumer, 76 net::URLRequestContextGetter* getter, 77 const std::string& refresh_token) 78 : OAuth2AccessTokenFetcherImpl(consumer, getter, refresh_token) {} 79 ~MockAccessTokenFetcher() {} 80 81 MOCK_METHOD3(Start, 82 void(const std::string& client_id, 83 const std::string& client_secret, 84 const std::vector<std::string>& scopes)); 85}; 86 87class MockApiCallFlow : public OAuth2ApiCallFlow { 88 public: 89 MockApiCallFlow(net::URLRequestContextGetter* context, 90 const std::string& refresh_token, 91 const std::string& access_token, 92 const std::vector<std::string>& scopes) 93 : OAuth2ApiCallFlow(context, refresh_token, access_token, scopes) {} 94 ~MockApiCallFlow() {} 95 96 MOCK_METHOD0(CreateApiCallUrl, GURL ()); 97 MOCK_METHOD0(CreateApiCallBody, std::string ()); 98 MOCK_METHOD1(ProcessApiCallSuccess, 99 void (const URLFetcher* source)); 100 MOCK_METHOD1(ProcessApiCallFailure, 101 void (const URLFetcher* source)); 102 MOCK_METHOD1(ProcessNewAccessToken, 103 void (const std::string& access_token)); 104 MOCK_METHOD1(ProcessMintAccessTokenFailure, 105 void (const GoogleServiceAuthError& error)); 106 MOCK_METHOD0(CreateAccessTokenFetcher, OAuth2AccessTokenFetcher* ()); 107}; 108 109} // namespace 110 111class OAuth2ApiCallFlowTest : public testing::Test { 112 protected: 113 void SetupAccessTokenFetcher(const std::vector<std::string>& scopes) { 114 EXPECT_CALL(*access_token_fetcher_, 115 Start(GaiaUrls::GetInstance()->oauth2_chrome_client_id(), 116 GaiaUrls::GetInstance()->oauth2_chrome_client_secret(), 117 scopes)).Times(1); 118 EXPECT_CALL(*flow_, CreateAccessTokenFetcher()) 119 .WillOnce(Return(access_token_fetcher_.release())); 120 } 121 122 TestURLFetcher* CreateURLFetcher( 123 const GURL& url, bool fetch_succeeds, 124 int response_code, const std::string& body) { 125 TestURLFetcher* url_fetcher = new TestURLFetcher(0, url, flow_.get()); 126 URLRequestStatus::Status status = 127 fetch_succeeds ? URLRequestStatus::SUCCESS : URLRequestStatus::FAILED; 128 url_fetcher->set_status(URLRequestStatus(status, 0)); 129 130 if (response_code != 0) 131 url_fetcher->set_response_code(response_code); 132 133 if (!body.empty()) 134 url_fetcher->SetResponseString(body); 135 136 return url_fetcher; 137 } 138 139 void CreateFlow(const std::string& refresh_token, 140 const std::string& access_token, 141 const std::vector<std::string>& scopes) { 142 scoped_refptr<net::TestURLRequestContextGetter> request_context_getter = 143 new net::TestURLRequestContextGetter( 144 message_loop_.message_loop_proxy()); 145 flow_.reset(new MockApiCallFlow( 146 request_context_getter, refresh_token, access_token, scopes)); 147 access_token_fetcher_.reset(new MockAccessTokenFetcher( 148 flow_.get(), request_context_getter, refresh_token)); 149 } 150 151 TestURLFetcher* SetupApiCall(bool succeeds, net::HttpStatusCode status) { 152 std::string body(CreateBody()); 153 GURL url(CreateApiUrl()); 154 EXPECT_CALL(*flow_, CreateApiCallBody()).WillOnce(Return(body)); 155 EXPECT_CALL(*flow_, CreateApiCallUrl()).WillOnce(Return(url)); 156 TestURLFetcher* url_fetcher = 157 CreateURLFetcher(url, succeeds, status, std::string()); 158 EXPECT_CALL(factory_, CreateURLFetcher(_, url, _, _)) 159 .WillOnce(Return(url_fetcher)); 160 return url_fetcher; 161 } 162 163 MockUrlFetcherFactory factory_; 164 scoped_ptr<MockApiCallFlow> flow_; 165 scoped_ptr<MockAccessTokenFetcher> access_token_fetcher_; 166 base::MessageLoop message_loop_; 167}; 168 169TEST_F(OAuth2ApiCallFlowTest, FirstApiCallSucceeds) { 170 std::string rt = "refresh_token"; 171 std::string at = "access_token"; 172 std::vector<std::string> scopes(CreateTestScopes()); 173 174 CreateFlow(rt, at, scopes); 175 TestURLFetcher* url_fetcher = SetupApiCall(true, net::HTTP_OK); 176 EXPECT_CALL(*flow_, ProcessApiCallSuccess(url_fetcher)); 177 flow_->Start(); 178 flow_->OnURLFetchComplete(url_fetcher); 179} 180 181TEST_F(OAuth2ApiCallFlowTest, SecondApiCallSucceeds) { 182 std::string rt = "refresh_token"; 183 std::string at = "access_token"; 184 std::vector<std::string> scopes(CreateTestScopes()); 185 186 CreateFlow(rt, at, scopes); 187 TestURLFetcher* url_fetcher1 = SetupApiCall(true, net::HTTP_UNAUTHORIZED); 188 flow_->Start(); 189 SetupAccessTokenFetcher(scopes); 190 flow_->OnURLFetchComplete(url_fetcher1); 191 TestURLFetcher* url_fetcher2 = SetupApiCall(true, net::HTTP_OK); 192 EXPECT_CALL(*flow_, ProcessApiCallSuccess(url_fetcher2)); 193 flow_->OnGetTokenSuccess( 194 at, 195 base::Time::Now() + base::TimeDelta::FromMinutes(3600)); 196 flow_->OnURLFetchComplete(url_fetcher2); 197} 198 199TEST_F(OAuth2ApiCallFlowTest, SecondApiCallFails) { 200 std::string rt = "refresh_token"; 201 std::string at = "access_token"; 202 std::vector<std::string> scopes(CreateTestScopes()); 203 204 CreateFlow(rt, at, scopes); 205 TestURLFetcher* url_fetcher1 = SetupApiCall(true, net::HTTP_UNAUTHORIZED); 206 flow_->Start(); 207 SetupAccessTokenFetcher(scopes); 208 flow_->OnURLFetchComplete(url_fetcher1); 209 TestURLFetcher* url_fetcher2 = SetupApiCall(false, net::HTTP_UNAUTHORIZED); 210 EXPECT_CALL(*flow_, ProcessApiCallFailure(url_fetcher2)); 211 flow_->OnGetTokenSuccess( 212 at, 213 base::Time::Now() + base::TimeDelta::FromMinutes(3600)); 214 flow_->OnURLFetchComplete(url_fetcher2); 215} 216 217TEST_F(OAuth2ApiCallFlowTest, NewTokenGenerationFails) { 218 std::string rt = "refresh_token"; 219 std::string at = "access_token"; 220 std::vector<std::string> scopes(CreateTestScopes()); 221 222 CreateFlow(rt, at, scopes); 223 TestURLFetcher* url_fetcher = SetupApiCall(true, net::HTTP_UNAUTHORIZED); 224 flow_->Start(); 225 SetupAccessTokenFetcher(scopes); 226 flow_->OnURLFetchComplete(url_fetcher); 227 GoogleServiceAuthError error( 228 GoogleServiceAuthError::INVALID_GAIA_CREDENTIALS); 229 EXPECT_CALL(*flow_, ProcessMintAccessTokenFailure(error)); 230 flow_->OnGetTokenFailure(error); 231} 232 233TEST_F(OAuth2ApiCallFlowTest, EmptyAccessTokenFirstApiCallSucceeds) { 234 std::string rt = "refresh_token"; 235 std::string at = "access_token"; 236 std::vector<std::string> scopes(CreateTestScopes()); 237 238 CreateFlow(rt, std::string(), scopes); 239 SetupAccessTokenFetcher(scopes); 240 TestURLFetcher* url_fetcher = SetupApiCall(true, net::HTTP_OK); 241 EXPECT_CALL(*flow_, ProcessApiCallSuccess(url_fetcher)); 242 flow_->Start(); 243 flow_->OnGetTokenSuccess( 244 at, 245 base::Time::Now() + base::TimeDelta::FromMinutes(3600)); 246 flow_->OnURLFetchComplete(url_fetcher); 247} 248 249TEST_F(OAuth2ApiCallFlowTest, EmptyAccessTokenApiCallFails) { 250 std::string rt = "refresh_token"; 251 std::string at = "access_token"; 252 std::vector<std::string> scopes(CreateTestScopes()); 253 254 CreateFlow(rt, std::string(), scopes); 255 SetupAccessTokenFetcher(scopes); 256 TestURLFetcher* url_fetcher = SetupApiCall(false, net::HTTP_BAD_GATEWAY); 257 EXPECT_CALL(*flow_, ProcessApiCallFailure(url_fetcher)); 258 flow_->Start(); 259 flow_->OnGetTokenSuccess( 260 at, 261 base::Time::Now() + base::TimeDelta::FromMinutes(3600)); 262 flow_->OnURLFetchComplete(url_fetcher); 263} 264 265TEST_F(OAuth2ApiCallFlowTest, EmptyAccessTokenNewTokenGenerationFails) { 266 std::string rt = "refresh_token"; 267 std::string at = "access_token"; 268 std::vector<std::string> scopes(CreateTestScopes()); 269 270 CreateFlow(rt, std::string(), scopes); 271 SetupAccessTokenFetcher(scopes); 272 GoogleServiceAuthError error( 273 GoogleServiceAuthError::INVALID_GAIA_CREDENTIALS); 274 EXPECT_CALL(*flow_, ProcessMintAccessTokenFailure(error)); 275 flow_->Start(); 276 flow_->OnGetTokenFailure(error); 277} 278 279TEST_F(OAuth2ApiCallFlowTest, CreateURLFetcher) { 280 std::string rt = "refresh_token"; 281 std::string at = "access_token"; 282 std::vector<std::string> scopes(CreateTestScopes()); 283 std::string body = CreateBody(); 284 GURL url(CreateApiUrl()); 285 286 CreateFlow(rt, at, scopes); 287 scoped_ptr<TestURLFetcher> url_fetcher(SetupApiCall(true, net::HTTP_OK)); 288 flow_->CreateURLFetcher(); 289 HttpRequestHeaders headers; 290 url_fetcher->GetExtraRequestHeaders(&headers); 291 std::string auth_header; 292 EXPECT_TRUE(headers.GetHeader("Authorization", &auth_header)); 293 EXPECT_EQ("Bearer access_token", auth_header); 294 EXPECT_EQ(url, url_fetcher->GetOriginalURL()); 295 EXPECT_EQ(body, url_fetcher->upload_data()); 296} 297