Skip to content

Commit

Permalink
Merge pull request #43 from kyleknap/waiters
Browse files Browse the repository at this point in the history
Implement Resource Waiter Actions
  • Loading branch information
kyleknap committed Dec 17, 2014
2 parents 8b10f29 + 8124dde commit 279d058
Show file tree
Hide file tree
Showing 8 changed files with 309 additions and 20 deletions.
106 changes: 95 additions & 11 deletions boto3/docs.py
Original file line number Diff line number Diff line change
Expand Up @@ -146,9 +146,11 @@ def docs_for(service_name):

docs = '{0}\n{1}\n\n'.format(official_name, '=' * len(official_name))

docs += '.. contents:: Table of Contents\n\n'
docs += '.. contents:: Table of Contents\n :depth: 2\n\n'

docs += document_client(service_name, official_name, service_model)
docs += document_client_waiter(session, official_name, service_name,
service_model)

filename = (os.path.dirname(__file__) + '/data/resources/'
'{0}-{1}.resources.json').format(service_name,
Expand All @@ -167,7 +169,7 @@ def docs_for(service_name):
}

docs += document_resource(service_name, official_name, model,
service_model)
service_model, session)

# First, collect all the models...
for name, model in sorted(data['resources'].items(),
Expand All @@ -189,7 +191,7 @@ def docs_for(service_name):
model = item['model']
if item['type'] == 'resource':
docs += document_resource(service_name, official_name,
model, service_model)
model, service_model, session)
elif item['type'] == 'collection':
docs += document_collection(
service_name, official_name, model,
Expand Down Expand Up @@ -223,7 +225,7 @@ def document_client(service_name, official_name, service_model):
wdoc += ' .. py:method:: get_waiter(name)\n\n'
wdoc += ' Get a waiter by name. Available waiters:\n\n'
for waiter in client.waiter_names:
wdoc += ' * {0}\n'.format(waiter)
wdoc += ' * `{0}`_\n'.format(waiter)
wdoc += '\n'

waiter_included = False
Expand All @@ -240,8 +242,40 @@ def document_client(service_name, official_name, service_model):

return docs

def document_client_waiter(session, official_name, service_name,
service_model):
client = boto3.client(service_name, aws_access_key_id='dummy',
aws_secret_access_key='dummy',
region_name='us-east-1')
waiter_spec_doc = ''
if client.waiter_names:
waiter_spec_doc = 'Waiter\n------\n\n'
service_waiter_model = session.get_waiter_model(service_name)
for waiter in service_waiter_model.waiter_names:
snake_cased = xform_name(waiter)
waiter_spec_doc += '{0}\n{1}\n\n'.format(snake_cased,
'~' * len(snake_cased))
waiter_model = service_waiter_model.get_waiter(waiter)
operation_model = service_model.operation_model(
waiter_model.operation)
description = (
' This polls :py:meth:`{0}.Client.{1}` every {2} '
'seconds until a successful state is reached. An error is '
'returned after {3} failed checks.'.format(
service_name, xform_name(waiter_model.operation),
waiter_model.delay, waiter_model.max_attempts)
)
waiter_spec_doc += document_operation(
operation_model=operation_model, service_name=service_name,
operation_name='wait', rtype=None, description=description,
example_instance='client.get_waiter(\'{0}\')'.format(
snake_cased))
waiter_spec_doc += '\n'

return waiter_spec_doc

def document_resource(service_name, official_name, resource_model,
service_model):
service_model, session):
"""
Generate reference documentation from a resource model.
"""
Expand Down Expand Up @@ -369,6 +403,16 @@ def document_resource(service_name, official_name, resource_model,
collection.resource.type, service_name,
xform_name(collection.request.operation))

if resource_model.waiters:
docs += (' .. rst-class:: admonition-title\n\n Waiters\n\n'
' Waiters provide an interface to wait for a resource'
' to reach a specific state.\n\n')
service_waiter_model = session.get_waiter_model(service_name)
for waiter in sorted(resource_model.waiters,
key=lambda i: i.resource_waiter_name):
docs += document_waiter(waiter, service_name, resource_model,
service_model, service_waiter_model)

return docs

def document_collection(service_name, official_name, collection_model,
Expand Down Expand Up @@ -407,6 +451,42 @@ def document_collection(service_name, official_name, collection_model,

return docs

def document_waiter(waiter, service_name, resource_model, service_model,
service_waiter_model):
"""
Document a resource waiter, including the low-level client waiter
and parameters.
"""
try:
waiter_model = service_waiter_model.get_waiter(waiter.waiter_name)
except:
print('Cannot get waiter ' + waiter.waiter_name)
return ''

try:
operation_model = service_model.operation_model(waiter_model.operation)
except:
print('Cannot get operation ' + action.request.operation +
' for waiter ' + waiter.waiter_name)
return ''
description = (' Waits until this {0} is {1}.\n'
' This method calls ``wait()`` on'
' :py:meth:`{2}.Client.get_waiter` using `{3}`_ .').format(
resource_model.name,
xform_name(waiter.name).replace('_', ' '),
service_name,
xform_name(waiter.waiter_name))

# Here we split because we only care about top-level parameter names
ignore_params = [p.target.split('.')[0].strip('[]') for p in waiter.params]

return document_operation(
operation_model=operation_model, service_name=service_name,
operation_name=xform_name(waiter.resource_waiter_name),
description=description,
example_instance = xform_name(resource_model.name),
ignore_params=ignore_params, rtype=None)

def document_action(action, service_name, resource_model, service_model,
action_type='action'):
"""
Expand All @@ -421,7 +501,8 @@ def document_action(action, service_name, resource_model, service_model,
return ''

# Here we split because we only care about top-level parameter names
ignore_params = [p.target.split('.')[0] for p in action.request.params]
ignore_params = [p.target.split('.')[0].strip('[]')
for p in action.request.params]

rtype = 'dict'
if action_type == 'action':
Expand All @@ -442,7 +523,8 @@ def document_action(action, service_name, resource_model, service_model,
return document_operation(
operation_model, service_name, operation_name=xform_name(action.name),
description=description, ignore_params=ignore_params, rtype=rtype,
example_instance=service_name, example_response=example_response)
example_instance=xform_name(resource_model.name),
example_response=example_response)

def document_operation(operation_model, service_name, operation_name=None,
description=None, ignore_params=None, rtype='dict',
Expand Down Expand Up @@ -497,8 +579,10 @@ def document_operation(operation_model, service_name, operation_name=None,
default = py_default(value.type_name)
dummy_params.append('{0}={1}'.format(
key, default))
docs += ' Example::\n\n {0} = {1}.{2}({3})\n\n'.format(
example_response, example_instance, operation_name,
docs += ' Example::\n\n '
if example_response is not None:
docs += '{0} = '.format(example_response)
docs += '{0}.{1}({2})\n\n'.format(example_instance, operation_name,
', '.join(dummy_params))

for key, value in params.items():
Expand All @@ -510,7 +594,7 @@ def document_operation(operation_model, service_name, operation_name=None,
docs += (' :param {0} {1}: *{2}* - {3}\n'.format(
param_type, key, required,
html_to_rst(value.documentation, indent=9)))

docs += '\n\n :rtype: {0}\n\n'.format(rtype)
if rtype is not None:
docs += '\n\n :rtype: {0}\n\n'.format(rtype)

return docs
45 changes: 45 additions & 0 deletions boto3/resources/action.py
Original file line number Diff line number Diff line change
Expand Up @@ -149,3 +149,48 @@ def __call__(self, parent, *args, **kwargs):
self._response_handler(parent, params, response))

return responses


class WaiterAction(object):
"""
A class representing a callable waiter action on a resource, for example
``s3.Bucket('foo').wait_until_bucket_exists()``.
The waiter action may construct parameters from existing resource
identifiers.
:type waiter_model: :py:class`~boto3.resources.model.Waiter`
:param waiter_model: The action waiter.
:type waiter_resource_name: string
:param waiter_resource_name: The name of the waiter action for the
resource. It usually begins with a
``wait_until_``
"""
def __init__(self, waiter_model, waiter_resource_name):
self._waiter_model = waiter_model
self._waiter_resource_name = waiter_resource_name

def __call__(self, parent, *args, **kwargs):
"""
Perform the wait operation after building operation
parameters.
:type parent: :py:class:`~boto3.resources.base.ServiceResource`
:param parent: The resource instance to which this action is attached.
"""
client_waiter_name = xform_name(self._waiter_model.waiter_name)

# First, build predefined params and then update with the
# user-supplied kwargs, which allows overriding the pre-built
# params if needed.
params = create_request_parameters(parent, self._waiter_model)
params.update(kwargs)

logger.info('Calling %s:%s with %r',
parent.meta['service_name'],
self._waiter_resource_name, params)

client = parent.meta['client']
waiter = client.get_waiter(client_waiter_name)
response = waiter.wait(**params)

logger.debug('Response: %r', response)
27 changes: 27 additions & 0 deletions boto3/resources/factory.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@
from botocore import xform_name

from .action import ServiceAction
from .action import WaiterAction
from .base import ServiceResource
from .collection import CollectionFactory
from .model import ResourceModel
Expand Down Expand Up @@ -87,6 +88,7 @@ def load_from_definition(self, service_name, resource_name, model,
service_model)
self._load_references(attrs, service_name, resource_name,
resource_model, resource_defs, service_model)
self._load_waiters(attrs, resource_model)

# Create the name based on the requested service and resource
cls_name = resource_name
Expand Down Expand Up @@ -214,6 +216,18 @@ def _load_references(self, attrs, service_name, resource_name,
reference.resource.type, snake_cased, reference, service_name,
resource_name, model, resource_defs, service_model)

def _load_waiters(self, attrs, model):
"""
Load resource waiters from the model. Each waiter allows you to
wait until a resource reaches a specific state by polling the state
of the resource.
"""
for waiter in model.waiters:
snake_cased = xform_name(waiter.resource_waiter_name)
snake_cased = self._check_allowed_name(
attrs, snake_cased, 'waiter', model.name)
attrs[snake_cased] = self._create_waiter(waiter, snake_cased)

def _check_allowed_name(self, attrs, name, category, resource_name):
"""
Determine if a given name is allowed on the instance, and if not,
Expand Down Expand Up @@ -271,6 +285,19 @@ def property_loader(self):
property_loader.__doc__ = 'TODO'
return property(property_loader)

def _create_waiter(factory_self, waiter_model, snake_cased):
"""
Creates a new wait method for each resource where both a waiter and
resource model is defined.
"""
waiter = WaiterAction(waiter_model, waiter_resource_name=snake_cased)
def do_waiter(self, *args, **kwargs):
waiter(self, *args, **kwargs)

do_waiter.__name__ = str(snake_cased)
do_waiter.__doc__ = 'TODO'
return do_waiter

def _create_collection(factory_self, service_name, resource_name,
snake_cased, collection_model,
resource_defs, service_model):
Expand Down
3 changes: 3 additions & 0 deletions boto3/resources/model.py
Original file line number Diff line number Diff line change
Expand Up @@ -150,6 +150,9 @@ def __init__(self, name, definition):
#: (``string``) The name of this waiter
self.name = name

#: (``string``) The name of the waiter in the resource
self.resource_waiter_name = 'WaitUntil' + name

#: (``string``) The name of the underlying event waiter
self.waiter_name = definition.get('waiterName')

Expand Down
45 changes: 38 additions & 7 deletions tests/integration/test_s3.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,22 +18,30 @@

class TestS3Resource(unittest.TestCase):
def setUp(self):
self.session = boto3.session.Session(region_name='us-west-2')
self.region = 'us-west-2'
self.session = boto3.session.Session(region_name=self.region)
self.s3 = self.session.resource('s3')
self.bucket_name = unique_id('boto3-test')

def create_bucket_resource(self, bucket_name, region=None):
if region is None:
region = self.region
kwargs = {'Bucket': bucket_name}
if region != 'us-east-1':
kwargs['CreateBucketConfiguration'] = {
'LocationConstraint': region
}
bucket = self.s3.create_bucket(**kwargs)
self.addCleanup(bucket.delete)
return bucket

def test_s3(self):
client = self.s3.meta['client']

# Create a bucket (resource action with a resource response)
bucket = self.s3.create_bucket(
Bucket=self.bucket_name,
CreateBucketConfiguration={
'LocationConstraint': 'us-west-2'
})
bucket = self.create_bucket_resource(self.bucket_name)
waiter = client.get_waiter('bucket_exists')
waiter.wait(Bucket=self.bucket_name)
self.addCleanup(bucket.delete)

# Create an object
obj = bucket.Object('test.txt')
Expand All @@ -52,3 +60,26 @@ def test_s3(self):
# Perform a resource action with a low-level response
self.assertEqual(b'hello, world',
obj.get()['Body'].read())

def test_s3_resource_waiter(self):
# Create a bucket
bucket = self.create_bucket_resource(self.bucket_name)
# Wait till the bucket exists
bucket.wait_until_exists()
# Confirm the bucket exists by finding it in a list of all of our
# buckets
self.assertIn(self.bucket_name,
[b.name for b in self.s3.buckets.all()])


# Create an object
obj = bucket.Object('test.txt')
obj.put(
Body='hello, world')
self.addCleanup(obj.delete)

# Wait till the bucket exists
obj.wait_until_exists()

# List objects and make sure ours is present
self.assertIn('test.txt', [o.key for o in bucket.objects.all()])
Loading

0 comments on commit 279d058

Please sign in to comment.