Add notifications for sending an image

An image.send notification is to be sent to the notifier every time an image is
transmitted from glance. This can be used to track things such as bandwidth
usage.

Addresses bug 914440

Change-Id: If8b6504c4250fa6444d17d611de43d9704ca9aae
This commit is contained in:
Alex Meade 2012-01-06 18:26:38 +00:00
parent 7f4b1c5899
commit e2f9d15e4c
5 changed files with 201 additions and 6 deletions

View File

@ -17,7 +17,7 @@
Notifications Notifications
============= =============
Notifications can be generated for each upload, update or delete image Notifications can be generated for each send, upload, update or delete image
event. These can be used for auditing, troubleshooting, etc. event. These can be used for auditing, troubleshooting, etc.
Strategies Strategies
@ -70,16 +70,28 @@ Every message contains a handful of attributes.
Payload Payload
------- -------
WARN and ERROR events contain a text message in the payload. * image.send
The payload for INFO, WARN, and ERROR events contain the following::
image_id - ID of the image (UUID)
owner_id - Tenant or User ID that owns this image (string)
receiver_tenant_id - Tenant ID of the account receiving the image (string)
receiver_user_id - User ID of the account receiving the image (string)
destination_ip
bytes_sent - The number of bytes actually sent
* image.upload * image.upload
For INFO events, it is the image metadata. For INFO events, it is the image metadata.
WARN and ERROR events contain a text message in the payload.
* image.update * image.update
For INFO events, it is the image metadata. For INFO events, it is the image metadata.
WARN and ERROR events contain a text message in the payload.
* image.delete * image.delete
For INFO events, it is the image id. For INFO events, it is the image id.
WARN and ERROR events contain a text message in the payload.

View File

@ -46,7 +46,7 @@ class CacheFilter(wsgi.Middleware):
def __init__(self, app, conf, **local_conf): def __init__(self, app, conf, **local_conf):
self.conf = conf self.conf = conf
self.cache = image_cache.ImageCache(conf) self.cache = image_cache.ImageCache(conf)
self.serializer = images.ImageSerializer() self.serializer = images.ImageSerializer(conf)
logger.info(_("Initialized image cache middleware")) logger.info(_("Initialized image cache middleware"))
super(CacheFilter, self).__init__(app) super(CacheFilter, self).__init__(app)
@ -80,7 +80,7 @@ class CacheFilter(wsgi.Middleware):
try: try:
image_meta = registry.get_image_metadata(context, image_id) image_meta = registry.get_image_metadata(context, image_id)
response = webob.Response() response = webob.Response(request=request)
return self.serializer.show(response, { return self.serializer.show(response, {
'image_iterator': image_iterator, 'image_iterator': image_iterator,
'image_meta': image_meta}) 'image_meta': image_meta})

View File

@ -631,6 +631,10 @@ class ImageDeserializer(wsgi.JSONRequestDeserializer):
class ImageSerializer(wsgi.JSONResponseSerializer): class ImageSerializer(wsgi.JSONResponseSerializer):
"""Handles serialization of specific controller method responses.""" """Handles serialization of specific controller method responses."""
def __init__(self, conf):
self.conf = conf
self.notifier = notifier.Notifier(conf)
def _inject_location_header(self, response, image_meta): def _inject_location_header(self, response, image_meta):
location = self._get_image_location(image_meta) location = self._get_image_location(image_meta)
response.headers['Location'] = location response.headers['Location'] = location
@ -666,6 +670,28 @@ class ImageSerializer(wsgi.JSONResponseSerializer):
self._inject_checksum_header(response, image_meta) self._inject_checksum_header(response, image_meta)
return response return response
def image_send_notification(self, bytes_written, expected_size,
image_meta, request):
"""Send an image.send message to the notifier."""
try:
context = request.context
payload = {
'bytes_sent': bytes_written,
'image_id': image_meta['id'],
'owner_id': image_meta['owner'],
'receiver_tenant_id': context.tenant,
'receiver_user_id': context.user,
'destination_ip': request.remote_addr,
}
if bytes_written != expected_size:
self.notifier.error('image.send', payload)
else:
self.notifier.info('image.send', payload)
except Exception, err:
msg = _("An error occurred during image.send"
" notification: %(err)s") % locals()
logger.error(msg)
def show(self, response, result): def show(self, response, result):
image_meta = result['image_meta'] image_meta = result['image_meta']
image_id = image_meta['id'] image_id = image_meta['id']
@ -678,6 +704,16 @@ class ImageSerializer(wsgi.JSONResponseSerializer):
# size of the image file. See LP Bug #882585. # size of the image file. See LP Bug #882585.
def checked_iter(image_id, expected_size, image_iter): def checked_iter(image_id, expected_size, image_iter):
bytes_written = 0 bytes_written = 0
def notify_image_sent_hook(env):
self.image_send_notification(bytes_written, expected_size,
image_meta, response.request)
# Add hook to process after response is fully sent
if 'eventlet.posthooks' in response.environ:
response.environ['eventlet.posthooks'].append(
(notify_image_sent_hook, (), {}))
try: try:
for chunk in image_iter: for chunk in image_iter:
yield chunk yield chunk
@ -731,5 +767,5 @@ class ImageSerializer(wsgi.JSONResponseSerializer):
def create_resource(conf): def create_resource(conf):
"""Images resource factory method""" """Images resource factory method"""
deserializer = ImageDeserializer() deserializer = ImageDeserializer()
serializer = ImageSerializer() serializer = ImageSerializer(conf)
return wsgi.Resource(Controller(conf), deserializer, serializer) return wsgi.Resource(Controller(conf), deserializer, serializer)

View File

@ -391,7 +391,7 @@ class Resource(object):
action_result = self.dispatch(self.controller, action, action_result = self.dispatch(self.controller, action,
request, **action_args) request, **action_args)
try: try:
response = webob.Response() response = webob.Response(request=request)
self.dispatch(self.serializer, action, response, action_result) self.dispatch(self.serializer, action, response, action_result)
return response return response

View File

@ -25,6 +25,7 @@ import unittest
import stubout import stubout
import webob import webob
from glance.api.v1 import images
from glance.api.v1 import router from glance.api.v1 import router
from glance.common import context from glance.common import context
from glance.common import utils from glance.common import utils
@ -2688,3 +2689,149 @@ class TestGlanceAPI(unittest.TestCase):
res = req.get_response(self.api) res = req.get_response(self.api)
self.assertEquals(res.status_int, webob.exc.HTTPUnauthorized.code) self.assertEquals(res.status_int, webob.exc.HTTPUnauthorized.code)
class TestImageSerializer(unittest.TestCase):
def setUp(self):
"""Establish a clean test environment"""
self.stubs = stubout.StubOutForTesting()
stubs.stub_out_registry_and_store_server(self.stubs)
stubs.stub_out_filesystem_backend()
conf = test_utils.TestConfigOpts(CONF)
self.receiving_user = 'fake_user'
self.receiving_tenant = 2
self.context = rcontext.RequestContext(is_admin=True,
user=self.receiving_user,
tenant=self.receiving_tenant)
self.serializer = images.ImageSerializer(conf)
def image_iter():
for x in ['chunk', '678911234', '56789']:
yield x
self.FIXTURE = {
'image_iterator': image_iter(),
'image_meta': {
'id': UUID2,
'name': 'fake image #2',
'status': 'active',
'disk_format': 'vhd',
'container_format': 'ovf',
'is_public': True,
'created_at': datetime.datetime.utcnow(),
'updated_at': datetime.datetime.utcnow(),
'deleted_at': None,
'deleted': False,
'checksum': None,
'size': 19,
'owner': _gen_uuid(),
'location': "file:///tmp/glance-tests/2",
'properties': {}}
}
def tearDown(self):
"""Clear the test environment"""
stubs.clean_out_fake_filesystem_backend()
self.stubs.UnsetAll()
def test_meta(self):
exp_headers = {'x-image-meta-id': UUID2,
'x-image-meta-location': 'file:///tmp/glance-tests/2',
'ETag': self.FIXTURE['image_meta']['checksum'],
'x-image-meta-name': 'fake image #2'}
req = webob.Request.blank("/images/%s" % UUID2)
req.method = 'HEAD'
req.remote_addr = "1.2.3.4"
req.context = self.context
response = webob.Response(request=req)
self.serializer.meta(response, self.FIXTURE)
for key, value in exp_headers.iteritems():
self.assertEquals(value, response.headers[key])
def test_show(self):
exp_headers = {'x-image-meta-id': UUID2,
'x-image-meta-location': 'file:///tmp/glance-tests/2',
'ETag': self.FIXTURE['image_meta']['checksum'],
'x-image-meta-name': 'fake image #2'}
req = webob.Request.blank("/images/%s" % UUID2)
req.method = 'GET'
req.context = self.context
response = webob.Response(request=req)
self.serializer.show(response, self.FIXTURE)
for key, value in exp_headers.iteritems():
self.assertEquals(value, response.headers[key])
self.assertEqual(response.body, 'chunk67891123456789')
def test_show_notify(self):
"""Make sure an eventlet posthook for notify_image_sent is added."""
req = webob.Request.blank("/images/%s" % UUID2)
req.method = 'GET'
req.context = self.context
response = webob.Response(request=req)
response.environ['eventlet.posthooks'] = []
self.serializer.show(response, self.FIXTURE)
#just make sure the app_iter is called
for chunk in response.app_iter:
pass
self.assertNotEqual(response.environ['eventlet.posthooks'], [])
def test_image_send_notification(self):
req = webob.Request.blank("/images/%s" % UUID2)
req.method = 'GET'
req.remote_addr = '1.2.3.4'
req.context = self.context
image_meta = self.FIXTURE['image_meta']
called = {"notified": False}
expected_payload = {
'bytes_sent': 19,
'image_id': UUID2,
'owner_id': image_meta['owner'],
'receiver_tenant_id': self.receiving_tenant,
'receiver_user_id': self.receiving_user,
'destination_ip': '1.2.3.4',
}
def fake_info(_event_type, _payload):
self.assertDictEqual(_payload, expected_payload)
called['notified'] = True
self.stubs.Set(self.serializer.notifier, 'info', fake_info)
self.serializer.image_send_notification(19, 19, image_meta, req)
self.assertTrue(called['notified'])
def test_image_send_notification_error(self):
"""Ensure image.send notification is sent on error."""
req = webob.Request.blank("/images/%s" % UUID2)
req.method = 'GET'
req.remote_addr = '1.2.3.4'
req.context = self.context
image_meta = self.FIXTURE['image_meta']
called = {"notified": False}
expected_payload = {
'bytes_sent': 17,
'image_id': UUID2,
'owner_id': image_meta['owner'],
'receiver_tenant_id': self.receiving_tenant,
'receiver_user_id': self.receiving_user,
'destination_ip': '1.2.3.4',
}
def fake_error(_event_type, _payload):
self.assertDictEqual(_payload, expected_payload)
called['notified'] = True
self.stubs.Set(self.serializer.notifier, 'error', fake_error)
#expected and actually sent bytes differ
self.serializer.image_send_notification(17, 19, image_meta, req)
self.assertTrue(called['notified'])