blob: 147335b7a884480a0c42ae819fd013a779c57cf3 [file] [log] [blame]
Éric Lemoine6a1abe92016-11-03 14:02:43 +00001# -*- coding: utf-8 -*-
2
3import re
4
5_valid_dimension_re = re.compile(r'^[a-z0-9_/-]+$')
6_disallowed_dimensions = ('name', 'value', 'hostname', 'member',
7 'no_alerting', 'tag_fields')
8
9
Éric Lemoine7dc12132016-11-07 16:25:09 +000010def alarm_message_matcher(alarm, triggers):
Éric Lemoine6a1abe92016-11-03 14:02:43 +000011 """
12 Return an Heka message matcher expression for a given alarm and a
13 dict of triggers.
14
15 For example the function may return this:
16
17 Fields[name] == 'cpu_idle' || Fields[name] = 'cpu_wait'
18 """
19 matchers = set()
20 for trigger_name in alarm.get('triggers', []):
21 trigger = triggers.get(trigger_name)
22 if trigger and trigger.get('enabled', True):
23 for rule in trigger.get('rules', []):
24 matcher = "Fields[name] == '{}'".format(rule['metric'])
25 matchers.add(matcher)
26 return ' || '.join(matchers)
27
28
Swann Croiseteed005a2016-11-10 15:37:53 +010029def alarm_activate_alerting(alerting):
30 return 'true' if alerting in ['enabled', 'enabled_with_notification'] else 'false'
31
32
33def alarm_enable_notification(alerting):
34 return 'true' if alerting == 'enabled_with_notification' else 'false'
35
36
Éric Lemoine7dc12132016-11-07 16:25:09 +000037def alarm_cluster_message_matcher(alarm_cluster):
38 """
39 Return an Heka message matcher expression for a given alarm cluster.
40
41 For example the function may return this:
42
43 Fields[service] == 'rabbitmq-cluster'
44 """
45 matchers = set()
46 match_items = alarm_cluster.get('match', {}).items()
47 for match_name, match_value in match_items:
48 matcher = "Fields[{}] == '{}'".format(match_name, match_value)
49 matchers.add(matcher)
50 match_items = alarm_cluster.get('match_re', {}).items()
51 for match_name, match_value in match_items:
52 matcher = "Fields[{}] =~ /{}/".format(match_name, match_value)
53 matchers.add(matcher)
54 return ' && '.join(matchers)
55
56
Éric Lemoinefc2ae372016-11-08 13:55:03 +000057def dimensions(alarm_or_alarm_cluster):
Éric Lemoine6a1abe92016-11-03 14:02:43 +000058 """
59 Return a dict alarm dimensions. Each dimension is validated, and an
60 Exception is raised if a dimension is invalid.
61
62 Valid characters are a-z, 0-9, _, - and /.
63 """
Éric Lemoinefc2ae372016-11-08 13:55:03 +000064 dimensions = alarm_or_alarm_cluster.get('dimension', {})
Éric Lemoine6a1abe92016-11-03 14:02:43 +000065 for name, value in dimensions.items():
66 if name in _disallowed_dimensions:
67 raise Exception(
68 '{} is not allowed as a dimension name'.format(name))
69 if not _valid_dimension_re.match(name):
70 raise Exception(
71 'Dimension name {} includes disallowed chars'.format(name))
72 if not _valid_dimension_re.match(value):
73 raise Exception(
74 'Dimension value {} includes disallowed chars'.format(value))
75 return dimensions
Éric Lemoine74f7bd32016-11-15 13:18:33 +000076
77
78def grains_for_mine(grains):
79 """
80 Return grains that need to be sent to Salt Mine. Only the alarm
81 and alarm cluster data is to be sent to Mine.
82 """
83 filtered_grains = {}
84 for service_name, service_data in grains.items():
85 alarm = service_data.get('alarm')
86 if alarm:
87 filtered_grains[service_name] = {'alarm': alarm}
Éric Lemoined26770f2016-11-16 13:01:59 +000088 trigger = service_data.get('trigger')
89 if trigger:
90 if service_name in filtered_grains:
91 filtered_grains[service_name].update(
92 {'trigger': trigger})
93 else:
94 filtered_grains[service_name] = {'trigger': trigger}
Éric Lemoine74f7bd32016-11-15 13:18:33 +000095 alarm_cluster = service_data.get('alarm_cluster')
96 if alarm_cluster:
97 if service_name in filtered_grains:
98 filtered_grains[service_name].update(
99 {'alarm_cluster': alarm_cluster})
100 else:
101 filtered_grains[service_name] = \
102 {'alarm_cluster': alarm_cluster}
103 return filtered_grains