154 lines
5.8 KiB
Python
154 lines
5.8 KiB
Python
# Copyright 2014 Red Hat, Inc.
|
|
#
|
|
# 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.
|
|
|
|
import webob
|
|
|
|
from nova.api.openstack import extensions
|
|
from nova.api.openstack import wsgi
|
|
from nova.api.openstack import xmlutil
|
|
from nova import compute
|
|
from nova import exception
|
|
from nova.i18n import _
|
|
from nova import objects
|
|
from nova.objects import external_event as external_event_obj
|
|
from nova.openstack.common import log as logging
|
|
|
|
|
|
LOG = logging.getLogger(__name__)
|
|
authorize = extensions.extension_authorizer('compute',
|
|
'os-server-external-events')
|
|
|
|
|
|
class EventTemplate(xmlutil.TemplateBuilder):
|
|
def construct(self):
|
|
root = xmlutil.TemplateElement('events')
|
|
elem1 = xmlutil.SubTemplateElement(root, 'event', selector='events')
|
|
elem2 = xmlutil.SubTemplateElement(elem1, xmlutil.Selector(0),
|
|
selector=xmlutil.get_items)
|
|
elem2.text = 1
|
|
return xmlutil.MasterTemplate(root, 1)
|
|
|
|
|
|
class EventDeserializer(wsgi.MetadataXMLDeserializer):
|
|
def _extract_event(self, event_node):
|
|
event = {}
|
|
for key in ('name', 'tag', 'server_uuid', 'status'):
|
|
node = self.find_first_child_named(event_node, key)
|
|
event[key] = self.extract_text(node)
|
|
return event
|
|
|
|
def default(self, string):
|
|
events = []
|
|
dom = xmlutil.safe_minidom_parse_string(string)
|
|
events_node = self.find_first_child_named(dom, 'events')
|
|
for event_node in self.find_children_named(events_node, 'event'):
|
|
events.append(self._extract_event(event_node))
|
|
return {'body': {'events': events}}
|
|
|
|
|
|
class ServerExternalEventsController(wsgi.Controller):
|
|
|
|
def __init__(self):
|
|
self.compute_api = compute.API()
|
|
super(ServerExternalEventsController, self).__init__()
|
|
|
|
@wsgi.deserializers(xml=EventDeserializer)
|
|
@wsgi.serializers(xml=EventTemplate)
|
|
def create(self, req, body):
|
|
"""Creates a new instance event."""
|
|
context = req.environ['nova.context']
|
|
authorize(context, action='create')
|
|
|
|
events = []
|
|
accepted = []
|
|
instances = {}
|
|
result = 200
|
|
|
|
body_events = body.get('events', [])
|
|
if not isinstance(body_events, list) or not len(body_events):
|
|
raise webob.exc.HTTPBadRequest()
|
|
|
|
for _event in body_events:
|
|
client_event = dict(_event)
|
|
event = objects.InstanceExternalEvent(context)
|
|
|
|
try:
|
|
event.instance_uuid = client_event.pop('server_uuid')
|
|
event.name = client_event.pop('name')
|
|
event.status = client_event.pop('status', 'completed')
|
|
event.tag = client_event.pop('tag', None)
|
|
except KeyError as missing_key:
|
|
msg = _('event entity requires key %(key)s') % missing_key
|
|
raise webob.exc.HTTPBadRequest(explanation=msg)
|
|
|
|
if client_event:
|
|
msg = (_('event entity contains unsupported items: %s') %
|
|
', '.join(client_event.keys()))
|
|
raise webob.exc.HTTPBadRequest(explanation=msg)
|
|
|
|
if event.status not in external_event_obj.EVENT_STATUSES:
|
|
raise webob.exc.HTTPBadRequest(
|
|
_('Invalid event status `%s\'') % event.status)
|
|
|
|
events.append(_event)
|
|
if event.instance_uuid not in instances:
|
|
try:
|
|
instance = objects.Instance.get_by_uuid(
|
|
context, event.instance_uuid)
|
|
instances[event.instance_uuid] = instance
|
|
except exception.InstanceNotFound:
|
|
LOG.debug('Dropping event %(name)s:%(tag)s for unknown '
|
|
'instance %(instance_uuid)s',
|
|
dict(event.iteritems()))
|
|
_event['status'] = 'failed'
|
|
_event['code'] = 404
|
|
result = 207
|
|
|
|
if event.instance_uuid in instances:
|
|
accepted.append(event)
|
|
_event['code'] = 200
|
|
LOG.audit(_('Create event %(name)s:%(tag)s for instance '
|
|
'%(instance_uuid)s'),
|
|
dict(event.iteritems()))
|
|
|
|
if accepted:
|
|
self.compute_api.external_instance_event(context,
|
|
instances.values(),
|
|
accepted)
|
|
else:
|
|
msg = _('No instances found for any event')
|
|
raise webob.exc.HTTPNotFound(explanation=msg)
|
|
|
|
# FIXME(cyeoh): This needs some infrastructure support so that
|
|
# we have a general way to do this
|
|
robj = wsgi.ResponseObject({'events': events})
|
|
robj._code = result
|
|
return robj
|
|
|
|
|
|
class Server_external_events(extensions.ExtensionDescriptor):
|
|
"""Server External Event Triggers."""
|
|
|
|
name = "ServerExternalEvents"
|
|
alias = "os-server-external-events"
|
|
namespace = ("http://docs.openstack.org/compute/ext/"
|
|
"server-external-events/api/v2")
|
|
updated = "2014-02-18T00:00:00Z"
|
|
|
|
def get_resources(self):
|
|
resource = extensions.ResourceExtension('os-server-external-events',
|
|
ServerExternalEventsController())
|
|
|
|
return [resource]
|