Skip to content

Commit

Permalink
Merge pull request #572 from dhermes/add-gae-gce-support-for-implicit-id
Browse files Browse the repository at this point in the history
Adding implicit dataset ID support for GAE and GCE.
  • Loading branch information
dhermes committed Jan 27, 2015
2 parents 6bbb144 + c3286a2 commit 50f0684
Show file tree
Hide file tree
Showing 3 changed files with 204 additions and 5 deletions.
16 changes: 11 additions & 5 deletions gcloud/datastore/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -72,18 +72,24 @@
def set_default_dataset_id(dataset_id=None):
"""Set default dataset ID either explicitly or implicitly as fall-back.
In implicit case, currently only supports enviroment variable but will
support App Engine, Compute Engine and other environments in the future.
Local environment variable used is:
- GCLOUD_DATASET_ID
In implicit case, supports three cases. In order of precedence, the
implicit cases are:
- GCLOUD_DATASET_ID environment variable
- Google App Engine application ID
- Google Compute Engine project ID (from metadata server)
:type dataset_id: string
:param dataset_id: Optional. The dataset ID to use as default.
"""
if dataset_id is None:
dataset_id = os.getenv(_DATASET_ENV_VAR_NAME)

if dataset_id is None:
dataset_id = _implicit_environ.app_engine_id()

if dataset_id is None:
dataset_id = _implicit_environ.compute_engine_id()

if dataset_id is not None:
_implicit_environ.DATASET_ID = dataset_id

Expand Down
54 changes: 54 additions & 0 deletions gcloud/datastore/_implicit_environ.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,9 +18,63 @@
imply the current dataset ID and connection from the enviroment.
"""

import httplib2
import socket

try:
from google.appengine.api import app_identity
except ImportError:
app_identity = None


DATASET_ID = None
"""Module global to allow persistent implied dataset ID from enviroment."""

CONNECTION = None
"""Module global to allow persistent implied connection from enviroment."""


def app_engine_id():
"""Gets the App Engine application ID if it can be inferred.
:rtype: string or ``NoneType``
:returns: App Engine application ID if running in App Engine,
else ``None``.
"""
if app_identity is None:
return None

return app_identity.get_application_id()


def compute_engine_id():
"""Gets the Compute Engine project ID if it can be inferred.
Uses 169.254.169.254 for the metadata server to avoid request
latency from DNS lookup.
See https://cloud.google.com/compute/docs/metadata#metadataserver
for information about this IP address. (This IP is also used for
Amazon EC2 instances, so the metadata flavor is crucial.)
See https://github.com/google/oauth2client/issues/93 for context about
DNS latency.
:rtype: string or ``NoneType``
:returns: Compute Engine project ID if the metadata service is available,
else ``None``.
"""
http = httplib2.Http(timeout=0.1)
uri = 'http://169.254.169.254/computeMetadata/v1/project/project-id'
headers = {'Metadata-Flavor': 'Google'}

response = content = None
try:
response, content = http.request(uri, method='GET', headers=headers)
except socket.timeout:
pass

if response is None or response['status'] != '200':
return None

return content
139 changes: 139 additions & 0 deletions gcloud/datastore/test___init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -78,6 +78,104 @@ def test_set_explicit_None_w_env_var_set(self):
self._callFUT(None)
self.assertEqual(_implicit_environ.DATASET_ID, IMPLICIT_DATASET_ID)

def test_set_implicit_from_appengine(self):
from gcloud._testing import _Monkey
from gcloud.datastore import _implicit_environ

APP_ENGINE_ID = 'GAE'
APP_IDENTITY = _AppIdentity(APP_ENGINE_ID)

with self._monkey(None):
with _Monkey(_implicit_environ, app_identity=APP_IDENTITY):
self._callFUT()

self.assertEqual(_implicit_environ.DATASET_ID, APP_ENGINE_ID)

def test_set_implicit_both_env_and_appengine(self):
from gcloud._testing import _Monkey
from gcloud.datastore import _implicit_environ

IMPLICIT_DATASET_ID = 'IMPLICIT'
APP_IDENTITY = _AppIdentity('GAE')

with self._monkey(IMPLICIT_DATASET_ID):
with _Monkey(_implicit_environ, app_identity=APP_IDENTITY):
self._callFUT()

self.assertEqual(_implicit_environ.DATASET_ID, IMPLICIT_DATASET_ID)

def _implicit_compute_engine_helper(self, status):
from gcloud._testing import _Monkey
from gcloud.datastore import _implicit_environ

COMPUTE_ENGINE_ID = 'GCE'
HTTPLIB2 = _Httplib2(COMPUTE_ENGINE_ID, status=status)
if status == '200':
EXPECTED_ID = COMPUTE_ENGINE_ID
else:
EXPECTED_ID = None

with self._monkey(None):
with _Monkey(_implicit_environ, httplib2=HTTPLIB2):
self._callFUT()

self.assertEqual(_implicit_environ.DATASET_ID, EXPECTED_ID)
self.assertEqual(len(HTTPLIB2._http_instances), 1)

(timeout, http_instance), = HTTPLIB2._http_instances
self.assertEqual(timeout, 0.1)
self.assertEqual(
http_instance._called_uris,
['http://169.254.169.254/computeMetadata/v1/project/project-id'])
expected_kwargs = {
'method': 'GET',
'headers': {
'Metadata-Flavor': 'Google',
},
}
self.assertEqual(http_instance._called_kwargs, [expected_kwargs])

def test_set_implicit_from_compute_engine(self):
self._implicit_compute_engine_helper('200')

def test_set_implicit_from_compute_engine_bad_status(self):
self._implicit_compute_engine_helper('404')

def test_set_implicit_from_compute_engine_raise_timeout(self):
self._implicit_compute_engine_helper('RAISE')

def test_set_implicit_both_appengine_and_compute(self):
from gcloud._testing import _Monkey
from gcloud.datastore import _implicit_environ

APP_ENGINE_ID = 'GAE'
APP_IDENTITY = _AppIdentity(APP_ENGINE_ID)
HTTPLIB2 = _Httplib2('GCE')

with self._monkey(None):
with _Monkey(_implicit_environ, app_identity=APP_IDENTITY,
httplib2=HTTPLIB2):
self._callFUT()

self.assertEqual(_implicit_environ.DATASET_ID, APP_ENGINE_ID)
self.assertEqual(len(HTTPLIB2._http_instances), 0)

def test_set_implicit_three_env_appengine_and_compute(self):
from gcloud._testing import _Monkey
from gcloud.datastore import _implicit_environ

IMPLICIT_DATASET_ID = 'IMPLICIT'
APP_IDENTITY = _AppIdentity('GAE')
HTTPLIB2 = _Httplib2('GCE')

with self._monkey(IMPLICIT_DATASET_ID):
with _Monkey(_implicit_environ, app_identity=APP_IDENTITY,
httplib2=HTTPLIB2):
self._callFUT()

self.assertEqual(_implicit_environ.DATASET_ID, IMPLICIT_DATASET_ID)
self.assertEqual(len(HTTPLIB2._http_instances), 0)


class Test_set_default_connection(unittest2.TestCase):

Expand Down Expand Up @@ -165,3 +263,44 @@ def test_it(self):
self.assertTrue(isinstance(found, Connection))
self.assertTrue(found._credentials is client._signed)
self.assertTrue(client._get_app_default_called)


class _AppIdentity(object):

def __init__(self, app_id):
self.app_id = app_id

def get_application_id(self):
return self.app_id


class _Http(object):

def __init__(self, parent):
self.parent = parent
self._called_uris = []
self._called_kwargs = []

def request(self, uri, **kwargs):
import socket

self._called_uris.append(uri)
self._called_kwargs.append(kwargs)

if self.parent.status == 'RAISE':
raise socket.timeout('timed out')
else:
return {'status': self.parent.status}, self.parent.project_id


class _Httplib2(object):

def __init__(self, project_id, status='200'):
self.project_id = project_id
self.status = status
self._http_instances = []

def Http(self, timeout=None):
result = _Http(self)
self._http_instances.append((timeout, result))
return result

0 comments on commit 50f0684

Please sign in to comment.