# Copyright 2015 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. """Base classes for client used to interact with Google Cloud APIs.""" from oauth2client.service_account import ServiceAccountCredentials import six from gcloud._helpers import _determine_default_project from gcloud.connection import Connection from gcloud.credentials import get_credentials class _ClientFactoryMixin(object): """Mixin to allow factories that create credentials. .. note:: This class is virtual. """ @classmethod def from_service_account_json(cls, json_credentials_path, *args, **kwargs): """Factory to retrieve JSON credentials while creating client. :type json_credentials_path: string :param json_credentials_path: The path to a private key file (this file was given to you when you created the service account). This file must contain a JSON object with a private key and other credentials information (downloaded from the Google APIs console). :type args: tuple :param args: Remaining positional arguments to pass to constructor. :type kwargs: dict :param kwargs: Remaining keyword arguments to pass to constructor. :rtype: :class:`gcloud.pubsub.client.Client` :returns: The client created with the retrieved JSON credentials. :raises: :class:`TypeError` if there is a conflict with the kwargs and the credentials created by the factory. """ if 'credentials' in kwargs: raise TypeError('credentials must not be in keyword arguments') credentials = ServiceAccountCredentials.from_json_keyfile_name( json_credentials_path) kwargs['credentials'] = credentials return cls(*args, **kwargs) @classmethod def from_service_account_p12(cls, client_email, private_key_path, *args, **kwargs): """Factory to retrieve P12 credentials while creating client. .. note:: Unless you have an explicit reason to use a PKCS12 key for your service account, we recommend using a JSON key. :type client_email: string :param client_email: The e-mail attached to the service account. :type private_key_path: string :param private_key_path: The path to a private key file (this file was given to you when you created the service account). This file must be in P12 format. :type args: tuple :param args: Remaining positional arguments to pass to constructor. :type kwargs: dict :param kwargs: Remaining keyword arguments to pass to constructor. :rtype: :class:`gcloud.client.Client` :returns: The client created with the retrieved P12 credentials. :raises: :class:`TypeError` if there is a conflict with the kwargs and the credentials created by the factory. """ if 'credentials' in kwargs: raise TypeError('credentials must not be in keyword arguments') credentials = ServiceAccountCredentials.from_p12_keyfile( client_email, private_key_path) kwargs['credentials'] = credentials return cls(*args, **kwargs) class Client(_ClientFactoryMixin): """Client to bundle configuration needed for API requests. Assumes that the associated ``_connection_class`` only accepts ``http`` and ``credentials`` in its constructor. :type credentials: :class:`oauth2client.client.OAuth2Credentials` or :class:`NoneType` :param credentials: The OAuth2 Credentials to use for the connection owned by this client. If not passed (and if no ``http`` object is passed), falls back to the default inferred from the environment. :type http: :class:`httplib2.Http` or class that defines ``request()``. :param http: An optional HTTP object to make requests. If not passed, an ``http`` object is created that is bound to the ``credentials`` for the current object. """ _connection_class = Connection def __init__(self, credentials=None, http=None): if credentials is None and http is None: credentials = get_credentials() self.connection = self._connection_class( credentials=credentials, http=http) class _ClientProjectMixin(object): """Mixin to allow setting the project on the client. :type project: string :param project: the project which the client acts on behalf of. If not passed falls back to the default inferred from the environment. :raises: :class:`EnvironmentError` if the project is neither passed in nor set in the environment. :class:`ValueError` if the project value is invalid. """ def __init__(self, project=None): project = self._determine_default(project) if project is None: raise EnvironmentError('Project was not passed and could not be ' 'determined from the environment.') if isinstance(project, six.binary_type): project = project.decode('utf-8') if not isinstance(project, six.string_types): raise ValueError('Project must be a string.') self.project = project @staticmethod def _determine_default(project): """Helper: use default project detection.""" return _determine_default_project(project) class JSONClient(Client, _ClientProjectMixin): """Client to for Google JSON-based API. Assumes such APIs use the ``project`` and the client needs to store this value. :type project: string :param project: the project which the client acts on behalf of. If not passed falls back to the default inferred from the environment. :type credentials: :class:`oauth2client.client.OAuth2Credentials` or :class:`NoneType` :param credentials: The OAuth2 Credentials to use for the connection owned by this client. If not passed (and if no ``http`` object is passed), falls back to the default inferred from the environment. :type http: :class:`httplib2.Http` or class that defines ``request()``. :param http: An optional HTTP object to make requests. If not passed, an ``http`` object is created that is bound to the ``credentials`` for the current object. :raises: :class:`ValueError` if the project is neither passed in nor set in the environment. """ def __init__(self, project=None, credentials=None, http=None): _ClientProjectMixin.__init__(self, project=project) Client.__init__(self, credentials=credentials, http=http)