# Copyright 2016 Google Inc. All Rights Reserved.
|
#
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
# you may not use this file except in compliance with the License.
|
# You may obtain a copy of the License at
|
#
|
# http://www.apache.org/licenses/LICENSE-2.0
|
#
|
# Unless required by applicable law or agreed to in writing, software
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
# See the License for the specific language governing permissions and
|
# limitations under the License.
|
|
"""Helpers for authentication using oauth2client or google-auth."""
|
|
import httplib2
|
|
try:
|
import google.auth
|
import google.auth.credentials
|
HAS_GOOGLE_AUTH = True
|
except ImportError: # pragma: NO COVER
|
HAS_GOOGLE_AUTH = False
|
|
try:
|
import google_auth_httplib2
|
except ImportError: # pragma: NO COVER
|
google_auth_httplib2 = None
|
|
try:
|
import oauth2client
|
import oauth2client.client
|
HAS_OAUTH2CLIENT = True
|
except ImportError: # pragma: NO COVER
|
HAS_OAUTH2CLIENT = False
|
|
|
def default_credentials():
|
"""Returns Application Default Credentials."""
|
if HAS_GOOGLE_AUTH:
|
credentials, _ = google.auth.default()
|
return credentials
|
elif HAS_OAUTH2CLIENT:
|
return oauth2client.client.GoogleCredentials.get_application_default()
|
else:
|
raise EnvironmentError(
|
'No authentication library is available. Please install either '
|
'google-auth or oauth2client.')
|
|
|
def with_scopes(credentials, scopes):
|
"""Scopes the credentials if necessary.
|
|
Args:
|
credentials (Union[
|
google.auth.credentials.Credentials,
|
oauth2client.client.Credentials]): The credentials to scope.
|
scopes (Sequence[str]): The list of scopes.
|
|
Returns:
|
Union[google.auth.credentials.Credentials,
|
oauth2client.client.Credentials]: The scoped credentials.
|
"""
|
if HAS_GOOGLE_AUTH and isinstance(
|
credentials, google.auth.credentials.Credentials):
|
return google.auth.credentials.with_scopes_if_required(
|
credentials, scopes)
|
else:
|
try:
|
if credentials.create_scoped_required():
|
return credentials.create_scoped(scopes)
|
else:
|
return credentials
|
except AttributeError:
|
return credentials
|
|
|
def authorized_http(credentials):
|
"""Returns an http client that is authorized with the given credentials.
|
|
Args:
|
credentials (Union[
|
google.auth.credentials.Credentials,
|
oauth2client.client.Credentials]): The credentials to use.
|
|
Returns:
|
Union[httplib2.Http, google_auth_httplib2.AuthorizedHttp]: An
|
authorized http client.
|
"""
|
from googleapiclient.http import build_http
|
|
if HAS_GOOGLE_AUTH and isinstance(
|
credentials, google.auth.credentials.Credentials):
|
if google_auth_httplib2 is None:
|
raise ValueError(
|
'Credentials from google.auth specified, but '
|
'google-api-python-client is unable to use these credentials '
|
'unless google-auth-httplib2 is installed. Please install '
|
'google-auth-httplib2.')
|
return google_auth_httplib2.AuthorizedHttp(credentials,
|
http=build_http())
|
else:
|
return credentials.authorize(build_http())
|
|
|
def refresh_credentials(credentials):
|
# Refresh must use a new http instance, as the one associated with the
|
# credentials could be a AuthorizedHttp or an oauth2client-decorated
|
# Http instance which would cause a weird recursive loop of refreshing
|
# and likely tear a hole in spacetime.
|
refresh_http = httplib2.Http()
|
if HAS_GOOGLE_AUTH and isinstance(
|
credentials, google.auth.credentials.Credentials):
|
request = google_auth_httplib2.Request(refresh_http)
|
return credentials.refresh(request)
|
else:
|
return credentials.refresh(refresh_http)
|
|
|
def apply_credentials(credentials, headers):
|
# oauth2client and google-auth have the same interface for this.
|
if not is_valid(credentials):
|
refresh_credentials(credentials)
|
return credentials.apply(headers)
|
|
|
def is_valid(credentials):
|
if HAS_GOOGLE_AUTH and isinstance(
|
credentials, google.auth.credentials.Credentials):
|
return credentials.valid
|
else:
|
return (
|
credentials.access_token is not None and
|
not credentials.access_token_expired)
|
|
|
def get_credentials_from_http(http):
|
if http is None:
|
return None
|
elif hasattr(http.request, 'credentials'):
|
return http.request.credentials
|
elif (hasattr(http, 'credentials')
|
and not isinstance(http.credentials, httplib2.Credentials)):
|
return http.credentials
|
else:
|
return None
|