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: add to_json method to google.oauth2.credentials.Credentials #367

Merged
merged 3 commits into from
Dec 5, 2019
Merged

feat: add to_json method to google.oauth2.credentials.Credentials #367

merged 3 commits into from
Dec 5, 2019

Conversation

patkasper
Copy link
Contributor

This adds a Storage base interface class and a file module based on Storage to save and load credentials.
The credentials module was modified to rely on the newer Storage class while retaining compatibility.

@googlebot googlebot added the cla: yes This human has signed the Contributor License Agreement. label Aug 28, 2019
@busunkim96
Copy link
Contributor

Hi @patkasper - apologies for it taking a long time to comment on this.

I brought up credentials storage in a weekly meeting and it looks like folks are generally opposed to the idea of this library handling credentials storage. It may be convenient to store credentials in memory or in a file, but it isn't great from a security perspective.

There was interest in adding a method like to_json to make it easier to convert a credentials object back into JSON. Could you perhaps rework this PR to add just that method?

@patkasper
Copy link
Contributor Author

I've removed the storage interface and I've kept the to_json method in google.oauth2's Credentials. Please let me know what you think.

I'm happy to add the storage interface + automated loading/storing to google-auth-oauthlib, should I go ahead with that?

Thanks.

@@ -232,3 +241,34 @@ def unpadded_urlsafe_b64encode(value):
Union[str|bytes]: The encoded value
"""
return base64.urlsafe_b64encode(value).rstrip(b'=')


def validate_file(filename):
Copy link
Contributor

Choose a reason for hiding this comment

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

I don't think this function is used anymore?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Indeed, that was a remnant from my earlier additions. Removing.

Raises:
ValueError: If the file is not in the expected format.
"""
with io.open(filename, 'r', encoding='utf-8') as json_file:
data = json.load(json_file)
return cls.from_authorized_user_info(data, scopes)

def to_json(self, strip=None, indent=None):
Copy link
Contributor

Choose a reason for hiding this comment

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

I'd prefer to remove indent. If the user would like to do more formatting they can do it with the return value from this function.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Ok, no problem.

@@ -160,13 +165,12 @@ def from_authorized_user_info(cls, info, scopes=None):

Args:
info (Mapping[str, str]): The authorized user info in Google
format.
format.
Copy link
Contributor

Choose a reason for hiding this comment

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

Did you mean to change the indent here?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Funny, there are a few occurrences where the indent is not what I had locally. Not sure what happened.

Copy link
Contributor

Choose a reason for hiding this comment

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

Would you mind reverting these whitespace changes?

warnings.warn(_MISSING_FILE_MESSAGE.format(filename))


def parse_expiry(expiry):
Copy link
Contributor

Choose a reason for hiding this comment

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

Where is this function used?

google/oauth2/credentials.py Outdated Show resolved Hide resolved
google/oauth2/credentials.py Outdated Show resolved Hide resolved
google/oauth2/credentials.py Outdated Show resolved Hide resolved
Copy link
Contributor Author

@patkasper patkasper left a comment

Choose a reason for hiding this comment

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

Changes coming!

@@ -232,3 +241,34 @@ def unpadded_urlsafe_b64encode(value):
Union[str|bytes]: The encoded value
"""
return base64.urlsafe_b64encode(value).rstrip(b'=')


def validate_file(filename):
Copy link
Contributor Author

Choose a reason for hiding this comment

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

Indeed, that was a remnant from my earlier additions. Removing.

docs/reference/google.oauth2.rst Outdated Show resolved Hide resolved
docs/reference/google.oauth2.file.rst Outdated Show resolved Hide resolved
google/oauth2/credentials.py Outdated Show resolved Hide resolved
@@ -160,13 +165,12 @@ def from_authorized_user_info(cls, info, scopes=None):

Args:
info (Mapping[str, str]): The authorized user info in Google
format.
format.
Copy link
Contributor Author

Choose a reason for hiding this comment

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

Funny, there are a few occurrences where the indent is not what I had locally. Not sure what happened.

google/oauth2/credentials.py Outdated Show resolved Hide resolved
Raises:
ValueError: If the file is not in the expected format.
"""
with io.open(filename, 'r', encoding='utf-8') as json_file:
data = json.load(json_file)
return cls.from_authorized_user_info(data, scopes)

def to_json(self, strip=None, indent=None):
Copy link
Contributor Author

Choose a reason for hiding this comment

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

Ok, no problem.

google/oauth2/credentials.py Outdated Show resolved Hide resolved
@@ -160,13 +165,12 @@ def from_authorized_user_info(cls, info, scopes=None):

Args:
info (Mapping[str, str]): The authorized user info in Google
format.
format.
Copy link
Contributor

Choose a reason for hiding this comment

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

Would you mind reverting these whitespace changes?

Args:
strip (Sequence[str]): Optional list of members to exclude from the
generated JSON.
indent (int): This parameter is passed to `json.dumps` to create
Copy link
Contributor

Choose a reason for hiding this comment

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

Please remove the documentation for this arg

Raises:
ValueError: If the file is not in the expected format.
"""
with io.open(filename, 'r', encoding='utf-8') as json_file:
data = json.load(json_file)
return cls.from_authorized_user_info(data, scopes)

def to_json(self, strip=None):
"""Utility function that creates JSON repr. of a Credentials object.
Copy link
Contributor

Choose a reason for hiding this comment

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

Suggested change
"""Utility function that creates JSON repr. of a Credentials object.
"""Utility function that creates JSON representation of a Credentials object.

@patkasper patkasper changed the title credential storage support feat: add to_json method to google.oauth2.credentials.Credentials Oct 8, 2019
@busunkim96 busunkim96 marked this pull request as ready for review October 8, 2019 17:53
@patkasper patkasper requested a review from busunkim96 October 11, 2019 09:58
Copy link
Contributor

@busunkim96 busunkim96 left a comment

Choose a reason for hiding this comment

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

@patkasper This looks good! I am doing some cleanup of the CI in another PR, but I will merge this once that is resolved.

Thank you for all the work you've put into this!

@busunkim96
Copy link
Contributor

👋 Hi @patkasper. Would you mind resolving the merge conflicts here?

@busunkim96 busunkim96 added the kokoro:force-run Add this label to force Kokoro to re-run the tests. label Nov 21, 2019
@yoshi-kokoro yoshi-kokoro removed the kokoro:force-run Add this label to force Kokoro to re-run the tests. label Nov 21, 2019
@busunkim96 busunkim96 added the kokoro:force-run Add this label to force Kokoro to re-run the tests. label Nov 22, 2019
@yoshi-kokoro yoshi-kokoro removed the kokoro:force-run Add this label to force Kokoro to re-run the tests. label Nov 22, 2019
@busunkim96
Copy link
Contributor

@patkasper Would you be able to run the code formatter?

pip install nox
nox -s blacken

If nox gives you trouble for any reason you can also do:

pip install black
black google tests

@michaelawyu If you're around next week could you help see this PR over the finish line? Since Patrick is an external contributor the CI does not trigger automatically. You'll have to add the label kokoro:force-run. Thank you!

@busunkim96 busunkim96 added the kokoro:force-run Add this label to force Kokoro to re-run the tests. label Dec 5, 2019
@yoshi-kokoro yoshi-kokoro removed the kokoro:force-run Add this label to force Kokoro to re-run the tests. label Dec 5, 2019
@busunkim96 busunkim96 merged commit bfb1f8c into googleapis:master Dec 5, 2019
@nicain
Copy link
Contributor

nicain commented Sep 4, 2021

@patkasper @busunkim96 I noticed this evening that I cannot round-trip a credential using "to_json" to serialize the object, and then reload.

I hit this when deserializing my credential and attempting to use it with a Dialogflow client; in that case, I received:

google.api_core.exceptions.PermissionDenied: 403 Your application has authenticated using end user credentials from the Google Cloud SDK or Google Cloud Shell which are not supported by the dialogflow.googleapis.com. We recommend configuring the billing/quota_project setting in gcloud or using a service account through the auth/impersonate_service_account setting. For more information about service accounts and how to use them in your application, see https://cloud.google.com/docs/authentication/. If you are getting this error with curl or similar tools, you may need to specify 'X-Goog-User-Project' HTTP header for quota and billing purposes. For more information regarding 'X-Goog-User-Project' header, please check https://cloud.google.com/apis/docs/system-parameters.

The hint here was the "billing quota"; when I compared the dict representations of the original and serialized-then-serialized clone of the original object, I saw that the "_quota_project_id" attribute was missing on the clone. Sure enough, after I added this in manually, my credential works.

Was the "quota_project_id" attribute intentionally excluded from the contents of the json blob, on serialization? If not, I am happy to work up a PR to include that attribute in the serialization (it is a kwarg of the initializaer of the Credentials class, so very easy to round-trip).


One other minor note; the serialized-then-serialized clone of the original credential cannot itself be serialized with to_json if the expiry kwarg is naively set to the expiry string from the serialization dict. This happened to me when I tried to naively deserialize by just "**" unpackinging the dictionary contents into the kwargs of the initializer. Then this line fails because self.expiry is a string, not a datetime. This should be pretty easy to fix with a isinstance or try-catch, and I am happy to provide a PR here as well.

What are your thoughts on these issues?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
cla: yes This human has signed the Contributor License Agreement.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants