test__client.py 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633
  1. # Copyright 2016 Google LLC
  2. #
  3. # Licensed under the Apache License, Version 2.0 (the "License");
  4. # you may not use this file except in compliance with the License.
  5. # You may obtain a copy of the License at
  6. #
  7. # http://www.apache.org/licenses/LICENSE-2.0
  8. #
  9. # Unless required by applicable law or agreed to in writing, software
  10. # distributed under the License is distributed on an "AS IS" BASIS,
  11. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. # See the License for the specific language governing permissions and
  13. # limitations under the License.
  14. import datetime
  15. import http.client as http_client
  16. import json
  17. import os
  18. import urllib
  19. import mock
  20. import pytest # type: ignore
  21. from google.auth import _helpers
  22. from google.auth import crypt
  23. from google.auth import exceptions
  24. from google.auth import iam
  25. from google.auth import jwt
  26. from google.auth import transport
  27. from google.oauth2 import _client
  28. import yatest.common as yc
  29. DATA_DIR = os.path.join(os.path.dirname(yc.source_path(__file__)), "..", "data")
  30. with open(os.path.join(DATA_DIR, "privatekey.pem"), "rb") as fh:
  31. PRIVATE_KEY_BYTES = fh.read()
  32. SIGNER = crypt.RSASigner.from_string(PRIVATE_KEY_BYTES, "1")
  33. SCOPES_AS_LIST = [
  34. "https://www.googleapis.com/auth/pubsub",
  35. "https://www.googleapis.com/auth/logging.write",
  36. ]
  37. SCOPES_AS_STRING = (
  38. "https://www.googleapis.com/auth/pubsub"
  39. " https://www.googleapis.com/auth/logging.write"
  40. )
  41. ACCESS_TOKEN_REQUEST_METRICS_HEADER_VALUE = (
  42. "gl-python/3.7 auth/1.1 auth-request-type/at cred-type/sa"
  43. )
  44. ID_TOKEN_REQUEST_METRICS_HEADER_VALUE = (
  45. "gl-python/3.7 auth/1.1 auth-request-type/it cred-type/sa"
  46. )
  47. @pytest.mark.parametrize("retryable", [True, False])
  48. def test__handle_error_response(retryable):
  49. response_data = {"error": "help", "error_description": "I'm alive"}
  50. with pytest.raises(exceptions.RefreshError) as excinfo:
  51. _client._handle_error_response(response_data, retryable)
  52. assert excinfo.value.retryable == retryable
  53. assert excinfo.match(r"help: I\'m alive")
  54. def test__handle_error_response_no_error():
  55. response_data = {"foo": "bar"}
  56. with pytest.raises(exceptions.RefreshError) as excinfo:
  57. _client._handle_error_response(response_data, False)
  58. assert not excinfo.value.retryable
  59. assert excinfo.match(r"{\"foo\": \"bar\"}")
  60. def test__handle_error_response_not_json():
  61. response_data = "this is an error message"
  62. with pytest.raises(exceptions.RefreshError) as excinfo:
  63. _client._handle_error_response(response_data, False)
  64. assert not excinfo.value.retryable
  65. assert excinfo.match(response_data)
  66. def test__can_retry_retryable():
  67. retryable_codes = transport.DEFAULT_RETRYABLE_STATUS_CODES
  68. for status_code in range(100, 600):
  69. if status_code in retryable_codes:
  70. assert _client._can_retry(status_code, {"error": "invalid_scope"})
  71. else:
  72. assert not _client._can_retry(status_code, {"error": "invalid_scope"})
  73. @pytest.mark.parametrize(
  74. "response_data", [{"error": "internal_failure"}, {"error": "server_error"}]
  75. )
  76. def test__can_retry_message(response_data):
  77. assert _client._can_retry(http_client.OK, response_data)
  78. @pytest.mark.parametrize(
  79. "response_data",
  80. [
  81. {"error": "invalid_scope"},
  82. {"error": {"foo": "bar"}},
  83. {"error_description": {"foo", "bar"}},
  84. ],
  85. )
  86. def test__can_retry_no_retry_message(response_data):
  87. assert not _client._can_retry(http_client.OK, response_data)
  88. @pytest.mark.parametrize("mock_expires_in", [500, "500"])
  89. @mock.patch("google.auth._helpers.utcnow", return_value=datetime.datetime.min)
  90. def test__parse_expiry(unused_utcnow, mock_expires_in):
  91. result = _client._parse_expiry({"expires_in": mock_expires_in})
  92. assert result == datetime.datetime.min + datetime.timedelta(seconds=500)
  93. def test__parse_expiry_none():
  94. assert _client._parse_expiry({}) is None
  95. def make_request(response_data, status=http_client.OK):
  96. response = mock.create_autospec(transport.Response, instance=True)
  97. response.status = status
  98. response.data = json.dumps(response_data).encode("utf-8")
  99. request = mock.create_autospec(transport.Request)
  100. request.return_value = response
  101. return request
  102. def test__token_endpoint_request():
  103. request = make_request({"test": "response"})
  104. result = _client._token_endpoint_request(
  105. request, "http://example.com", {"test": "params"}
  106. )
  107. # Check request call
  108. request.assert_called_with(
  109. method="POST",
  110. url="http://example.com",
  111. headers={"Content-Type": "application/x-www-form-urlencoded"},
  112. body="test=params".encode("utf-8"),
  113. )
  114. # Check result
  115. assert result == {"test": "response"}
  116. def test__token_endpoint_request_use_json():
  117. request = make_request({"test": "response"})
  118. result = _client._token_endpoint_request(
  119. request,
  120. "http://example.com",
  121. {"test": "params"},
  122. access_token="access_token",
  123. use_json=True,
  124. )
  125. # Check request call
  126. request.assert_called_with(
  127. method="POST",
  128. url="http://example.com",
  129. headers={
  130. "Content-Type": "application/json",
  131. "Authorization": "Bearer access_token",
  132. },
  133. body=b'{"test": "params"}',
  134. )
  135. # Check result
  136. assert result == {"test": "response"}
  137. def test__token_endpoint_request_error():
  138. request = make_request({}, status=http_client.BAD_REQUEST)
  139. with pytest.raises(exceptions.RefreshError):
  140. _client._token_endpoint_request(request, "http://example.com", {})
  141. def test__token_endpoint_request_internal_failure_error():
  142. request = make_request(
  143. {"error_description": "internal_failure"}, status=http_client.BAD_REQUEST
  144. )
  145. with pytest.raises(exceptions.RefreshError):
  146. _client._token_endpoint_request(
  147. request, "http://example.com", {"error_description": "internal_failure"}
  148. )
  149. # request with 2 retries
  150. assert request.call_count == 3
  151. request = make_request(
  152. {"error": "internal_failure"}, status=http_client.BAD_REQUEST
  153. )
  154. with pytest.raises(exceptions.RefreshError):
  155. _client._token_endpoint_request(
  156. request, "http://example.com", {"error": "internal_failure"}
  157. )
  158. # request with 2 retries
  159. assert request.call_count == 3
  160. def test__token_endpoint_request_internal_failure_and_retry_failure_error():
  161. retryable_error = mock.create_autospec(transport.Response, instance=True)
  162. retryable_error.status = http_client.BAD_REQUEST
  163. retryable_error.data = json.dumps({"error_description": "internal_failure"}).encode(
  164. "utf-8"
  165. )
  166. unretryable_error = mock.create_autospec(transport.Response, instance=True)
  167. unretryable_error.status = http_client.BAD_REQUEST
  168. unretryable_error.data = json.dumps({"error_description": "invalid_scope"}).encode(
  169. "utf-8"
  170. )
  171. request = mock.create_autospec(transport.Request)
  172. request.side_effect = [retryable_error, retryable_error, unretryable_error]
  173. with pytest.raises(exceptions.RefreshError):
  174. _client._token_endpoint_request(
  175. request, "http://example.com", {"error_description": "invalid_scope"}
  176. )
  177. # request should be called three times. Two retryable errors and one
  178. # unretryable error to break the retry loop.
  179. assert request.call_count == 3
  180. def test__token_endpoint_request_internal_failure_and_retry_succeeds():
  181. retryable_error = mock.create_autospec(transport.Response, instance=True)
  182. retryable_error.status = http_client.BAD_REQUEST
  183. retryable_error.data = json.dumps({"error_description": "internal_failure"}).encode(
  184. "utf-8"
  185. )
  186. response = mock.create_autospec(transport.Response, instance=True)
  187. response.status = http_client.OK
  188. response.data = json.dumps({"hello": "world"}).encode("utf-8")
  189. request = mock.create_autospec(transport.Request)
  190. request.side_effect = [retryable_error, response]
  191. _ = _client._token_endpoint_request(
  192. request, "http://example.com", {"test": "params"}
  193. )
  194. assert request.call_count == 2
  195. def test__token_endpoint_request_string_error():
  196. response = mock.create_autospec(transport.Response, instance=True)
  197. response.status = http_client.BAD_REQUEST
  198. response.data = "this is an error message"
  199. request = mock.create_autospec(transport.Request)
  200. request.return_value = response
  201. with pytest.raises(exceptions.RefreshError) as excinfo:
  202. _client._token_endpoint_request(request, "http://example.com", {})
  203. assert excinfo.match("this is an error message")
  204. def verify_request_params(request, params):
  205. request_body = request.call_args[1]["body"].decode("utf-8")
  206. request_params = urllib.parse.parse_qs(request_body)
  207. for key, value in params.items():
  208. assert request_params[key][0] == value
  209. @mock.patch("google.auth._helpers.utcnow", return_value=datetime.datetime.min)
  210. def test_jwt_grant(utcnow):
  211. request = make_request(
  212. {"access_token": "token", "expires_in": 500, "extra": "data"}
  213. )
  214. token, expiry, extra_data = _client.jwt_grant(
  215. request, "http://example.com", "assertion_value"
  216. )
  217. # Check request call
  218. verify_request_params(
  219. request, {"grant_type": _client._JWT_GRANT_TYPE, "assertion": "assertion_value"}
  220. )
  221. # Check result
  222. assert token == "token"
  223. assert expiry == utcnow() + datetime.timedelta(seconds=500)
  224. assert extra_data["extra"] == "data"
  225. def test_jwt_grant_no_access_token():
  226. request = make_request(
  227. {
  228. # No access token.
  229. "expires_in": 500,
  230. "extra": "data",
  231. }
  232. )
  233. with pytest.raises(exceptions.RefreshError) as excinfo:
  234. _client.jwt_grant(request, "http://example.com", "assertion_value")
  235. assert not excinfo.value.retryable
  236. def test_call_iam_generate_id_token_endpoint():
  237. now = _helpers.utcnow()
  238. id_token_expiry = _helpers.datetime_to_secs(now)
  239. id_token = jwt.encode(SIGNER, {"exp": id_token_expiry}).decode("utf-8")
  240. request = make_request({"token": id_token})
  241. token, expiry = _client.call_iam_generate_id_token_endpoint(
  242. request,
  243. iam._IAM_IDTOKEN_ENDPOINT,
  244. "fake_email",
  245. "fake_audience",
  246. "fake_access_token",
  247. "googleapis.com",
  248. )
  249. assert (
  250. request.call_args[1]["url"]
  251. == "https://iamcredentials.googleapis.com/v1/projects/-/serviceAccounts/fake_email:generateIdToken"
  252. )
  253. assert request.call_args[1]["headers"]["Content-Type"] == "application/json"
  254. assert (
  255. request.call_args[1]["headers"]["Authorization"] == "Bearer fake_access_token"
  256. )
  257. response_body = json.loads(request.call_args[1]["body"])
  258. assert response_body["audience"] == "fake_audience"
  259. assert response_body["includeEmail"] == "true"
  260. assert response_body["useEmailAzp"] == "true"
  261. # Check result
  262. assert token == id_token
  263. # JWT does not store microseconds
  264. now = now.replace(microsecond=0)
  265. assert expiry == now
  266. def test_call_iam_generate_id_token_endpoint_no_id_token():
  267. request = make_request(
  268. {
  269. # No access token.
  270. "error": "no token"
  271. }
  272. )
  273. with pytest.raises(exceptions.RefreshError) as excinfo:
  274. _client.call_iam_generate_id_token_endpoint(
  275. request,
  276. iam._IAM_IDTOKEN_ENDPOINT,
  277. "fake_email",
  278. "fake_audience",
  279. "fake_access_token",
  280. "googleapis.com",
  281. )
  282. assert excinfo.match("No ID token in response")
  283. def test_id_token_jwt_grant():
  284. now = _helpers.utcnow()
  285. id_token_expiry = _helpers.datetime_to_secs(now)
  286. id_token = jwt.encode(SIGNER, {"exp": id_token_expiry}).decode("utf-8")
  287. request = make_request({"id_token": id_token, "extra": "data"})
  288. token, expiry, extra_data = _client.id_token_jwt_grant(
  289. request, "http://example.com", "assertion_value"
  290. )
  291. # Check request call
  292. verify_request_params(
  293. request, {"grant_type": _client._JWT_GRANT_TYPE, "assertion": "assertion_value"}
  294. )
  295. # Check result
  296. assert token == id_token
  297. # JWT does not store microseconds
  298. now = now.replace(microsecond=0)
  299. assert expiry == now
  300. assert extra_data["extra"] == "data"
  301. def test_id_token_jwt_grant_no_access_token():
  302. request = make_request(
  303. {
  304. # No access token.
  305. "expires_in": 500,
  306. "extra": "data",
  307. }
  308. )
  309. with pytest.raises(exceptions.RefreshError) as excinfo:
  310. _client.id_token_jwt_grant(request, "http://example.com", "assertion_value")
  311. assert not excinfo.value.retryable
  312. @mock.patch("google.auth._helpers.utcnow", return_value=datetime.datetime.min)
  313. def test_refresh_grant(unused_utcnow):
  314. request = make_request(
  315. {
  316. "access_token": "token",
  317. "refresh_token": "new_refresh_token",
  318. "expires_in": 500,
  319. "extra": "data",
  320. }
  321. )
  322. token, refresh_token, expiry, extra_data = _client.refresh_grant(
  323. request,
  324. "http://example.com",
  325. "refresh_token",
  326. "client_id",
  327. "client_secret",
  328. rapt_token="rapt_token",
  329. )
  330. # Check request call
  331. verify_request_params(
  332. request,
  333. {
  334. "grant_type": _client._REFRESH_GRANT_TYPE,
  335. "refresh_token": "refresh_token",
  336. "client_id": "client_id",
  337. "client_secret": "client_secret",
  338. "rapt": "rapt_token",
  339. },
  340. )
  341. # Check result
  342. assert token == "token"
  343. assert refresh_token == "new_refresh_token"
  344. assert expiry == datetime.datetime.min + datetime.timedelta(seconds=500)
  345. assert extra_data["extra"] == "data"
  346. @mock.patch("google.auth._helpers.utcnow", return_value=datetime.datetime.min)
  347. def test_refresh_grant_with_scopes(unused_utcnow):
  348. request = make_request(
  349. {
  350. "access_token": "token",
  351. "refresh_token": "new_refresh_token",
  352. "expires_in": 500,
  353. "extra": "data",
  354. "scope": SCOPES_AS_STRING,
  355. }
  356. )
  357. token, refresh_token, expiry, extra_data = _client.refresh_grant(
  358. request,
  359. "http://example.com",
  360. "refresh_token",
  361. "client_id",
  362. "client_secret",
  363. SCOPES_AS_LIST,
  364. )
  365. # Check request call.
  366. verify_request_params(
  367. request,
  368. {
  369. "grant_type": _client._REFRESH_GRANT_TYPE,
  370. "refresh_token": "refresh_token",
  371. "client_id": "client_id",
  372. "client_secret": "client_secret",
  373. "scope": SCOPES_AS_STRING,
  374. },
  375. )
  376. # Check result.
  377. assert token == "token"
  378. assert refresh_token == "new_refresh_token"
  379. assert expiry == datetime.datetime.min + datetime.timedelta(seconds=500)
  380. assert extra_data["extra"] == "data"
  381. def test_refresh_grant_no_access_token():
  382. request = make_request(
  383. {
  384. # No access token.
  385. "refresh_token": "new_refresh_token",
  386. "expires_in": 500,
  387. "extra": "data",
  388. }
  389. )
  390. with pytest.raises(exceptions.RefreshError) as excinfo:
  391. _client.refresh_grant(
  392. request, "http://example.com", "refresh_token", "client_id", "client_secret"
  393. )
  394. assert not excinfo.value.retryable
  395. @mock.patch(
  396. "google.auth.metrics.token_request_access_token_sa_assertion",
  397. return_value=ACCESS_TOKEN_REQUEST_METRICS_HEADER_VALUE,
  398. )
  399. @mock.patch("google.oauth2._client._parse_expiry", return_value=None)
  400. @mock.patch.object(_client, "_token_endpoint_request", autospec=True)
  401. def test_jwt_grant_retry_default(
  402. mock_token_endpoint_request, mock_expiry, mock_metrics_header_value
  403. ):
  404. _client.jwt_grant(mock.Mock(), mock.Mock(), mock.Mock())
  405. mock_token_endpoint_request.assert_called_with(
  406. mock.ANY,
  407. mock.ANY,
  408. mock.ANY,
  409. can_retry=True,
  410. headers={"x-goog-api-client": ACCESS_TOKEN_REQUEST_METRICS_HEADER_VALUE},
  411. )
  412. @pytest.mark.parametrize("can_retry", [True, False])
  413. @mock.patch(
  414. "google.auth.metrics.token_request_access_token_sa_assertion",
  415. return_value=ACCESS_TOKEN_REQUEST_METRICS_HEADER_VALUE,
  416. )
  417. @mock.patch("google.oauth2._client._parse_expiry", return_value=None)
  418. @mock.patch.object(_client, "_token_endpoint_request", autospec=True)
  419. def test_jwt_grant_retry_with_retry(
  420. mock_token_endpoint_request, mock_expiry, mock_metrics_header_value, can_retry
  421. ):
  422. _client.jwt_grant(mock.Mock(), mock.Mock(), mock.Mock(), can_retry=can_retry)
  423. mock_token_endpoint_request.assert_called_with(
  424. mock.ANY,
  425. mock.ANY,
  426. mock.ANY,
  427. can_retry=can_retry,
  428. headers={"x-goog-api-client": ACCESS_TOKEN_REQUEST_METRICS_HEADER_VALUE},
  429. )
  430. @mock.patch(
  431. "google.auth.metrics.token_request_id_token_sa_assertion",
  432. return_value=ID_TOKEN_REQUEST_METRICS_HEADER_VALUE,
  433. )
  434. @mock.patch("google.auth.jwt.decode", return_value={"exp": 0})
  435. @mock.patch.object(_client, "_token_endpoint_request", autospec=True)
  436. def test_id_token_jwt_grant_retry_default(
  437. mock_token_endpoint_request, mock_jwt_decode, mock_metrics_header_value
  438. ):
  439. _client.id_token_jwt_grant(mock.Mock(), mock.Mock(), mock.Mock())
  440. mock_token_endpoint_request.assert_called_with(
  441. mock.ANY,
  442. mock.ANY,
  443. mock.ANY,
  444. can_retry=True,
  445. headers={"x-goog-api-client": ID_TOKEN_REQUEST_METRICS_HEADER_VALUE},
  446. )
  447. @pytest.mark.parametrize("can_retry", [True, False])
  448. @mock.patch(
  449. "google.auth.metrics.token_request_id_token_sa_assertion",
  450. return_value=ID_TOKEN_REQUEST_METRICS_HEADER_VALUE,
  451. )
  452. @mock.patch("google.auth.jwt.decode", return_value={"exp": 0})
  453. @mock.patch.object(_client, "_token_endpoint_request", autospec=True)
  454. def test_id_token_jwt_grant_retry_with_retry(
  455. mock_token_endpoint_request, mock_jwt_decode, mock_metrics_header_value, can_retry
  456. ):
  457. _client.id_token_jwt_grant(
  458. mock.Mock(), mock.Mock(), mock.Mock(), can_retry=can_retry
  459. )
  460. mock_token_endpoint_request.assert_called_with(
  461. mock.ANY,
  462. mock.ANY,
  463. mock.ANY,
  464. can_retry=can_retry,
  465. headers={"x-goog-api-client": ID_TOKEN_REQUEST_METRICS_HEADER_VALUE},
  466. )
  467. @mock.patch("google.oauth2._client._parse_expiry", return_value=None)
  468. @mock.patch.object(_client, "_token_endpoint_request", autospec=True)
  469. def test_refresh_grant_retry_default(mock_token_endpoint_request, mock_parse_expiry):
  470. _client.refresh_grant(
  471. mock.Mock(), mock.Mock(), mock.Mock(), mock.Mock(), mock.Mock()
  472. )
  473. mock_token_endpoint_request.assert_called_with(
  474. mock.ANY, mock.ANY, mock.ANY, can_retry=True
  475. )
  476. @pytest.mark.parametrize("can_retry", [True, False])
  477. @mock.patch("google.oauth2._client._parse_expiry", return_value=None)
  478. @mock.patch.object(_client, "_token_endpoint_request", autospec=True)
  479. def test_refresh_grant_retry_with_retry(
  480. mock_token_endpoint_request, mock_parse_expiry, can_retry
  481. ):
  482. _client.refresh_grant(
  483. mock.Mock(),
  484. mock.Mock(),
  485. mock.Mock(),
  486. mock.Mock(),
  487. mock.Mock(),
  488. can_retry=can_retry,
  489. )
  490. mock_token_endpoint_request.assert_called_with(
  491. mock.ANY, mock.ANY, mock.ANY, can_retry=can_retry
  492. )
  493. @pytest.mark.parametrize("can_retry", [True, False])
  494. def test__token_endpoint_request_no_throw_with_retry(can_retry):
  495. response_data = {"error": "help", "error_description": "I'm alive"}
  496. body = "dummy body"
  497. mock_response = mock.create_autospec(transport.Response, instance=True)
  498. mock_response.status = http_client.INTERNAL_SERVER_ERROR
  499. mock_response.data = json.dumps(response_data).encode("utf-8")
  500. mock_request = mock.create_autospec(transport.Request)
  501. mock_request.return_value = mock_response
  502. _client._token_endpoint_request_no_throw(
  503. mock_request, mock.Mock(), body, mock.Mock(), mock.Mock(), can_retry=can_retry
  504. )
  505. if can_retry:
  506. assert mock_request.call_count == 3
  507. else:
  508. assert mock_request.call_count == 1