heat/heat/engine/resources/openstack/aodh/alarm.py

437 lines
16 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
from heat.common.i18n import _
from heat.engine import constraints
from heat.engine import properties
from heat.engine.resources import alarm_base
from heat.engine.resources.openstack.heat import none_resource
from heat.engine import support
from heat.engine import translation
class AodhAlarm(alarm_base.BaseAlarm):
"""A resource that implements alarming service of Aodh.
A resource that allows for the setting alarms based on threshold evaluation
for a collection of samples. Also, you can define actions to take if state
of watched resource will be satisfied specified conditions. For example, it
can watch for the memory consumption and when it reaches 70% on a given
instance if the instance has been up for more than 10 min, some action will
be called.
"""
support_status = support.SupportStatus(
status=support.DEPRECATED,
message=_('Theshold alarm relies on ceilometer-api and has been '
'deprecated in aodh since Ocata. Use '
'OS::Aodh::GnocchiAggregationByResourcesAlarm instead.'),
version='10.0.0',
previous_status=support.SupportStatus(version='2014.1'))
PROPERTIES = (
COMPARISON_OPERATOR, EVALUATION_PERIODS, METER_NAME, PERIOD,
STATISTIC, THRESHOLD, MATCHING_METADATA, QUERY,
) = (
'comparison_operator', 'evaluation_periods', 'meter_name', 'period',
'statistic', 'threshold', 'matching_metadata', 'query',
)
properties_schema = {
COMPARISON_OPERATOR: properties.Schema(
properties.Schema.STRING,
_('Operator used to compare specified statistic with threshold.'),
constraints=[alarm_base.BaseAlarm.QF_OP_VALS],
update_allowed=True
),
EVALUATION_PERIODS: properties.Schema(
properties.Schema.INTEGER,
_('Number of periods to evaluate over.'),
update_allowed=True
),
METER_NAME: properties.Schema(
properties.Schema.STRING,
_('Meter name watched by the alarm.'),
required=True
),
PERIOD: properties.Schema(
properties.Schema.INTEGER,
_('Period (seconds) to evaluate over.'),
update_allowed=True
),
STATISTIC: properties.Schema(
properties.Schema.STRING,
_('Meter statistic to evaluate.'),
constraints=[
constraints.AllowedValues(['count', 'avg', 'sum', 'min',
'max']),
],
update_allowed=True
),
THRESHOLD: properties.Schema(
properties.Schema.NUMBER,
_('Threshold to evaluate against.'),
required=True,
update_allowed=True
),
MATCHING_METADATA: properties.Schema(
properties.Schema.MAP,
_('Meter should match this resource metadata (key=value) '
'additionally to the meter_name.'),
default={},
update_allowed=True
),
QUERY: properties.Schema(
properties.Schema.LIST,
_('A list of query factors, each comparing '
'a Sample attribute with a value. '
'Implicitly combined with matching_metadata, if any.'),
update_allowed=True,
support_status=support.SupportStatus(version='2015.1'),
schema=properties.Schema(
properties.Schema.MAP,
schema={
alarm_base.BaseAlarm.QF_FIELD: properties.Schema(
properties.Schema.STRING,
_('Name of attribute to compare. '
'Names of the form metadata.user_metadata.X '
'or metadata.metering.X are equivalent to what '
'you can address through matching_metadata; '
'the former for Nova meters, '
'the latter for all others. '
'To see the attributes of your Samples, '
'use `ceilometer --debug sample-list`.')
),
alarm_base.BaseAlarm.QF_TYPE: properties.Schema(
properties.Schema.STRING,
_('The type of the attribute.'),
default='string',
constraints=[alarm_base.BaseAlarm.QF_TYPE_VALS],
support_status=support.SupportStatus(version='8.0.0')
),
alarm_base.BaseAlarm.QF_OP: properties.Schema(
properties.Schema.STRING,
_('Comparison operator.'),
constraints=[alarm_base.BaseAlarm.QF_OP_VALS]
),
alarm_base.BaseAlarm.QF_VALUE: properties.Schema(
properties.Schema.STRING,
_('String value with which to compare.')
)
}
)
)
}
properties_schema.update(alarm_base.common_properties_schema)
def get_alarm_props(self, props):
"""Apply all relevant compatibility xforms."""
kwargs = self.actions_to_urls(props)
kwargs['type'] = self.alarm_type
if kwargs.get(self.METER_NAME) in alarm_base.NOVA_METERS:
prefix = 'user_metadata.'
else:
prefix = 'metering.'
rule = {}
for field in ['period', 'evaluation_periods', 'threshold',
'statistic', 'comparison_operator', 'meter_name']:
if field in kwargs:
rule[field] = kwargs[field]
del kwargs[field]
mmd = props.get(self.MATCHING_METADATA) or {}
query = props.get(self.QUERY) or []
# make sure the matching_metadata appears in the query like this:
# {field: metadata.$prefix.x, ...}
for m_k, m_v in six.iteritems(mmd):
key = 'metadata.%s' % prefix
if m_k.startswith('metadata.'):
m_k = m_k[len('metadata.'):]
if m_k.startswith('metering.') or m_k.startswith('user_metadata.'):
# check prefix
m_k = m_k.split('.', 1)[-1]
key = '%s%s' % (key, m_k)
# NOTE(prazumovsky): type of query value must be a string, but
# matching_metadata value type can not be a string, so we
# must convert value to a string type.
query.append(dict(field=key, op='eq', value=six.text_type(m_v)))
if self.MATCHING_METADATA in kwargs:
del kwargs[self.MATCHING_METADATA]
if self.QUERY in kwargs:
del kwargs[self.QUERY]
if query:
rule['query'] = query
kwargs['threshold_rule'] = rule
return kwargs
def handle_create(self):
props = self.get_alarm_props(self.properties)
props['name'] = self.physical_resource_name()
alarm = self.client().alarm.create(props)
self.resource_id_set(alarm['alarm_id'])
def handle_update(self, json_snippet, tmpl_diff, prop_diff):
if prop_diff:
new_props = json_snippet.properties(self.properties_schema,
self.context)
self.client().alarm.update(self.resource_id,
self.get_alarm_props(new_props))
def parse_live_resource_data(self, resource_properties, resource_data):
record_reality = {}
threshold_data = resource_data.get('threshold_rule').copy()
threshold_data.update(resource_data)
props_upd_allowed = (set(self.PROPERTIES +
alarm_base.COMMON_PROPERTIES) -
{self.METER_NAME, alarm_base.TIME_CONSTRAINTS} -
set(alarm_base.INTERNAL_PROPERTIES))
for key in props_upd_allowed:
record_reality.update({key: threshold_data.get(key)})
return record_reality
def handle_check(self):
self.client().alarm.get(self.resource_id)
class CombinationAlarm(none_resource.NoneResource):
"""A resource that implements combination of Aodh alarms.
This resource is now deleted from Aodh, so will directly inherit from
NoneResource (placeholder resource). For old resources (which not a
placeholder resource), still can be deleted through client. Any newly
created resources will be considered as placeholder resources like none
resource. We will schedule to delete it from heat resources list.
"""
default_client_name = 'aodh'
entity = 'alarm'
support_status = support.SupportStatus(
status=support.HIDDEN,
message=_('OS::Aodh::CombinationAlarm is deprecated and has been '
'removed from Aodh, use OS::Aodh::CompositeAlarm instead.'),
version='9.0.0',
previous_status=support.SupportStatus(
status=support.DEPRECATED,
version='7.0.0',
previous_status=support.SupportStatus(version='2014.1')
)
)
class EventAlarm(alarm_base.BaseAlarm):
"""A resource that implements event alarms.
Allows users to define alarms which can be evaluated based on events
passed from other OpenStack services. The events can be emitted when
the resources from other OpenStack services have been updated, created
or deleted, such as 'compute.instance.reboot.end',
'scheduler.select_destinations.end'.
"""
alarm_type = 'event'
support_status = support.SupportStatus(version='8.0.0')
PROPERTIES = (
EVENT_TYPE, QUERY
) = (
'event_type', 'query'
)
properties_schema = {
EVENT_TYPE: properties.Schema(
properties.Schema.STRING,
_('Event type to evaluate against. '
'If not specified will match all events.'),
update_allowed=True,
default='*'
),
QUERY: properties.Schema(
properties.Schema.LIST,
_('A list for filtering events. Query conditions used '
'to filter specific events when evaluating the alarm.'),
update_allowed=True,
schema=properties.Schema(
properties.Schema.MAP,
schema={
alarm_base.BaseAlarm.QF_FIELD: properties.Schema(
properties.Schema.STRING,
_('Name of attribute to compare.')
),
alarm_base.BaseAlarm.QF_TYPE: properties.Schema(
properties.Schema.STRING,
_('The type of the attribute.'),
default='string',
constraints=[alarm_base.BaseAlarm.QF_TYPE_VALS]
),
alarm_base.BaseAlarm.QF_OP: properties.Schema(
properties.Schema.STRING,
_('Comparison operator.'),
constraints=[alarm_base.BaseAlarm.QF_OP_VALS]
),
alarm_base.BaseAlarm.QF_VALUE: properties.Schema(
properties.Schema.STRING,
_('String value with which to compare.')
)
}
)
)
}
properties_schema.update(alarm_base.common_properties_schema)
def get_alarm_props(self, props):
"""Apply all relevant compatibility xforms."""
kwargs = self.actions_to_urls(props)
kwargs['type'] = self.alarm_type
rule = {}
for prop in (self.EVENT_TYPE, self.QUERY):
if prop in kwargs:
del kwargs[prop]
query = props.get(self.QUERY)
if query:
rule[self.QUERY] = query
event_type = props.get(self.EVENT_TYPE)
if event_type:
rule[self.EVENT_TYPE] = event_type
kwargs['event_rule'] = rule
return kwargs
def handle_create(self):
props = self.get_alarm_props(self.properties)
props['name'] = self.physical_resource_name()
alarm = self.client().alarm.create(props)
self.resource_id_set(alarm['alarm_id'])
def handle_update(self, json_snippet, tmpl_diff, prop_diff):
if prop_diff:
new_props = json_snippet.properties(self.properties_schema,
self.context)
self.client().alarm.update(self.resource_id,
self.get_alarm_props(new_props))
class LBMemberHealthAlarm(alarm_base.BaseAlarm):
"""A resource that implements a Loadbalancer Member Health Alarm.
Allows setting alarms based on the health of load balancer pool members,
where the health of a member is determined by the member reporting an
operating_status of ERROR beyond an initial grace period after creation
(120 seconds by default).
"""
alarm_type = "loadbalancer_member_health"
support_status = support.SupportStatus(version='13.0.0')
PROPERTIES = (
POOL, STACK, AUTOSCALING_GROUP_ID
) = (
"pool", "stack", "autoscaling_group_id"
)
RULE_PROPERTIES = (
POOL_ID, STACK_ID
) = (
"pool_id", "stack_id"
)
properties_schema = {
POOL: properties.Schema(
properties.Schema.STRING,
_("Name or ID of the loadbalancer pool for which the health of "
"each member will be evaluated."),
update_allowed=True,
required=True,
),
STACK: properties.Schema(
properties.Schema.STRING,
_("Name or ID of the root / top level Heat stack containing the "
"loadbalancer pool and members. An update will be triggered "
"on the root Stack if an unhealthy member is detected in the "
"loadbalancer pool."),
update_allowed=False,
required=True,
),
AUTOSCALING_GROUP_ID: properties.Schema(
properties.Schema.STRING,
_("ID of the Heat autoscaling group that contains the "
"loadbalancer members. Unhealthy members will be marked "
"as such before an update is triggered on the root stack."),
update_allowed=True,
required=True,
),
}
properties_schema.update(alarm_base.common_properties_schema)
def get_alarm_props(self, props):
"""Apply all relevant compatibility xforms."""
kwargs = self.actions_to_urls(props)
kwargs['type'] = self.alarm_type
for prop in (self.POOL, self.STACK, self.AUTOSCALING_GROUP_ID):
if prop in kwargs:
del kwargs[prop]
rule = {
self.POOL_ID: props[self.POOL],
self.STACK_ID: props[self.STACK],
self.AUTOSCALING_GROUP_ID: props[self.AUTOSCALING_GROUP_ID]
}
kwargs["loadbalancer_member_health_rule"] = rule
return kwargs
def translation_rules(self, properties):
translation_rules = [
translation.TranslationRule(
properties,
translation.TranslationRule.RESOLVE,
[self.POOL],
client_plugin=self.client_plugin('octavia'),
finder='get_pool'
),
]
return translation_rules
def handle_create(self):
props = self.get_alarm_props(self.properties)
props['name'] = self.physical_resource_name()
alarm = self.client().alarm.create(props)
self.resource_id_set(alarm['alarm_id'])
def handle_update(self, json_snippet, tmpl_diff, prop_diff):
if prop_diff:
new_props = json_snippet.properties(self.properties_schema,
self.context)
self.client().alarm.update(self.resource_id,
self.get_alarm_props(new_props))
def resource_mapping():
return {
'OS::Aodh::Alarm': AodhAlarm,
'OS::Aodh::CombinationAlarm': CombinationAlarm,
'OS::Aodh::EventAlarm': EventAlarm,
'OS::Aodh::LBMemberHealthAlarm': LBMemberHealthAlarm,
}