test__client.py 19 KB

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