heat/heat/tests/openstack/aodh/test_gnocchi_alarm.py

566 lines
22 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
import mox
from heat.common import exception
from heat.common import template_format
from heat.engine.clients.os import aodh
from heat.engine.resources.openstack.aodh.gnocchi import (
alarm as gnocchi)
from heat.engine import scheduler
from heat.tests import common
from heat.tests import utils
gnocchi_resources_alarm_template = '''
heat_template_version: 2013-05-23
description: Gnocchi Resources Alarm Test
resources:
GnoResAlarm:
type: OS::Aodh::GnocchiResourcesAlarm
properties:
description: Do stuff with gnocchi
metric: cpu_util
aggregation_method: mean
granularity: 60
evaluation_periods: 1
threshold: 50
alarm_actions: []
resource_type: instance
resource_id: 5a517ceb-b068-4aca-9eb9-3e4eb9b90d9a
comparison_operator: gt
'''
gnocchi_aggregation_by_metrics_alarm_template = '''
heat_template_version: 2013-05-23
description: Gnocchi Aggregation by Metrics Alarm Test
resources:
GnoAggregationByMetricsAlarm:
type: OS::Aodh::GnocchiAggregationByMetricsAlarm
properties:
description: Do stuff with gnocchi metrics
metrics: ["911fce07-e0d7-4210-8c8c-4a9d811fcabc",
"2543d435-fe93-4443-9351-fb0156930f94"]
aggregation_method: mean
granularity: 60
evaluation_periods: 1
threshold: 50
alarm_actions: []
comparison_operator: gt
'''
gnocchi_aggregation_by_resources_alarm_template = '''
heat_template_version: 2013-05-23
description: Gnocchi Aggregation by Resources Alarm Test
resources:
GnoAggregationByResourcesAlarm:
type: OS::Aodh::GnocchiAggregationByResourcesAlarm
properties:
description: Do stuff with gnocchi aggregation by resource
aggregation_method: mean
granularity: 60
evaluation_periods: 1
threshold: 50
metric: cpu_util
alarm_actions: []
resource_type: instance
query: '{"=": {"server_group": "my_autoscaling_group"}}'
comparison_operator: gt
'''
FakeAodhAlarm = {'other_attrs': 'val',
'alarm_id': 'foo'}
class GnocchiResourcesAlarmTest(common.HeatTestCase):
def setUp(self):
super(GnocchiResourcesAlarmTest, self).setUp()
self.fc = mock.Mock()
def create_alarm(self):
self.patchobject(aodh.AodhClientPlugin,
'_create').return_value = self.fc
self.m.StubOutWithMock(self.fc.alarm, 'create')
self.fc.alarm.create(
{
'alarm_actions': [],
'description': u'Do stuff with gnocchi',
'enabled': True,
'insufficient_data_actions': [],
'ok_actions': [],
'name': mox.IgnoreArg(),
'type': 'gnocchi_resources_threshold',
'repeat_actions': True,
'gnocchi_resources_threshold_rule': {
"metric": "cpu_util",
"aggregation_method": "mean",
"granularity": 60,
"evaluation_periods": 1,
"threshold": 50,
"resource_type": "instance",
"resource_id": "5a517ceb-b068-4aca-9eb9-3e4eb9b90d9a",
"comparison_operator": "gt",
},
'time_constraints': [],
'severity': 'low'
}).AndReturn(FakeAodhAlarm)
self.tmpl = template_format.parse(gnocchi_resources_alarm_template)
self.stack = utils.parse_stack(self.tmpl)
resource_defns = self.stack.t.resource_definitions(self.stack)
return gnocchi.AodhGnocchiResourcesAlarm(
'GnoResAlarm', resource_defns['GnoResAlarm'], self.stack)
def test_update(self):
rsrc = self.create_alarm()
self.m.StubOutWithMock(self.fc.alarm, 'update')
self.fc.alarm.update(
'foo',
{
'alarm_actions': [],
'description': u'Do stuff with gnocchi',
'enabled': True,
'insufficient_data_actions': [],
'ok_actions': [],
'repeat_actions': True,
'gnocchi_resources_threshold_rule': {
"metric": "cpu_util",
"aggregation_method": "mean",
"granularity": 60,
"evaluation_periods": 1,
"threshold": 50,
"resource_type": "instance",
"resource_id": "d3d6c642-921e-4fc2-9c5f-15d9a5afb598",
"comparison_operator": "gt",
},
'time_constraints': [],
'severity': 'low'
}
)
self.m.ReplayAll()
scheduler.TaskRunner(rsrc.create)()
props = self.tmpl['resources']['GnoResAlarm']['properties']
props['resource_id'] = 'd3d6c642-921e-4fc2-9c5f-15d9a5afb598'
update_template = rsrc.t.freeze(properties=props)
scheduler.TaskRunner(rsrc.update, update_template)()
self.assertEqual((rsrc.UPDATE, rsrc.COMPLETE), rsrc.state)
self.m.VerifyAll()
def _prepare_resource(self, for_check=True):
snippet = template_format.parse(gnocchi_resources_alarm_template)
self.stack = utils.parse_stack(snippet)
res = self.stack['GnoResAlarm']
if for_check:
res.state_set(res.CREATE, res.COMPLETE)
res.client = mock.Mock()
mock_alarm = mock.Mock(enabled=True, state='ok')
res.client().alarm.get.return_value = mock_alarm
return res
def test_create(self):
rsrc = self.create_alarm()
self.m.ReplayAll()
scheduler.TaskRunner(rsrc.create)()
self.assertEqual((rsrc.CREATE, rsrc.COMPLETE), rsrc.state)
self.assertEqual('foo', rsrc.resource_id)
self.m.VerifyAll()
def test_suspend(self):
rsrc = self.create_alarm()
self.m.StubOutWithMock(self.fc.alarm, 'update')
self.fc.alarm.update('foo', {'enabled': False})
self.m.ReplayAll()
scheduler.TaskRunner(rsrc.create)()
scheduler.TaskRunner(rsrc.suspend)()
self.assertEqual((rsrc.SUSPEND, rsrc.COMPLETE), rsrc.state)
self.m.VerifyAll()
def test_resume(self):
rsrc = self.create_alarm()
self.m.StubOutWithMock(self.fc.alarm, 'update')
self.fc.alarm.update('foo', {'enabled': True})
self.m.ReplayAll()
scheduler.TaskRunner(rsrc.create)()
rsrc.state_set(rsrc.SUSPEND, rsrc.COMPLETE)
scheduler.TaskRunner(rsrc.resume)()
self.assertEqual((rsrc.RESUME, rsrc.COMPLETE), rsrc.state)
self.m.VerifyAll()
def test_check(self):
res = self._prepare_resource()
scheduler.TaskRunner(res.check)()
self.assertEqual((res.CHECK, res.COMPLETE), res.state)
def test_check_failure(self):
res = self._prepare_resource()
res.client().alarm.get.side_effect = Exception('Boom')
self.assertRaises(exception.ResourceFailure,
scheduler.TaskRunner(res.check))
self.assertEqual((res.CHECK, res.FAILED), res.state)
self.assertIn('Boom', res.status_reason)
def test_show_resource(self):
res = self._prepare_resource(for_check=False)
res.client().alarm.create.return_value = FakeAodhAlarm
res.client().alarm.get.return_value = FakeAodhAlarm
scheduler.TaskRunner(res.create)()
self.assertEqual(FakeAodhAlarm, res.FnGetAtt('show'))
def test_gnocchi_alarm_live_state(self):
snippet = template_format.parse(gnocchi_resources_alarm_template)
self.stack = utils.parse_stack(snippet)
resource_defns = self.stack.t.resource_definitions(self.stack)
self.rsrc_defn = resource_defns['GnoResAlarm']
self.client = mock.Mock()
self.patchobject(gnocchi.AodhGnocchiResourcesAlarm, 'client',
return_value=self.client)
alarm_res = gnocchi.AodhGnocchiResourcesAlarm(
'alarm', self.rsrc_defn, self.stack)
alarm_res.create()
value = {
'description': 'Do stuff with gnocchi',
'alarm_actions': [],
'time_constraints': [],
'gnocchi_resources_threshold_rule': {
'resource_id': '5a517ceb-b068-4aca-9eb9-3e4eb9b90d9a',
'metric': 'cpu_util',
'evaluation_periods': 1,
'aggregation_method': 'mean',
'granularity': 60,
'threshold': 50,
'comparison_operator': 'gt',
'resource_type': 'instance'
}
}
self.client.alarm.get.return_value = value
expected_data = {
'description': 'Do stuff with gnocchi',
'alarm_actions': [],
'resource_id': '5a517ceb-b068-4aca-9eb9-3e4eb9b90d9a',
'metric': 'cpu_util',
'evaluation_periods': 1,
'aggregation_method': 'mean',
'granularity': 60,
'threshold': 50,
'comparison_operator': 'gt',
'resource_type': 'instance',
'insufficient_data_actions': None,
'enabled': None,
'ok_actions': None,
'repeat_actions': None,
'severity': None
}
reality = alarm_res.get_live_state(alarm_res.properties)
self.assertEqual(expected_data, reality)
class GnocchiAggregationByMetricsAlarmTest(GnocchiResourcesAlarmTest):
def create_alarm(self):
self.patchobject(aodh.AodhClientPlugin,
'_create').return_value = self.fc
self.m.StubOutWithMock(self.fc.alarm, 'create')
self.fc.alarm.create(
{
'alarm_actions': [],
'description': u'Do stuff with gnocchi metrics',
'enabled': True,
'insufficient_data_actions': [],
'ok_actions': [],
'name': mox.IgnoreArg(),
'type': 'gnocchi_aggregation_by_metrics_threshold',
'repeat_actions': True,
'gnocchi_aggregation_by_metrics_threshold_rule': {
"aggregation_method": "mean",
"granularity": 60,
"evaluation_periods": 1,
"threshold": 50,
"comparison_operator": "gt",
"metrics": ["911fce07-e0d7-4210-8c8c-4a9d811fcabc",
"2543d435-fe93-4443-9351-fb0156930f94"],
},
'time_constraints': [],
'severity': 'low'}
).AndReturn(FakeAodhAlarm)
self.tmpl = template_format.parse(
gnocchi_aggregation_by_metrics_alarm_template)
self.stack = utils.parse_stack(self.tmpl)
resource_defns = self.stack.t.resource_definitions(self.stack)
return gnocchi.AodhGnocchiAggregationByMetricsAlarm(
'GnoAggregationByMetricsAlarm',
resource_defns['GnoAggregationByMetricsAlarm'], self.stack)
def test_update(self):
rsrc = self.create_alarm()
self.m.StubOutWithMock(self.fc.alarm, 'update')
self.fc.alarm.update(
'foo',
{
'alarm_actions': [],
'description': u'Do stuff with gnocchi metrics',
'enabled': True,
'insufficient_data_actions': [],
'ok_actions': [],
'repeat_actions': True,
'gnocchi_aggregation_by_metrics_threshold_rule': {
"aggregation_method": "mean",
"granularity": 60,
"evaluation_periods": 1,
"threshold": 50,
"comparison_operator": "gt",
'metrics': ['d3d6c642-921e-4fc2-9c5f-15d9a5afb598',
'bc60f822-18a0-4a0c-94e7-94c554b00901']
},
'time_constraints': [],
'severity': 'low'
}
)
self.m.ReplayAll()
scheduler.TaskRunner(rsrc.create)()
snippet = self.tmpl['resources']['GnoAggregationByMetricsAlarm']
props = snippet['properties'].copy()
props['metrics'] = ['d3d6c642-921e-4fc2-9c5f-15d9a5afb598',
'bc60f822-18a0-4a0c-94e7-94c554b00901']
update_template = rsrc.t.freeze(properties=props)
scheduler.TaskRunner(rsrc.update, update_template)()
self.assertEqual((rsrc.UPDATE, rsrc.COMPLETE), rsrc.state)
self.m.VerifyAll()
def _prepare_resource(self, for_check=True):
snippet = template_format.parse(
gnocchi_aggregation_by_metrics_alarm_template)
self.stack = utils.parse_stack(snippet)
res = self.stack['GnoAggregationByMetricsAlarm']
if for_check:
res.state_set(res.CREATE, res.COMPLETE)
res.client = mock.Mock()
mock_alarm = mock.Mock(enabled=True, state='ok')
res.client().alarm.get.return_value = mock_alarm
return res
def test_show_resource(self):
res = self._prepare_resource(for_check=False)
res.client().alarm.create.return_value = FakeAodhAlarm
res.client().alarm.get.return_value = FakeAodhAlarm
scheduler.TaskRunner(res.create)()
self.assertEqual(FakeAodhAlarm, res.FnGetAtt('show'))
def test_gnocchi_alarm_aggr_by_metrics_live_state(self):
snippet = template_format.parse(
gnocchi_aggregation_by_metrics_alarm_template)
self.stack = utils.parse_stack(snippet)
resource_defns = self.stack.t.resource_definitions(self.stack)
self.rsrc_defn = resource_defns['GnoAggregationByMetricsAlarm']
self.client = mock.Mock()
self.patchobject(gnocchi.AodhGnocchiAggregationByMetricsAlarm,
'client', return_value=self.client)
alarm_res = gnocchi.AodhGnocchiAggregationByMetricsAlarm(
'alarm', self.rsrc_defn, self.stack)
alarm_res.create()
value = {
'description': 'Do stuff with gnocchi metrics',
'alarm_actions': [],
'time_constraints': [],
'gnocchi_aggregation_by_metrics_threshold_rule': {
'metrics': ['911fce07-e0d7-4210-8c8c-4a9d811fcabc',
'2543d435-fe93-4443-9351-fb0156930f94'],
'evaluation_periods': 1,
'aggregation_method': 'mean',
'granularity': 60,
'threshold': 50,
'comparison_operator': 'gt'
}
}
self.client.alarm.get.return_value = value
expected_data = {
'description': 'Do stuff with gnocchi metrics',
'alarm_actions': [],
'metrics': ['911fce07-e0d7-4210-8c8c-4a9d811fcabc',
'2543d435-fe93-4443-9351-fb0156930f94'],
'evaluation_periods': 1,
'aggregation_method': 'mean',
'granularity': 60,
'threshold': 50,
'comparison_operator': 'gt',
'insufficient_data_actions': None,
'enabled': None,
'ok_actions': None,
'repeat_actions': None,
'severity': None
}
reality = alarm_res.get_live_state(alarm_res.properties)
self.assertEqual(expected_data, reality)
class GnocchiAggregationByResourcesAlarmTest(GnocchiResourcesAlarmTest):
def create_alarm(self):
self.patchobject(aodh.AodhClientPlugin,
'_create').return_value = self.fc
self.m.StubOutWithMock(self.fc.alarm, 'create')
self.fc.alarm.create(
{
'alarm_actions': [],
'description': 'Do stuff with gnocchi aggregation by resource',
'enabled': True,
'insufficient_data_actions': [],
'ok_actions': [],
'name': mox.IgnoreArg(),
'type': 'gnocchi_aggregation_by_resources_threshold',
'repeat_actions': True,
'gnocchi_aggregation_by_resources_threshold_rule': {
"aggregation_method": "mean",
"granularity": 60,
"evaluation_periods": 1,
"threshold": 50,
"comparison_operator": "gt",
"metric": "cpu_util",
"resource_type": "instance",
"query": '{"=": {"server_group": "my_autoscaling_group"}}',
},
'time_constraints': [],
'severity': 'low'}
).AndReturn(FakeAodhAlarm)
self.tmpl = template_format.parse(
gnocchi_aggregation_by_resources_alarm_template)
self.stack = utils.parse_stack(self.tmpl)
resource_defns = self.stack.t.resource_definitions(self.stack)
return gnocchi.AodhGnocchiAggregationByResourcesAlarm(
'GnoAggregationByResourcesAlarm',
resource_defns['GnoAggregationByResourcesAlarm'], self.stack)
def test_update(self):
rsrc = self.create_alarm()
self.m.StubOutWithMock(self.fc.alarm, 'update')
self.fc.alarm.update(
'foo',
{
'alarm_actions': [],
'description': 'Do stuff with gnocchi aggregation by resource',
'enabled': True,
'insufficient_data_actions': [],
'ok_actions': [],
'repeat_actions': True,
'gnocchi_aggregation_by_resources_threshold_rule': {
"aggregation_method": "mean",
"granularity": 60,
"evaluation_periods": 1,
"threshold": 50,
"comparison_operator": "gt",
"metric": "cpu_util",
"resource_type": "instance",
"query": '{"=": {"server_group": "my_new_group"}}',
},
'time_constraints': [],
'severity': 'low'
}
)
self.m.ReplayAll()
scheduler.TaskRunner(rsrc.create)()
snippet = self.tmpl['resources']['GnoAggregationByResourcesAlarm']
props = snippet['properties'].copy()
props['query'] = '{"=": {"server_group": "my_new_group"}}'
update_template = rsrc.t.freeze(properties=props)
scheduler.TaskRunner(rsrc.update, update_template)()
self.assertEqual((rsrc.UPDATE, rsrc.COMPLETE), rsrc.state)
self.m.VerifyAll()
def _prepare_resource(self, for_check=True):
snippet = template_format.parse(
gnocchi_aggregation_by_resources_alarm_template)
self.stack = utils.parse_stack(snippet)
res = self.stack['GnoAggregationByResourcesAlarm']
if for_check:
res.state_set(res.CREATE, res.COMPLETE)
res.client = mock.Mock()
mock_alarm = mock.Mock(enabled=True, state='ok')
res.client().alarm.get.return_value = mock_alarm
return res
def test_show_resource(self):
res = self._prepare_resource(for_check=False)
res.client().alarm.create.return_value = FakeAodhAlarm
res.client().alarm.get.return_value = FakeAodhAlarm
scheduler.TaskRunner(res.create)()
self.assertEqual(FakeAodhAlarm, res.FnGetAtt('show'))
def test_gnocchi_alarm_aggr_by_resources_live_state(self):
snippet = template_format.parse(
gnocchi_aggregation_by_resources_alarm_template)
self.stack = utils.parse_stack(snippet)
resource_defns = self.stack.t.resource_definitions(self.stack)
self.rsrc_defn = resource_defns['GnoAggregationByResourcesAlarm']
self.client = mock.Mock()
self.patchobject(gnocchi.AodhGnocchiAggregationByResourcesAlarm,
'client', return_value=self.client)
alarm_res = gnocchi.AodhGnocchiAggregationByResourcesAlarm(
'alarm', self.rsrc_defn, self.stack)
alarm_res.create()
value = {
'description': 'Do stuff with gnocchi aggregation by resource',
'alarm_actions': [],
'time_constraints': [],
'gnocchi_aggregation_by_resources_threshold_rule': {
'metric': 'cpu_util',
'resource_type': 'instance',
'query': "{'=': {'server_group': 'my_autoscaling_group'}}",
'evaluation_periods': 1,
'aggregation_method': 'mean',
'granularity': 60,
'threshold': 50,
'comparison_operator': 'gt'
}
}
self.client.alarm.get.return_value = value
expected_data = {
'description': 'Do stuff with gnocchi aggregation by resource',
'alarm_actions': [],
'metric': 'cpu_util',
'resource_type': 'instance',
'query': "{'=': {'server_group': 'my_autoscaling_group'}}",
'evaluation_periods': 1,
'aggregation_method': 'mean',
'granularity': 60,
'threshold': 50,
'comparison_operator': 'gt',
'insufficient_data_actions': None,
'enabled': None,
'ok_actions': None,
'repeat_actions': None,
'severity': None
}
reality = alarm_res.get_live_state(alarm_res.properties)
self.assertEqual(expected_data, reality)