Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: allow scopes for self signed jwt #776

Merged
merged 4 commits into from Jun 16, 2021
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Jump to
Jump to file
Failed to load files.
Diff view
Diff view
3 changes: 2 additions & 1 deletion google/auth/jwt.py
Expand Up @@ -525,8 +525,9 @@ def _make_jwt(self):
"sub": self._subject,
"iat": _helpers.datetime_to_secs(now),
"exp": _helpers.datetime_to_secs(expiry),
"aud": self._audience,
}
if self._audience:
payload.update({"aud": self._audience})
arithmetic1728 marked this conversation as resolved.
Show resolved Hide resolved
arithmetic1728 marked this conversation as resolved.
Show resolved Hide resolved

payload.update(self._additional_claims)

Expand Down
7 changes: 2 additions & 5 deletions google/auth/transport/grpc.py
Expand Up @@ -79,12 +79,9 @@ def _get_authorization_headers(self, context):
# Attempt to use self-signed JWTs when a service account is used.
# A default host must be explicitly provided since it cannot always
# be determined from the context.service_url.
if (
isinstance(self._credentials, service_account.Credentials)
and self._default_host
):
if isinstance(self._credentials, service_account.Credentials):
self._credentials._create_self_signed_jwt(
"https://{}/".format(self._default_host)
"https://{}/".format(self._default_host) if self._default_host else None
)

self._credentials.before_request(
Expand Down
8 changes: 2 additions & 6 deletions google/auth/transport/requests.py
Expand Up @@ -358,13 +358,9 @@ def __init__(

# https://google.aip.dev/auth/4111
# Attempt to use self-signed JWTs when a service account is used.
# A default host must be explicitly provided.
if (
isinstance(self.credentials, service_account.Credentials)
and self._default_host
):
if isinstance(self.credentials, service_account.Credentials):
self.credentials._create_self_signed_jwt(
"https://{}/".format(self._default_host)
"https://{}/".format(self._default_host) if self._default_host else None
)

def configure_mtls_channel(self, client_cert_callback=None):
Expand Down
8 changes: 2 additions & 6 deletions google/auth/transport/urllib3.py
Expand Up @@ -293,13 +293,9 @@ def __init__(

# https://google.aip.dev/auth/4111
# Attempt to use self-signed JWTs when a service account is used.
# A default host must be explicitly provided.
if (
isinstance(self.credentials, service_account.Credentials)
and self._default_host
):
if isinstance(self.credentials, service_account.Credentials):
self.credentials._create_self_signed_jwt(
"https://{}/".format(self._default_host)
"https://{}/".format(self._default_host) if self._default_host else None
)

super(AuthorizedHttp, self).__init__()
Expand Down
49 changes: 47 additions & 2 deletions google/oauth2/service_account.py
Expand Up @@ -131,6 +131,7 @@ def __init__(
project_id=None,
quota_project_id=None,
additional_claims=None,
always_use_jwt_access=False,
):
"""
Args:
Expand All @@ -149,6 +150,8 @@ def __init__(
billing.
additional_claims (Mapping[str, str]): Any additional claims for
the JWT assertion used in the authorization grant.
always_use_jwt_access (Optional[bool]): Whether self signed JWT should
be always used.

.. note:: Typically one of the helper constructors
:meth:`from_service_account_file` or
Expand All @@ -165,6 +168,7 @@ def __init__(
self._project_id = project_id
self._quota_project_id = quota_project_id
self._token_uri = token_uri
self._always_use_jwt_access = always_use_jwt_access

self._jwt_credentials = None

Expand Down Expand Up @@ -266,6 +270,30 @@ def with_scopes(self, scopes, default_scopes=None):
project_id=self._project_id,
quota_project_id=self._quota_project_id,
additional_claims=self._additional_claims.copy(),
always_use_jwt_access=self._always_use_jwt_access,
)

def with_always_use_jwt_access(self, always_use_jwt_access):
"""Create a copy of these credentials with the specified always_use_jwt_access value.

Args:
always_use_jwt_access (bool): Whether always use self signed JWT or not.

Returns:
google.auth.service_account.Credentials: A new credentials
instance.
"""
return self.__class__(
self._signer,
service_account_email=self._service_account_email,
scopes=self._scopes,
default_scopes=self._default_scopes,
token_uri=self._token_uri,
subject=self._subject,
project_id=self._project_id,
quota_project_id=self._quota_project_id,
additional_claims=self._additional_claims.copy(),
always_use_jwt_access=always_use_jwt_access,
)

def with_subject(self, subject):
Expand All @@ -288,6 +316,7 @@ def with_subject(self, subject):
project_id=self._project_id,
quota_project_id=self._quota_project_id,
additional_claims=self._additional_claims.copy(),
always_use_jwt_access=self._always_use_jwt_access,
)

def with_claims(self, additional_claims):
Expand Down Expand Up @@ -315,6 +344,7 @@ def with_claims(self, additional_claims):
project_id=self._project_id,
quota_project_id=self._quota_project_id,
additional_claims=new_additional_claims,
always_use_jwt_access=self._always_use_jwt_access,
)

@_helpers.copy_docstring(credentials.CredentialsWithQuotaProject)
Expand All @@ -330,6 +360,7 @@ def with_quota_project(self, quota_project_id):
project_id=self._project_id,
quota_project_id=quota_project_id,
additional_claims=self._additional_claims.copy(),
always_use_jwt_access=self._always_use_jwt_access,
)

def _make_authorization_grant_assertion(self):
Expand Down Expand Up @@ -386,8 +417,22 @@ def _create_self_signed_jwt(self, audience):
audience (str): The service URL. ``https://[API_ENDPOINT]/``
"""
# https://google.aip.dev/auth/4111
# If the user has not defined scopes, create a self-signed jwt
if not self.scopes:
if self._always_use_jwt_access:
if self._scopes:
self._jwt_credentials = jwt.Credentials.from_signing_credentials(
self, None, additional_claims={"scope": " ".join(self._scopes)}
)
elif audience:
self._jwt_credentials = jwt.Credentials.from_signing_credentials(
self, audience
)
Comment on lines +425 to +428
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think the way google-api-core is written it will always pass through an audience. https://github.com/googleapis/python-api-core/blob/155da5e18cc2fdcfa57de6f956b7d078e79cd4b7/google/api_core/grpc_helpers.py#L249-L251 and the elif self._default_scopes case will never be reached.

Is it necessary to distinguish between the "default" audience and a user defined audience? It looks like Cody asked a question about this in the doc as well.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it is fine that self._default_scopes is never reached. We can just leave it here for logic completeness.

We don't need to distinguish a default audience and a user defined audience. If the user doesn't provide the scope but provides the audience, then they are responsible that the audience is correct.

elif self._default_scopes:
self._jwt_credentials = jwt.Credentials.from_signing_credentials(
self,
None,
additional_claims={"scope": " ".join(self._default_scopes)},
)
elif not self._scopes and audience:
self._jwt_credentials = jwt.Credentials.from_signing_credentials(
self, audience
)
Expand Down
66 changes: 66 additions & 0 deletions tests/oauth2/test_service_account.py
Expand Up @@ -155,6 +155,13 @@ def test_with_quota_project(self):
new_credentials.apply(hdrs, token="tok")
assert "x-goog-user-project" in hdrs

def test__with_always_use_jwt_access(self):
credentials = self.make_credentials()
assert not credentials._always_use_jwt_access

new_credentials = credentials.with_always_use_jwt_access(True)
assert new_credentials._always_use_jwt_access

def test__make_authorization_grant_assertion(self):
credentials = self.make_credentials()
token = credentials._make_authorization_grant_assertion()
Expand Down Expand Up @@ -225,6 +232,65 @@ def test__create_self_signed_jwt_with_user_scopes(self, jwt):
# JWT should not be created if there are user-defined scopes
jwt.from_signing_credentials.assert_not_called()

@mock.patch("google.auth.jwt.Credentials", instance=True, autospec=True)
def test__create_self_signed_jwt_always_use_jwt_access_with_audience(self, jwt):
credentials = service_account.Credentials(
SIGNER,
self.SERVICE_ACCOUNT_EMAIL,
self.TOKEN_URI,
default_scopes=["bar", "foo"],
always_use_jwt_access=True,
)

audience = "https://pubsub.googleapis.com"
credentials._create_self_signed_jwt(audience)
jwt.from_signing_credentials.assert_called_once_with(credentials, audience)

@mock.patch("google.auth.jwt.Credentials", instance=True, autospec=True)
def test__create_self_signed_jwt_always_use_jwt_access_with_scopes(self, jwt):
credentials = service_account.Credentials(
SIGNER,
self.SERVICE_ACCOUNT_EMAIL,
self.TOKEN_URI,
scopes=["bar", "foo"],
always_use_jwt_access=True,
)

audience = "https://pubsub.googleapis.com"
credentials._create_self_signed_jwt(audience)
jwt.from_signing_credentials.assert_called_once_with(
credentials, None, additional_claims={"scope": "bar foo"}
)

@mock.patch("google.auth.jwt.Credentials", instance=True, autospec=True)
def test__create_self_signed_jwt_always_use_jwt_access_with_default_scopes(
self, jwt
):
credentials = service_account.Credentials(
SIGNER,
self.SERVICE_ACCOUNT_EMAIL,
self.TOKEN_URI,
default_scopes=["bar", "foo"],
always_use_jwt_access=True,
)

credentials._create_self_signed_jwt(None)
jwt.from_signing_credentials.assert_called_once_with(
credentials, None, additional_claims={"scope": "bar foo"}
)

@mock.patch("google.auth.jwt.Credentials", instance=True, autospec=True)
def test__create_self_signed_jwt_always_use_jwt_access(self, jwt):
credentials = service_account.Credentials(
SIGNER,
self.SERVICE_ACCOUNT_EMAIL,
self.TOKEN_URI,
always_use_jwt_access=True,
)

credentials._create_self_signed_jwt(None)
jwt.from_signing_credentials.assert_not_called()

@mock.patch("google.oauth2._client.jwt_grant", autospec=True)
def test_refresh_success(self, jwt_grant):
credentials = self.make_credentials()
Expand Down
12 changes: 12 additions & 0 deletions tests/test_jwt.py
Expand Up @@ -390,6 +390,18 @@ def test_with_claims(self):
assert new_credentials._additional_claims == self.credentials._additional_claims
assert new_credentials._quota_project_id == self.credentials._quota_project_id

def test__make_jwt_without_audience(self):
cred = jwt.Credentials.from_service_account_info(
SERVICE_ACCOUNT_INFO.copy(),
subject=self.SUBJECT,
audience=None,
additional_claims={"scope": "foo bar"},
)
token, _ = cred._make_jwt()
payload = jwt.decode(token, PUBLIC_CERT_BYTES)
assert payload["scope"] == "foo bar"
assert "aud" not in payload

def test_with_quota_project(self):
quota_project_id = "project-foo"

Expand Down
3 changes: 1 addition & 2 deletions tests/transport/test_grpc.py
Expand Up @@ -111,8 +111,7 @@ def test__get_authorization_headers_with_service_account(self):

plugin._get_authorization_headers(context)

# self-signed JWT should not be created when default_host is not set
credentials._create_self_signed_jwt.assert_not_called()
credentials._create_self_signed_jwt.assert_called_once_with(None)

def test__get_authorization_headers_with_service_account_and_default_host(self):
credentials = mock.create_autospec(service_account.Credentials)
Expand Down
2 changes: 1 addition & 1 deletion tests/transport/test_requests.py
Expand Up @@ -378,7 +378,7 @@ def test_authorized_session_without_default_host(self):

authed_session = google.auth.transport.requests.AuthorizedSession(credentials)

authed_session.credentials._create_self_signed_jwt.assert_not_called()
authed_session.credentials._create_self_signed_jwt.assert_called_once_with(None)

def test_authorized_session_with_default_host(self):
default_host = "pubsub.googleapis.com"
Expand Down
2 changes: 1 addition & 1 deletion tests/transport/test_urllib3.py
Expand Up @@ -164,7 +164,7 @@ def test_urlopen_no_default_host(self):

authed_http = google.auth.transport.urllib3.AuthorizedHttp(credentials)

authed_http.credentials._create_self_signed_jwt.assert_not_called()
authed_http.credentials._create_self_signed_jwt.assert_called_once_with(None)

def test_urlopen_with_default_host(self):
default_host = "pubsub.googleapis.com"
Expand Down