4f3246db45
Implement a new mechanism to allow specifying a target in the environment to send events to. It adds zaqar as the first implementation. Depends-On: Ie04f9204f3ba0f75de32253f096f439c512cddee Change-Id: Icfc3864e08693cb4b4f921641af380b39bcf0bc0
138 lines
5.2 KiB
Python
138 lines
5.2 KiB
Python
#
|
|
# 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 six
|
|
|
|
import oslo_db.exception
|
|
|
|
from heat.common import exception
|
|
from heat.common.i18n import _
|
|
from heat.common import identifier
|
|
from heat.objects import event as event_object
|
|
|
|
|
|
class Event(object):
|
|
"""Class representing a Resource state change."""
|
|
|
|
def __init__(self, context, stack, action, status, reason,
|
|
physical_resource_id, resource_properties, resource_name,
|
|
resource_type, uuid=None, timestamp=None, id=None):
|
|
"""Initialise from a context, stack, and event information.
|
|
|
|
The timestamp and database ID may also be initialised if the event is
|
|
already in the database.
|
|
"""
|
|
self.context = context
|
|
self.stack = stack
|
|
self.action = action
|
|
self.status = status
|
|
self.reason = reason
|
|
self.physical_resource_id = physical_resource_id
|
|
self.resource_name = resource_name
|
|
self.resource_type = resource_type
|
|
try:
|
|
self.resource_properties = dict(resource_properties)
|
|
except ValueError as ex:
|
|
self.resource_properties = {'Error': six.text_type(ex)}
|
|
self.uuid = uuid
|
|
self.timestamp = timestamp
|
|
self.id = id
|
|
|
|
@classmethod
|
|
def load(cls, context, event_id, event=None, stack=None):
|
|
"""Retrieve an Event from the database."""
|
|
from heat.engine import stack as parser
|
|
|
|
ev = (event if event is not None else
|
|
event_object.Event.get_by_id(context, event_id))
|
|
if ev is None:
|
|
message = _('No event exists with id "%s"') % str(event_id)
|
|
raise exception.NotFound(message)
|
|
|
|
st = (stack if stack is not None else
|
|
parser.Stack.load(context, ev.stack_id))
|
|
|
|
return cls(context, st, ev.resource_action, ev.resource_status,
|
|
ev.resource_status_reason, ev.physical_resource_id,
|
|
ev.resource_properties, ev.resource_name,
|
|
ev.resource_type, ev.uuid, ev.created_at, ev.id)
|
|
|
|
def store(self):
|
|
"""Store the Event in the database."""
|
|
ev = {
|
|
'resource_name': self.resource_name,
|
|
'physical_resource_id': self.physical_resource_id,
|
|
'stack_id': self.stack.id,
|
|
'resource_action': self.action,
|
|
'resource_status': self.status,
|
|
'resource_status_reason': self.reason,
|
|
'resource_type': self.resource_type,
|
|
'resource_properties': self.resource_properties,
|
|
}
|
|
|
|
if self.uuid is not None:
|
|
ev['uuid'] = self.uuid
|
|
|
|
if self.timestamp is not None:
|
|
ev['created_at'] = self.timestamp
|
|
|
|
try:
|
|
new_ev = event_object.Event.create(self.context, ev)
|
|
except oslo_db.exception.DBError:
|
|
# Attempt do drop the largest key and re-store as we expect
|
|
# This to mostly happen with one large config blob property
|
|
max_key, max_val = max(ev['resource_properties'].items(),
|
|
key=lambda i: len(repr(i[1])))
|
|
err = 'Resource properties are too large to store'
|
|
ev['resource_properties'].update({'Error': err})
|
|
ev['resource_properties'][max_key] = '<Deleted, too large>'
|
|
try:
|
|
new_ev = event_object.Event.create(self.context, ev)
|
|
except oslo_db.exception.DBError:
|
|
# Give up and drop all properties..
|
|
ev['resource_properties'] = {'Error': err}
|
|
new_ev = event_object.Event.create(self.context, ev)
|
|
self.id = new_ev.id
|
|
self.timestamp = new_ev.created_at
|
|
self.uuid = new_ev.uuid
|
|
return self.id
|
|
|
|
def identifier(self):
|
|
"""Return a unique identifier for the event."""
|
|
if self.uuid is None:
|
|
return None
|
|
|
|
res_id = identifier.ResourceIdentifier(
|
|
resource_name=self.resource_name, **self.stack.identifier())
|
|
|
|
return identifier.EventIdentifier(event_id=str(self.uuid), **res_id)
|
|
|
|
def as_dict(self):
|
|
return {
|
|
'timestamp': self.timestamp.isoformat(),
|
|
'version': '0.1',
|
|
'type': 'os.heat.event',
|
|
'id': self.uuid,
|
|
'payload': {
|
|
'resource_name': self.resource_name,
|
|
'physical_resource_id': self.physical_resource_id,
|
|
'stack_id': self.stack.id,
|
|
'resource_action': self.action,
|
|
'resource_status': self.status,
|
|
'resource_status_reason': self.reason,
|
|
'resource_type': self.resource_type,
|
|
'resource_properties': self.resource_properties,
|
|
'version': '0.1'
|
|
}
|
|
}
|