1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177
|
from __future__ import annotations
import typing as t
import uuid
from unittest.mock import MagicMock, Mock
import pytest
import globus_sdk
from globus_sdk import (
GlobusAppConfig,
IDTokenDecoder,
NativeAppAuthClient,
OAuthTokenResponse,
TransferClient,
UserApp,
)
from globus_sdk.gare import GlobusAuthorizationParameters
from globus_sdk.login_flows import LoginFlowManager
from globus_sdk.scopes import Scope, TransferScopes
from globus_sdk.testing import RegisteredResponse
from globus_sdk.token_storage import MemoryTokenStorage, TokenStorageData
class GlobusAppConfigurator:
def __init__(
self,
starting_tokens: dict[str, list[Scope]] | None = None,
login_tokens: dict[str, list[Scope]] | None = None,
) -> None:
self.identity_id = str(uuid.uuid4())
self.token_storage = MemoryTokenStorage()
if starting_tokens:
self.token_storage.store_token_data_by_resource_server(
{
resource_server: self._generate_token_data(resource_server, scopes)
for resource_server, scopes in starting_tokens.items()
}
)
self._login_tokens: list[TokenStorageData] = [
self._generate_token_data(resource_server, scopes)
for resource_server, scopes in login_tokens.items() or {}.items()
]
self.login_flow_manager = _FakeLoginFlowManager(self._login_tokens)
def _generate_token_data(
self, resource_server: str, scopes: list[Scope]
) -> TokenStorageData:
return TokenStorageData(
identity_id=self.identity_id,
resource_server=resource_server,
scope=" ".join(str(s) for s in scopes),
access_token="generated-access-token",
refresh_token="generated-refresh-token",
expires_at_seconds=9999999999,
token_type="Bearer",
)
def config(self, **kwargs: t.Any) -> GlobusAppConfig:
decoder = MagicMock(spec=IDTokenDecoder)
decoder.decode.return_value = {"sub": self.identity_id}
return GlobusAppConfig(
login_flow_manager=self.login_flow_manager,
id_token_decoder=decoder,
token_storage=self.token_storage,
**kwargs,
)
class _FakeLoginFlowManager(LoginFlowManager):
def __init__(self, starting_tokens: list[TokenStorageData]) -> None:
super().__init__(login_client=MagicMock(spec=NativeAppAuthClient))
self._by_resource_server = {
token.resource_server: token.to_dict() for token in starting_tokens
}
self.login_count = 0
def run_login_flow(
self, auth_parameters: GlobusAuthorizationParameters
) -> OAuthTokenResponse:
self.login_count += 1
response = Mock()
response.id_token = "abcdefghjiklmnop"
response.by_resource_server = self._by_resource_server
return response
_GET_TASK_SUCCESS_RESPONSE = RegisteredResponse(
service="transfer",
path="/v0.10/task/foobar",
method="GET",
json={"task_id": "foobar"},
)
_GET_TASK_GARE_RESPONSE = RegisteredResponse(
service="transfer",
path="/v0.10/task/foobar",
method="GET",
status=403,
json={
"code": "AuthorizationRequired",
"authorization_parameters": {"required_scopes": ["my-cool-new-scope"]},
},
)
def test_app_can_redrive_gares():
"""
When enabled:
If an app-registered client encounters a GARE-compatible 403 http response, it
should "redrive" it by:
1. Running a login flow with the gare-included authorization
2. Re-attempting the original request with the new token(s)
"""
transfer_rs = TransferClient.resource_server
configurator = GlobusAppConfigurator(
# Start with a valid Transfer:all token
starting_tokens={transfer_rs: [TransferScopes.all]},
# On login, receive both Transfer:all and the gare-required scope
login_tokens={transfer_rs: [TransferScopes.all, Scope("my-cool-new-scope")]},
)
config = configurator.config(auto_redrive_gares=True)
app = UserApp(client_id="client_id", config=config)
transfer = TransferClient(app=app)
# Set up first a GARE response, then a successful response (on retry)
_GET_TASK_GARE_RESPONSE.add()
_GET_TASK_SUCCESS_RESPONSE.add()
assert transfer.get_task("foobar").http_status == 200
assert configurator.login_flow_manager.login_count == 1
def test_app_gare_redriving_is_disabled_by_default():
transfer_rs = TransferClient.resource_server
configurator = GlobusAppConfigurator(
# Start with a valid Transfer:all token
starting_tokens={transfer_rs: [TransferScopes.all]},
# On login, receive both Transfer:all and the gare-required scope
login_tokens={transfer_rs: [TransferScopes.all, Scope("my-cool-new-scope")]},
)
# Don't override the `auto_redrive_gares` default of False
config = configurator.config()
app = UserApp(client_id="client_id", config=config)
transfer = TransferClient(app=app)
# Set up first a GARE response, then a successful response (on retry)
_GET_TASK_GARE_RESPONSE.add()
_GET_TASK_SUCCESS_RESPONSE.add()
with pytest.raises(globus_sdk.GlobusAPIError):
transfer.get_task("foobar")
assert configurator.login_flow_manager.login_count == 0
def test_app_gare_redrive_only_occurs_once_per_request():
transfer_rs = TransferClient.resource_server
configurator = GlobusAppConfigurator(
# Start with a valid Transfer:all token
starting_tokens={transfer_rs: [TransferScopes.all]},
# On login, receive Transfer:all and a random scope (not required by the gare).
login_tokens={transfer_rs: [TransferScopes.all, Scope("my-stupid-new-scope")]},
)
config = configurator.config(auto_redrive_gares=True)
app = UserApp(client_id="client_id", config=config)
transfer = TransferClient(app=app)
# Set up exclusively a GARE response which will retry indefinitely.
_GET_TASK_GARE_RESPONSE.add()
with pytest.raises(globus_sdk.GlobusAPIError):
transfer.get_task("foobar")
assert configurator.login_flow_manager.login_count == 1
|