Transform metrics.update notification

The metrics.update notification has been transformed to
the versioned notification framework.

Change-Id: I2f6aba0c032baf181b103d2a16b8e768815170c9
Implements: bp versioned-notification-transformation-rocky
This commit is contained in:
Takashi NATSUME 2018-07-06 11:41:46 +09:00
parent b46e9896f4
commit c2439bac7c
7 changed files with 308 additions and 5 deletions

View File

@ -0,0 +1,137 @@
{
"event_type": "metrics.update",
"payload": {
"nova_object.version": "1.0",
"nova_object.name": "MetricsPayload",
"nova_object.namespace": "nova",
"nova_object.data": {
"host_ip": "10.0.2.15",
"host": "compute",
"nodename": "fake-mini",
"metrics":[
{
"nova_object.version": "1.0",
"nova_object.name": "MetricPayload",
"nova_object.namespace": "nova",
"nova_object.data": {
"timestamp": "2012-10-29T13:42:11Z",
"source": "fake.SmallFakeDriver",
"numa_membw_values": null,
"name": "cpu.iowait.percent",
"value": 0
}
},
{
"nova_object.version": "1.0",
"nova_object.name": "MetricPayload",
"nova_object.namespace": "nova",
"nova_object.data": {
"timestamp": "2012-10-29T13:42:11Z",
"source": "fake.SmallFakeDriver",
"numa_membw_values": null,
"name": "cpu.frequency",
"value": 800
}
},
{
"nova_object.version": "1.0",
"nova_object.name": "MetricPayload",
"nova_object.namespace": "nova",
"nova_object.data": {
"timestamp": "2012-10-29T13:42:11Z",
"source": "fake.SmallFakeDriver",
"numa_membw_values": null,
"name": "cpu.idle.percent",
"value": 97
}
},
{
"nova_object.version": "1.0",
"nova_object.name": "MetricPayload",
"nova_object.namespace": "nova",
"nova_object.data": {
"timestamp": "2012-10-29T13:42:11Z",
"source": "fake.SmallFakeDriver",
"numa_membw_values": null,
"name": "cpu.iowait.time",
"value": 6121490000000
}
},
{
"nova_object.version": "1.0",
"nova_object.name": "MetricPayload",
"nova_object.namespace": "nova",
"nova_object.data": {
"timestamp": "2012-10-29T13:42:11Z",
"source": "fake.SmallFakeDriver",
"numa_membw_values": null,
"name": "cpu.kernel.percent",
"value": 0
}
},
{
"nova_object.version": "1.0",
"nova_object.name": "MetricPayload",
"nova_object.namespace": "nova",
"nova_object.data": {
"timestamp": "2012-10-29T13:42:11Z",
"source": "fake.SmallFakeDriver",
"numa_membw_values": null,
"name": "cpu.kernel.time",
"value": 5664160000000
}
},
{
"nova_object.version": "1.0",
"nova_object.name": "MetricPayload",
"nova_object.namespace": "nova",
"nova_object.data": {
"timestamp": "2012-10-29T13:42:11Z",
"source": "fake.SmallFakeDriver",
"numa_membw_values": null,
"name": "cpu.percent",
"value": 2
}
},
{
"nova_object.version": "1.0",
"nova_object.name": "MetricPayload",
"nova_object.namespace": "nova",
"nova_object.data": {
"timestamp": "2012-10-29T13:42:11Z",
"source": "fake.SmallFakeDriver",
"numa_membw_values": null,
"name": "cpu.user.percent",
"value": 1
}
},
{
"nova_object.version": "1.0",
"nova_object.name": "MetricPayload",
"nova_object.namespace": "nova",
"nova_object.data": {
"timestamp": "2012-10-29T13:42:11Z",
"source": "fake.SmallFakeDriver",
"numa_membw_values": null,
"name": "cpu.user.time",
"value": 26728850000000
}
},
{
"nova_object.version": "1.0",
"nova_object.name": "MetricPayload",
"nova_object.namespace": "nova",
"nova_object.data": {
"timestamp": "2012-10-29T13:42:11Z",
"source": "fake.SmallFakeDriver",
"numa_membw_values": null,
"name": "cpu.idle.time",
"value": 1592705190000000
}
}
]
}
},
"priority": "INFO",
"publisher_id": "nova-compute:compute"
}

View File

@ -640,15 +640,17 @@ class ResourceTracker(object):
{'mon': monitor, 'exc': exc})
# TODO(jaypipes): Remove this when compute_node.metrics doesn't need
# to be populated as a JSONified string.
metrics = metrics.to_list()
if len(metrics):
metric_list = metrics.to_list()
if len(metric_list):
metrics_info['nodename'] = nodename
metrics_info['metrics'] = metrics
metrics_info['metrics'] = metric_list
metrics_info['host'] = self.host
metrics_info['host_ip'] = CONF.my_ip
notifier = rpc.get_notifier(service='compute', host=nodename)
notifier.info(context, 'compute.metrics.update', metrics_info)
return metrics
compute_utils.notify_about_metrics_update(
context, self.host, CONF.my_ip, nodename, metrics)
return metric_list
def update_available_resource(self, context, nodename):
"""Override in-memory calculations of compute node resource usage based

View File

@ -40,6 +40,7 @@ from nova.notifications.objects import exception as notification_exception
from nova.notifications.objects import flavor as flavor_notification
from nova.notifications.objects import instance as instance_notification
from nova.notifications.objects import keypair as keypair_notification
from nova.notifications.objects import metrics as metrics_notification
from nova.notifications.objects import server_group as sg_notification
from nova import objects
from nova.objects import fields
@ -759,6 +760,34 @@ def notify_about_instance_rebuild(context, instance, host,
notification.emit(context)
@rpc.if_notifications_enabled
def notify_about_metrics_update(context, host, host_ip, nodename,
monitor_metric_list):
"""Send versioned notification about updating metrics
:param context: the request context
:param host: the host emitting the notification
:param host_ip: the IP address of the host
:param nodename: the node name
:param monitor_metric_list: the MonitorMetricList object
"""
payload = metrics_notification.MetricsPayload(
host=host,
host_ip=host_ip,
nodename=nodename,
monitor_metric_list=monitor_metric_list)
notification = metrics_notification.MetricsNotification(
context=context,
priority=fields.NotificationPriority.INFO,
publisher=notification_base.NotificationPublisher(
host=host, source=fields.NotificationSource.COMPUTE),
event_type=notification_base.EventType(
object='metrics',
action=fields.NotificationAction.UPDATE),
payload=payload)
notification.emit(context)
def refresh_info_cache_for_instance(context, instance):
"""Refresh the info cache for an instance.

View File

@ -0,0 +1,85 @@
# Copyright 2018 NTT Corporation
#
# 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.
from nova.notifications.objects import base
from nova.objects import base as nova_base
from nova.objects import fields
@base.notification_sample('metrics-update.json')
@nova_base.NovaObjectRegistry.register_notification
class MetricsNotification(base.NotificationBase):
# Version 1.0: Initial version
VERSION = '1.0'
fields = {
'payload': fields.ObjectField('MetricsPayload')
}
@nova_base.NovaObjectRegistry.register_notification
class MetricPayload(base.NotificationPayloadBase):
# Version 1.0: Initial version
VERSION = '1.0'
SCHEMA = {
'name': ('monitor_metric', 'name'),
'value': ('monitor_metric', 'value'),
'numa_membw_values': ('monitor_metric', 'numa_membw_values'),
'timestamp': ('monitor_metric', 'timestamp'),
'source': ('monitor_metric', 'source'),
}
fields = {
'name': fields.MonitorMetricTypeField(),
'value': fields.IntegerField(),
'numa_membw_values': fields.DictOfIntegersField(nullable=True),
'timestamp': fields.DateTimeField(),
'source': fields.StringField(),
}
def __init__(self, monitor_metric):
super(MetricPayload, self).__init__()
self.populate_schema(monitor_metric=monitor_metric)
@classmethod
def from_monitor_metric_list_obj(cls, monitor_metric_list):
"""Returns a list of MetricPayload objects based on the passed
MonitorMetricList object.
"""
payloads = []
for monitor_metric in monitor_metric_list:
payloads.append(cls(monitor_metric))
return payloads
@nova_base.NovaObjectRegistry.register_notification
class MetricsPayload(base.NotificationPayloadBase):
# Version 1.0: Initial version
VERSION = '1.0'
fields = {
'host': fields.StringField(),
'host_ip': fields.StringField(),
'nodename': fields.StringField(),
'metrics': fields.ListOfObjectsField('MetricPayload'),
}
def __init__(self, host, host_ip, nodename, monitor_metric_list):
super(MetricsPayload, self).__init__()
self.host = host
self.host_ip = host_ip
self.nodename = nodename
self.metrics = MetricPayload.from_monitor_metric_list_obj(
monitor_metric_list)

View File

@ -0,0 +1,42 @@
# Copyright 2018 NTT Corporation
#
# 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 nova.conf
from nova import context
from nova.tests.functional.notification_sample_tests \
import notification_sample_base
from nova.tests.unit import fake_notifier
CONF = nova.conf.CONF
class TestMetricsNotificationSample(
notification_sample_base.NotificationSampleTestBase):
def setUp(self):
self.flags(compute_monitors=['cpu.virt_driver'])
super(TestMetricsNotificationSample, self).setUp()
# Reset the cpu stats of the 'cpu.virt_driver' monitor
self.compute.manager._resource_tracker.monitors[0]._cpu_stats = {}
def test_metrics_update(self):
self.compute.manager.update_available_resource(
context.get_admin_context())
self.assertEqual(1, len(fake_notifier.VERSIONED_NOTIFICATIONS))
self._verify_notification(
'metrics-update',
replacements={'host_ip': CONF.my_ip},
actual=fake_notifier.VERSIONED_NOTIFICATIONS[0])

View File

@ -2995,7 +2995,8 @@ class ComputeMonitorTestCase(BaseTestCase):
u'Cannot get the metrics from %(mon)s; error: %(exc)s', mock.ANY)
self.assertEqual(0, len(metrics))
def test_get_host_metrics(self):
@mock.patch('nova.compute.utils.notify_about_metrics_update')
def test_get_host_metrics(self, mock_notify):
fake_notifier.stub_notifier(self)
self.addCleanup(fake_notifier.reset)
@ -3022,6 +3023,10 @@ class ComputeMonitorTestCase(BaseTestCase):
metrics = self.rt._get_host_metrics(self.context, _NODENAME)
mock_notify.assert_called_once_with(
self.context, _HOSTNAME, '1.1.1.1', _NODENAME,
test.MatchType(objects.MonitorMetricList))
expected_metrics = [
{
'timestamp': FakeCPUMonitor.NOW_TS.isoformat(),

View File

@ -404,6 +404,9 @@ notification_object_data = {
'IpPayload': '1.0-8ecf567a99e516d4af094439a7632d34',
'KeypairNotification': '1.0-a73147b93b520ff0061865849d3dfa56',
'KeypairPayload': '1.0-6daebbbde0e1bf35c1556b1ecd9385c1',
'MetricPayload': '1.0-bcdbe85048f335132e4c82a1b8fa3da8',
'MetricsNotification': '1.0-a73147b93b520ff0061865849d3dfa56',
'MetricsPayload': '1.0-65c69b15b4de5a8c01971cb5bb9ab650',
'NotificationPublisher': '2.2-b6ad48126247e10b46b6b0240e52e614',
'ServerGroupNotification': '1.0-a73147b93b520ff0061865849d3dfa56',
'ServerGroupPayload': '1.1-4ded2997ea1b07038f7af33ef5c45f7f',