2022-04-03 11:04:27 +02:00
|
|
|
#!/usr/bin/python
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
2023-10-19 11:10:04 +02:00
|
|
|
# Copyright (c) 2015, Sebastian Kornehl <sebastian.kornehl@asideas.de>
|
|
|
|
# GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt)
|
|
|
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
2022-04-03 11:04:27 +02:00
|
|
|
|
|
|
|
from __future__ import absolute_import, division, print_function
|
|
|
|
__metaclass__ = type
|
|
|
|
|
|
|
|
|
|
|
|
DOCUMENTATION = '''
|
|
|
|
---
|
|
|
|
module: datadog_monitor
|
|
|
|
short_description: Manages Datadog monitors
|
|
|
|
description:
|
|
|
|
- Manages monitors within Datadog.
|
|
|
|
- Options as described on https://docs.datadoghq.com/api/.
|
|
|
|
author: Sebastian Kornehl (@skornehl)
|
|
|
|
requirements: [datadog]
|
2023-10-19 11:10:04 +02:00
|
|
|
extends_documentation_fragment:
|
|
|
|
- community.general.attributes
|
|
|
|
attributes:
|
|
|
|
check_mode:
|
|
|
|
support: none
|
|
|
|
diff_mode:
|
|
|
|
support: none
|
2022-04-03 11:04:27 +02:00
|
|
|
options:
|
|
|
|
api_key:
|
|
|
|
description:
|
|
|
|
- Your Datadog API key.
|
|
|
|
required: true
|
|
|
|
type: str
|
|
|
|
api_host:
|
|
|
|
description:
|
2023-10-19 11:10:04 +02:00
|
|
|
- The URL to the Datadog API. Default value is V(https://api.datadoghq.com).
|
|
|
|
- This value can also be set with the E(DATADOG_HOST) environment variable.
|
2022-04-03 11:04:27 +02:00
|
|
|
required: false
|
|
|
|
type: str
|
|
|
|
version_added: '0.2.0'
|
|
|
|
app_key:
|
|
|
|
description:
|
|
|
|
- Your Datadog app key.
|
|
|
|
required: true
|
|
|
|
type: str
|
|
|
|
state:
|
|
|
|
description:
|
|
|
|
- The designated state of the monitor.
|
|
|
|
required: true
|
|
|
|
choices: ['present', 'absent', 'mute', 'unmute']
|
|
|
|
type: str
|
|
|
|
tags:
|
|
|
|
description:
|
|
|
|
- A list of tags to associate with your monitor when creating or updating.
|
|
|
|
- This can help you categorize and filter monitors.
|
|
|
|
type: list
|
|
|
|
elements: str
|
|
|
|
type:
|
|
|
|
description:
|
|
|
|
- The type of the monitor.
|
2023-10-19 11:10:04 +02:00
|
|
|
- The types V(query alert), V(trace-analytics alert) and V(rum alert) were added in community.general 2.1.0.
|
|
|
|
- The type V(composite) was added in community.general 3.4.0.
|
|
|
|
- The type V(event-v2 alert) was added in community.general 4.8.0.
|
2022-04-03 11:04:27 +02:00
|
|
|
choices:
|
|
|
|
- metric alert
|
|
|
|
- service check
|
|
|
|
- event alert
|
2023-10-19 11:10:04 +02:00
|
|
|
- event-v2 alert
|
2022-04-03 11:04:27 +02:00
|
|
|
- process alert
|
|
|
|
- log alert
|
|
|
|
- query alert
|
|
|
|
- trace-analytics alert
|
|
|
|
- rum alert
|
|
|
|
- composite
|
|
|
|
type: str
|
|
|
|
query:
|
|
|
|
description:
|
|
|
|
- The monitor query to notify on.
|
|
|
|
- Syntax varies depending on what type of monitor you are creating.
|
|
|
|
type: str
|
|
|
|
name:
|
|
|
|
description:
|
|
|
|
- The name of the alert.
|
|
|
|
required: true
|
|
|
|
type: str
|
|
|
|
notification_message:
|
|
|
|
description:
|
|
|
|
- A message to include with notifications for this monitor.
|
|
|
|
- Email notifications can be sent to specific users by using the same '@username' notation as events.
|
|
|
|
- Monitor message template variables can be accessed by using double square brackets, i.e '[[' and ']]'.
|
|
|
|
type: str
|
|
|
|
silenced:
|
|
|
|
type: dict
|
|
|
|
description:
|
|
|
|
- Dictionary of scopes to silence, with timestamps or None.
|
|
|
|
- Each scope will be muted until the given POSIX timestamp or forever if the value is None.
|
|
|
|
notify_no_data:
|
|
|
|
description:
|
|
|
|
- Whether this monitor will notify when data stops reporting.
|
|
|
|
type: bool
|
2023-10-19 11:10:04 +02:00
|
|
|
default: false
|
2022-04-03 11:04:27 +02:00
|
|
|
no_data_timeframe:
|
|
|
|
description:
|
|
|
|
- The number of minutes before a monitor will notify when data stops reporting.
|
|
|
|
- Must be at least 2x the monitor timeframe for metric alerts or 2 minutes for service checks.
|
|
|
|
- If not specified, it defaults to 2x timeframe for metric, 2 minutes for service.
|
|
|
|
type: str
|
|
|
|
timeout_h:
|
|
|
|
description:
|
|
|
|
- The number of hours of the monitor not reporting data before it will automatically resolve from a triggered state.
|
|
|
|
type: str
|
|
|
|
renotify_interval:
|
|
|
|
description:
|
|
|
|
- The number of minutes after the last notification before a monitor will re-notify on the current status.
|
|
|
|
- It will only re-notify if it is not resolved.
|
|
|
|
type: str
|
|
|
|
escalation_message:
|
|
|
|
description:
|
|
|
|
- A message to include with a re-notification. Supports the '@username' notification we allow elsewhere.
|
2023-10-19 11:10:04 +02:00
|
|
|
- Not applicable if O(renotify_interval=none).
|
2022-04-03 11:04:27 +02:00
|
|
|
type: str
|
|
|
|
notify_audit:
|
|
|
|
description:
|
|
|
|
- Whether tagged users will be notified on changes to this monitor.
|
|
|
|
type: bool
|
2023-10-19 11:10:04 +02:00
|
|
|
default: false
|
2022-04-03 11:04:27 +02:00
|
|
|
thresholds:
|
|
|
|
type: dict
|
|
|
|
description:
|
|
|
|
- A dictionary of thresholds by status.
|
|
|
|
- Only available for service checks and metric alerts.
|
|
|
|
- Because each of them can have multiple thresholds, we do not define them directly in the query.
|
2023-10-19 11:10:04 +02:00
|
|
|
- "If not specified, it defaults to: V({'ok': 1, 'critical': 1, 'warning': 1})."
|
2022-04-03 11:04:27 +02:00
|
|
|
locked:
|
|
|
|
description:
|
|
|
|
- Whether changes to this monitor should be restricted to the creator or admins.
|
|
|
|
type: bool
|
2023-10-19 11:10:04 +02:00
|
|
|
default: false
|
2022-04-03 11:04:27 +02:00
|
|
|
require_full_window:
|
|
|
|
description:
|
|
|
|
- Whether this monitor needs a full window of data before it gets evaluated.
|
|
|
|
- We highly recommend you set this to False for sparse metrics, otherwise some evaluations will be skipped.
|
|
|
|
type: bool
|
|
|
|
new_host_delay:
|
|
|
|
description:
|
|
|
|
- A positive integer representing the number of seconds to wait before evaluating the monitor for new hosts.
|
|
|
|
- This gives the host time to fully initialize.
|
|
|
|
type: str
|
|
|
|
evaluation_delay:
|
|
|
|
description:
|
|
|
|
- Time to delay evaluation (in seconds).
|
|
|
|
- Effective for sparse values.
|
|
|
|
type: str
|
|
|
|
id:
|
|
|
|
description:
|
|
|
|
- The ID of the alert.
|
|
|
|
- If set, will be used instead of the name to locate the alert.
|
|
|
|
type: str
|
|
|
|
include_tags:
|
|
|
|
description:
|
|
|
|
- Whether notifications from this monitor automatically inserts its triggering tags into the title.
|
|
|
|
type: bool
|
2023-10-19 11:10:04 +02:00
|
|
|
default: true
|
2022-04-03 11:04:27 +02:00
|
|
|
version_added: 1.3.0
|
|
|
|
priority:
|
|
|
|
description:
|
|
|
|
- Integer from 1 (high) to 5 (low) indicating alert severity.
|
|
|
|
type: int
|
|
|
|
version_added: 4.6.0
|
2023-10-19 11:10:04 +02:00
|
|
|
notification_preset_name:
|
|
|
|
description:
|
|
|
|
- Toggles the display of additional content sent in the monitor notification.
|
|
|
|
choices:
|
|
|
|
- show_all
|
|
|
|
- hide_query
|
|
|
|
- hide_handles
|
|
|
|
- hide_all
|
|
|
|
type: str
|
|
|
|
version_added: 7.1.0
|
|
|
|
renotify_occurrences:
|
|
|
|
description:
|
|
|
|
- The number of times re-notification messages should be sent on the current status at the provided re-notification interval.
|
|
|
|
type: int
|
|
|
|
version_added: 7.1.0
|
|
|
|
renotify_statuses:
|
|
|
|
description:
|
|
|
|
- The types of monitor statuses for which re-notification messages are sent.
|
|
|
|
choices:
|
|
|
|
- alert
|
|
|
|
- warn
|
|
|
|
- no data
|
|
|
|
type: list
|
|
|
|
elements: str
|
|
|
|
version_added: 7.1.0
|
|
|
|
|
2022-04-03 11:04:27 +02:00
|
|
|
'''
|
|
|
|
|
|
|
|
EXAMPLES = '''
|
|
|
|
- name: Create a metric monitor
|
|
|
|
community.general.datadog_monitor:
|
|
|
|
type: "metric alert"
|
|
|
|
name: "Test monitor"
|
|
|
|
state: "present"
|
2023-10-19 11:10:04 +02:00
|
|
|
renotify_interval: 30
|
|
|
|
renotify_occurrences: 1
|
|
|
|
renotify_statuses: ["warn"]
|
|
|
|
notification_preset_name: "show_all"
|
2022-04-03 11:04:27 +02:00
|
|
|
query: "datadog.agent.up.over('host:host1').last(2).count_by_status()"
|
|
|
|
notification_message: "Host [[host.name]] with IP [[host.ip]] is failing to report to datadog."
|
|
|
|
api_key: "9775a026f1ca7d1c6c5af9d94d9595a4"
|
|
|
|
app_key: "87ce4a24b5553d2e482ea8a8500e71b8ad4554ff"
|
|
|
|
|
|
|
|
- name: Deletes a monitor
|
|
|
|
community.general.datadog_monitor:
|
|
|
|
name: "Test monitor"
|
|
|
|
state: "absent"
|
|
|
|
api_key: "9775a026f1ca7d1c6c5af9d94d9595a4"
|
|
|
|
app_key: "87ce4a24b5553d2e482ea8a8500e71b8ad4554ff"
|
|
|
|
|
|
|
|
- name: Mutes a monitor
|
|
|
|
community.general.datadog_monitor:
|
|
|
|
name: "Test monitor"
|
|
|
|
state: "mute"
|
|
|
|
silenced: '{"*":None}'
|
|
|
|
api_key: "9775a026f1ca7d1c6c5af9d94d9595a4"
|
|
|
|
app_key: "87ce4a24b5553d2e482ea8a8500e71b8ad4554ff"
|
|
|
|
|
|
|
|
- name: Unmutes a monitor
|
|
|
|
community.general.datadog_monitor:
|
|
|
|
name: "Test monitor"
|
|
|
|
state: "unmute"
|
|
|
|
api_key: "9775a026f1ca7d1c6c5af9d94d9595a4"
|
|
|
|
app_key: "87ce4a24b5553d2e482ea8a8500e71b8ad4554ff"
|
|
|
|
|
|
|
|
- name: Use datadoghq.eu platform instead of datadoghq.com
|
|
|
|
community.general.datadog_monitor:
|
|
|
|
name: "Test monitor"
|
|
|
|
state: "absent"
|
|
|
|
api_host: https://api.datadoghq.eu
|
|
|
|
api_key: "9775a026f1ca7d1c6c5af9d94d9595a4"
|
|
|
|
app_key: "87ce4a24b5553d2e482ea8a8500e71b8ad4554ff"
|
|
|
|
'''
|
|
|
|
import traceback
|
|
|
|
|
|
|
|
# Import Datadog
|
|
|
|
DATADOG_IMP_ERR = None
|
|
|
|
try:
|
|
|
|
from datadog import initialize, api
|
|
|
|
HAS_DATADOG = True
|
|
|
|
except Exception:
|
|
|
|
DATADOG_IMP_ERR = traceback.format_exc()
|
|
|
|
HAS_DATADOG = False
|
|
|
|
|
|
|
|
from ansible.module_utils.basic import AnsibleModule, missing_required_lib
|
|
|
|
from ansible.module_utils.common.text.converters import to_native
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
module = AnsibleModule(
|
|
|
|
argument_spec=dict(
|
|
|
|
api_key=dict(required=True, no_log=True),
|
|
|
|
api_host=dict(),
|
|
|
|
app_key=dict(required=True, no_log=True),
|
|
|
|
state=dict(required=True, choices=['present', 'absent', 'mute', 'unmute']),
|
2023-10-19 11:10:04 +02:00
|
|
|
type=dict(choices=['metric alert', 'service check', 'event alert', 'event-v2 alert', 'process alert',
|
2022-04-03 11:04:27 +02:00
|
|
|
'log alert', 'query alert', 'trace-analytics alert',
|
|
|
|
'rum alert', 'composite']),
|
|
|
|
name=dict(required=True),
|
|
|
|
query=dict(),
|
|
|
|
notification_message=dict(no_log=True),
|
|
|
|
silenced=dict(type='dict'),
|
|
|
|
notify_no_data=dict(default=False, type='bool'),
|
|
|
|
no_data_timeframe=dict(),
|
|
|
|
timeout_h=dict(),
|
|
|
|
renotify_interval=dict(),
|
|
|
|
escalation_message=dict(),
|
|
|
|
notify_audit=dict(default=False, type='bool'),
|
|
|
|
thresholds=dict(type='dict', default=None),
|
|
|
|
tags=dict(type='list', elements='str', default=None),
|
|
|
|
locked=dict(default=False, type='bool'),
|
|
|
|
require_full_window=dict(type='bool'),
|
|
|
|
new_host_delay=dict(),
|
|
|
|
evaluation_delay=dict(),
|
|
|
|
id=dict(),
|
|
|
|
include_tags=dict(required=False, default=True, type='bool'),
|
|
|
|
priority=dict(type='int'),
|
2023-10-19 11:10:04 +02:00
|
|
|
notification_preset_name=dict(choices=['show_all', 'hide_query', 'hide_handles', 'hide_all']),
|
|
|
|
renotify_occurrences=dict(type='int'),
|
|
|
|
renotify_statuses=dict(type='list', elements='str', choices=['alert', 'warn', 'no data']),
|
2022-04-03 11:04:27 +02:00
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
# Prepare Datadog
|
|
|
|
if not HAS_DATADOG:
|
|
|
|
module.fail_json(msg=missing_required_lib('datadogpy'), exception=DATADOG_IMP_ERR)
|
|
|
|
|
|
|
|
options = {
|
|
|
|
'api_key': module.params['api_key'],
|
|
|
|
'api_host': module.params['api_host'],
|
|
|
|
'app_key': module.params['app_key']
|
|
|
|
}
|
|
|
|
|
|
|
|
initialize(**options)
|
|
|
|
|
|
|
|
# Check if api_key and app_key is correct or not
|
|
|
|
# if not, then fail here.
|
|
|
|
response = api.Monitor.get_all()
|
|
|
|
if isinstance(response, dict):
|
|
|
|
msg = response.get('errors', None)
|
|
|
|
if msg:
|
|
|
|
module.fail_json(msg="Failed to connect Datadog server using given app_key and api_key : {0}".format(msg[0]))
|
|
|
|
|
|
|
|
if module.params['state'] == 'present':
|
|
|
|
install_monitor(module)
|
|
|
|
elif module.params['state'] == 'absent':
|
|
|
|
delete_monitor(module)
|
|
|
|
elif module.params['state'] == 'mute':
|
|
|
|
mute_monitor(module)
|
|
|
|
elif module.params['state'] == 'unmute':
|
|
|
|
unmute_monitor(module)
|
|
|
|
|
|
|
|
|
|
|
|
def _fix_template_vars(message):
|
|
|
|
if message:
|
|
|
|
return message.replace('[[', '{{').replace(']]', '}}')
|
|
|
|
return message
|
|
|
|
|
|
|
|
|
|
|
|
def _get_monitor(module):
|
|
|
|
if module.params['id'] is not None:
|
|
|
|
monitor = api.Monitor.get(module.params['id'])
|
|
|
|
if 'errors' in monitor:
|
|
|
|
module.fail_json(msg="Failed to retrieve monitor with id %s, errors are %s" % (module.params['id'], str(monitor['errors'])))
|
|
|
|
return monitor
|
|
|
|
else:
|
|
|
|
monitors = api.Monitor.get_all()
|
|
|
|
for monitor in monitors:
|
|
|
|
if monitor['name'] == _fix_template_vars(module.params['name']):
|
|
|
|
return monitor
|
|
|
|
return {}
|
|
|
|
|
|
|
|
|
|
|
|
def _post_monitor(module, options):
|
|
|
|
try:
|
|
|
|
kwargs = dict(type=module.params['type'], query=module.params['query'],
|
|
|
|
name=_fix_template_vars(module.params['name']),
|
|
|
|
message=_fix_template_vars(module.params['notification_message']),
|
|
|
|
escalation_message=_fix_template_vars(module.params['escalation_message']),
|
|
|
|
priority=module.params['priority'],
|
|
|
|
options=options)
|
|
|
|
if module.params['tags'] is not None:
|
|
|
|
kwargs['tags'] = module.params['tags']
|
|
|
|
msg = api.Monitor.create(**kwargs)
|
|
|
|
if 'errors' in msg:
|
|
|
|
module.fail_json(msg=str(msg['errors']))
|
|
|
|
else:
|
|
|
|
module.exit_json(changed=True, msg=msg)
|
|
|
|
except Exception as e:
|
|
|
|
module.fail_json(msg=to_native(e), exception=traceback.format_exc())
|
|
|
|
|
|
|
|
|
|
|
|
def _equal_dicts(a, b, ignore_keys):
|
|
|
|
ka = set(a).difference(ignore_keys)
|
|
|
|
kb = set(b).difference(ignore_keys)
|
|
|
|
return ka == kb and all(a[k] == b[k] for k in ka)
|
|
|
|
|
|
|
|
|
|
|
|
def _update_monitor(module, monitor, options):
|
|
|
|
try:
|
|
|
|
kwargs = dict(id=monitor['id'], query=module.params['query'],
|
|
|
|
name=_fix_template_vars(module.params['name']),
|
|
|
|
message=_fix_template_vars(module.params['notification_message']),
|
|
|
|
escalation_message=_fix_template_vars(module.params['escalation_message']),
|
|
|
|
priority=module.params['priority'],
|
|
|
|
options=options)
|
|
|
|
if module.params['tags'] is not None:
|
|
|
|
kwargs['tags'] = module.params['tags']
|
|
|
|
msg = api.Monitor.update(**kwargs)
|
|
|
|
|
|
|
|
if 'errors' in msg:
|
|
|
|
module.fail_json(msg=str(msg['errors']))
|
|
|
|
elif _equal_dicts(msg, monitor, ['creator', 'overall_state', 'modified', 'matching_downtimes', 'overall_state_modified']):
|
|
|
|
module.exit_json(changed=False, msg=msg)
|
|
|
|
else:
|
|
|
|
module.exit_json(changed=True, msg=msg)
|
|
|
|
except Exception as e:
|
|
|
|
module.fail_json(msg=to_native(e), exception=traceback.format_exc())
|
|
|
|
|
|
|
|
|
|
|
|
def install_monitor(module):
|
|
|
|
options = {
|
|
|
|
"silenced": module.params['silenced'],
|
|
|
|
"notify_no_data": module.boolean(module.params['notify_no_data']),
|
|
|
|
"no_data_timeframe": module.params['no_data_timeframe'],
|
|
|
|
"timeout_h": module.params['timeout_h'],
|
|
|
|
"renotify_interval": module.params['renotify_interval'],
|
|
|
|
"escalation_message": module.params['escalation_message'],
|
|
|
|
"notify_audit": module.boolean(module.params['notify_audit']),
|
|
|
|
"locked": module.boolean(module.params['locked']),
|
|
|
|
"require_full_window": module.params['require_full_window'],
|
|
|
|
"new_host_delay": module.params['new_host_delay'],
|
|
|
|
"evaluation_delay": module.params['evaluation_delay'],
|
|
|
|
"include_tags": module.params['include_tags'],
|
2023-10-19 11:10:04 +02:00
|
|
|
"notification_preset_name": module.params['notification_preset_name'],
|
|
|
|
"renotify_occurrences": module.params['renotify_occurrences'],
|
|
|
|
"renotify_statuses": module.params['renotify_statuses'],
|
2022-04-03 11:04:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if module.params['type'] == "service check":
|
|
|
|
options["thresholds"] = module.params['thresholds'] or {'ok': 1, 'critical': 1, 'warning': 1}
|
|
|
|
if module.params['type'] in ["metric alert", "log alert", "query alert", "trace-analytics alert", "rum alert"] and module.params['thresholds'] is not None:
|
|
|
|
options["thresholds"] = module.params['thresholds']
|
|
|
|
|
|
|
|
monitor = _get_monitor(module)
|
|
|
|
if not monitor:
|
|
|
|
_post_monitor(module, options)
|
|
|
|
else:
|
|
|
|
_update_monitor(module, monitor, options)
|
|
|
|
|
|
|
|
|
|
|
|
def delete_monitor(module):
|
|
|
|
monitor = _get_monitor(module)
|
|
|
|
if not monitor:
|
|
|
|
module.exit_json(changed=False)
|
|
|
|
try:
|
|
|
|
msg = api.Monitor.delete(monitor['id'])
|
|
|
|
module.exit_json(changed=True, msg=msg)
|
|
|
|
except Exception as e:
|
|
|
|
module.fail_json(msg=to_native(e), exception=traceback.format_exc())
|
|
|
|
|
|
|
|
|
|
|
|
def mute_monitor(module):
|
|
|
|
monitor = _get_monitor(module)
|
|
|
|
if not monitor:
|
|
|
|
module.fail_json(msg="Monitor %s not found!" % module.params['name'])
|
|
|
|
elif monitor['options']['silenced']:
|
|
|
|
module.fail_json(msg="Monitor is already muted. Datadog does not allow to modify muted alerts, consider unmuting it first.")
|
|
|
|
elif (module.params['silenced'] is not None and len(set(monitor['options']['silenced']) ^ set(module.params['silenced'])) == 0):
|
|
|
|
module.exit_json(changed=False)
|
|
|
|
try:
|
|
|
|
if module.params['silenced'] is None or module.params['silenced'] == "":
|
|
|
|
msg = api.Monitor.mute(id=monitor['id'])
|
|
|
|
else:
|
|
|
|
msg = api.Monitor.mute(id=monitor['id'], silenced=module.params['silenced'])
|
|
|
|
module.exit_json(changed=True, msg=msg)
|
|
|
|
except Exception as e:
|
|
|
|
module.fail_json(msg=to_native(e), exception=traceback.format_exc())
|
|
|
|
|
|
|
|
|
|
|
|
def unmute_monitor(module):
|
|
|
|
monitor = _get_monitor(module)
|
|
|
|
if not monitor:
|
|
|
|
module.fail_json(msg="Monitor %s not found!" % module.params['name'])
|
|
|
|
elif not monitor['options']['silenced']:
|
|
|
|
module.exit_json(changed=False)
|
|
|
|
try:
|
|
|
|
msg = api.Monitor.unmute(monitor['id'])
|
|
|
|
module.exit_json(changed=True, msg=msg)
|
|
|
|
except Exception as e:
|
|
|
|
module.fail_json(msg=to_native(e), exception=traceback.format_exc())
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|