
This adds optional notifications emitted when ironic changes a node's power state or when ironic detects a change in a node's power state. These notifications can be consumed by any external service listening to the message bus to perform functions like tracking node power state changes over time or automatically responding to anomalous power states. The event_types of the new notifications are: * baremetal.node.power_set.{start,end,error} * baremetal.node.power_state_corrected.success This also adds a new NodePayload class for notification payloads related to nodes. Change-Id: I82702e7f959d666bb02b59d1fc53ab50b519cb74 Closes-Bug: 1526408
262 lines
11 KiB
Python
262 lines
11 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 mock
|
|
|
|
from ironic.common import exception
|
|
from ironic.objects import base
|
|
from ironic.objects import fields
|
|
from ironic.objects import notification
|
|
from ironic.tests import base as test_base
|
|
|
|
|
|
class TestNotificationBase(test_base.TestCase):
|
|
|
|
@base.IronicObjectRegistry.register_if(False)
|
|
class TestObject(base.IronicObject):
|
|
VERSION = '1.0'
|
|
fields = {
|
|
'fake_field_1': fields.StringField(),
|
|
'fake_field_2': fields.IntegerField(nullable=True)
|
|
}
|
|
|
|
@base.IronicObjectRegistry.register_if(False)
|
|
class TestNotificationPayload(notification.NotificationPayloadBase):
|
|
VERSION = '1.0'
|
|
|
|
SCHEMA = {
|
|
'fake_field_a': ('test_obj', 'fake_field_1'),
|
|
'fake_field_b': ('test_obj', 'fake_field_2')
|
|
}
|
|
|
|
fields = {
|
|
'fake_field_a': fields.StringField(),
|
|
'fake_field_b': fields.IntegerField(),
|
|
'an_extra_field': fields.StringField(nullable=False),
|
|
'an_optional_field': fields.IntegerField(nullable=True)
|
|
}
|
|
|
|
@base.IronicObjectRegistry.register_if(False)
|
|
class TestNotificationPayloadEmptySchema(
|
|
notification.NotificationPayloadBase):
|
|
VERSION = '1.0'
|
|
|
|
fields = {
|
|
'fake_field': fields.StringField()
|
|
}
|
|
|
|
@base.IronicObjectRegistry.register_if(False)
|
|
class TestNotification(notification.NotificationBase):
|
|
VERSION = '1.0'
|
|
fields = {
|
|
'payload': fields.ObjectField('TestNotificationPayload')
|
|
}
|
|
|
|
@base.IronicObjectRegistry.register_if(False)
|
|
class TestNotificationEmptySchema(notification.NotificationBase):
|
|
VERSION = '1.0'
|
|
fields = {
|
|
'payload': fields.ObjectField('TestNotificationPayloadEmptySchema')
|
|
}
|
|
|
|
def setUp(self):
|
|
super(TestNotificationBase, self).setUp()
|
|
self.fake_obj = self.TestObject(fake_field_1='fake1', fake_field_2=2)
|
|
|
|
def _verify_notification(self, mock_notifier, mock_context,
|
|
expected_event_type, expected_payload,
|
|
expected_publisher, notif_level):
|
|
mock_notifier.prepare.assert_called_once_with(
|
|
publisher_id=expected_publisher)
|
|
# Handler actually sending out the notification depends on the
|
|
# notification level
|
|
mock_notify = getattr(mock_notifier.prepare.return_value, notif_level)
|
|
self.assertTrue(mock_notify.called)
|
|
self.assertEqual(mock_context, mock_notify.call_args[0][0])
|
|
self.assertEqual(expected_event_type,
|
|
mock_notify.call_args[1]['event_type'])
|
|
actual_payload = mock_notify.call_args[1]['payload']
|
|
self.assertJsonEqual(expected_payload, actual_payload)
|
|
|
|
@mock.patch('ironic.common.rpc.VERSIONED_NOTIFIER')
|
|
def test_emit_notification(self, mock_notifier):
|
|
self.config(notification_level='debug')
|
|
payload = self.TestNotificationPayload(an_extra_field='extra',
|
|
an_optional_field=1)
|
|
payload.populate_schema(test_obj=self.fake_obj)
|
|
notif = self.TestNotification(
|
|
event_type=notification.EventType(
|
|
object='test_object', action='test',
|
|
status=fields.NotificationStatus.START),
|
|
level=fields.NotificationLevel.DEBUG,
|
|
publisher=notification.NotificationPublisher(
|
|
service='ironic-conductor',
|
|
host='host'),
|
|
payload=payload)
|
|
|
|
mock_context = mock.Mock()
|
|
notif.emit(mock_context)
|
|
|
|
self._verify_notification(
|
|
mock_notifier,
|
|
mock_context,
|
|
expected_event_type='baremetal.test_object.test.start',
|
|
expected_payload={
|
|
'ironic_object.name': 'TestNotificationPayload',
|
|
'ironic_object.data': {
|
|
'fake_field_a': 'fake1',
|
|
'fake_field_b': 2,
|
|
'an_extra_field': 'extra',
|
|
'an_optional_field': 1
|
|
},
|
|
'ironic_object.version': '1.0',
|
|
'ironic_object.namespace': 'ironic'},
|
|
expected_publisher='ironic-conductor.host',
|
|
notif_level=fields.NotificationLevel.DEBUG)
|
|
|
|
@mock.patch('ironic.common.rpc.VERSIONED_NOTIFIER')
|
|
def test_no_emit_level_too_low(self, mock_notifier):
|
|
# Make sure notification doesn't emit when set notification
|
|
# level < config level
|
|
self.config(notification_level='warning')
|
|
payload = self.TestNotificationPayload(an_extra_field='extra',
|
|
an_optional_field=1)
|
|
payload.populate_schema(test_obj=self.fake_obj)
|
|
notif = self.TestNotification(
|
|
event_type=notification.EventType(
|
|
object='test_object', action='test',
|
|
status=fields.NotificationStatus.START),
|
|
level=fields.NotificationLevel.DEBUG,
|
|
publisher=notification.NotificationPublisher(
|
|
service='ironic-conductor',
|
|
host='host'),
|
|
payload=payload)
|
|
|
|
mock_context = mock.Mock()
|
|
notif.emit(mock_context)
|
|
|
|
self.assertFalse(mock_notifier.called)
|
|
|
|
@mock.patch('ironic.common.rpc.VERSIONED_NOTIFIER')
|
|
def test_no_emit_notifs_disabled(self, mock_notifier):
|
|
# Make sure notifications aren't emitted when notification_level
|
|
# isn't defined, indicating notifications should be disabled
|
|
payload = self.TestNotificationPayload(an_extra_field='extra',
|
|
an_optional_field=1)
|
|
payload.populate_schema(test_obj=self.fake_obj)
|
|
notif = self.TestNotification(
|
|
event_type=notification.EventType(
|
|
object='test_object', action='test',
|
|
status=fields.NotificationStatus.START),
|
|
level=fields.NotificationLevel.DEBUG,
|
|
publisher=notification.NotificationPublisher(
|
|
service='ironic-conductor',
|
|
host='host'),
|
|
payload=payload)
|
|
|
|
mock_context = mock.Mock()
|
|
notif.emit(mock_context)
|
|
|
|
self.assertFalse(mock_notifier.called)
|
|
|
|
@mock.patch('ironic.common.rpc.VERSIONED_NOTIFIER')
|
|
def test_no_emit_schema_not_populated(self, mock_notifier):
|
|
self.config(notification_level='debug')
|
|
payload = self.TestNotificationPayload(an_extra_field='extra',
|
|
an_optional_field=1)
|
|
notif = self.TestNotification(
|
|
event_type=notification.EventType(
|
|
object='test_object', action='test',
|
|
status=fields.NotificationStatus.START),
|
|
level=fields.NotificationLevel.DEBUG,
|
|
publisher=notification.NotificationPublisher(
|
|
service='ironic-conductor',
|
|
host='host'),
|
|
payload=payload)
|
|
|
|
mock_context = mock.Mock()
|
|
self.assertRaises(exception.NotificationPayloadError, notif.emit,
|
|
mock_context)
|
|
self.assertFalse(mock_notifier.called)
|
|
|
|
@mock.patch('ironic.common.rpc.VERSIONED_NOTIFIER')
|
|
def test_emit_notification_empty_schema(self, mock_notifier):
|
|
self.config(notification_level='debug')
|
|
payload = self.TestNotificationPayloadEmptySchema(fake_field='123')
|
|
notif = self.TestNotificationEmptySchema(
|
|
event_type=notification.EventType(
|
|
object='test_object', action='test',
|
|
status=fields.NotificationStatus.ERROR),
|
|
level=fields.NotificationLevel.ERROR,
|
|
publisher=notification.NotificationPublisher(
|
|
service='ironic-conductor',
|
|
host='host'),
|
|
payload=payload)
|
|
|
|
mock_context = mock.Mock()
|
|
notif.emit(mock_context)
|
|
|
|
self._verify_notification(
|
|
mock_notifier,
|
|
mock_context,
|
|
expected_event_type='baremetal.test_object.test.error',
|
|
expected_payload={
|
|
'ironic_object.name': 'TestNotificationPayloadEmptySchema',
|
|
'ironic_object.data': {
|
|
'fake_field': '123',
|
|
},
|
|
'ironic_object.version': '1.0',
|
|
'ironic_object.namespace': 'ironic'},
|
|
expected_publisher='ironic-conductor.host',
|
|
notif_level=fields.NotificationLevel.ERROR)
|
|
|
|
def test_populate_schema(self):
|
|
payload = self.TestNotificationPayload(an_extra_field='extra',
|
|
an_optional_field=1)
|
|
payload.populate_schema(test_obj=self.fake_obj)
|
|
self.assertEqual('extra', payload.an_extra_field)
|
|
self.assertEqual(1, payload.an_optional_field)
|
|
self.assertEqual(self.fake_obj.fake_field_1, payload.fake_field_a)
|
|
self.assertEqual(self.fake_obj.fake_field_2, payload.fake_field_b)
|
|
|
|
def test_populate_schema_missing_obj_field(self):
|
|
test_obj = self.TestObject(fake_field_1='populated')
|
|
payload = self.TestNotificationPayload(an_extra_field='too extra')
|
|
self.assertRaises(exception.NotificationSchemaKeyError,
|
|
payload.populate_schema,
|
|
test_obj=test_obj)
|
|
|
|
def test_event_type_with_status(self):
|
|
event_type = notification.EventType(
|
|
object="some_obj", action="some_action", status="success")
|
|
self.assertEqual("baremetal.some_obj.some_action.success",
|
|
event_type.to_event_type_field())
|
|
|
|
def test_event_type_without_status_fails(self):
|
|
event_type = notification.EventType(
|
|
object="some_obj", action="some_action")
|
|
self.assertRaises(NotImplementedError,
|
|
event_type.to_event_type_field)
|
|
|
|
def test_event_type_invalid_status_fails(self):
|
|
self.assertRaises(ValueError,
|
|
notification.EventType, object="some_obj",
|
|
action="some_action", status="invalid")
|
|
|
|
def test_event_type_make_status_invalid(self):
|
|
def make_status_invalid():
|
|
event_type.status = "Roar"
|
|
|
|
event_type = notification.EventType(
|
|
object='test_object', action='test', status='start')
|
|
self.assertRaises(ValueError, make_status_invalid)
|