mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-24 13:04:00 -07:00 
			
		
		
		
	
		
			Some checks are pending
		
		
	
	EOL CI / EOL Sanity (Ⓐ2.17) (push) Waiting to run
				
			EOL CI / EOL Units (Ⓐ2.17+py3.10) (push) Waiting to run
				
			EOL CI / EOL Units (Ⓐ2.17+py3.12) (push) Waiting to run
				
			EOL CI / EOL Units (Ⓐ2.17+py3.7) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+alpine319+py:azp/posix/1/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+alpine319+py:azp/posix/2/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+alpine319+py:azp/posix/3/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+fedora39+py:azp/posix/1/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+fedora39+py:azp/posix/2/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+fedora39+py:azp/posix/3/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+ubuntu2004+py:azp/posix/1/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+ubuntu2004+py:azp/posix/2/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+ubuntu2004+py:azp/posix/3/) (push) Waiting to run
				
			nox / Run extra sanity tests (push) Waiting to run
				
			* Adjust all __future__ imports: for i in $(grep -REl "__future__.*absolute_import" plugins/ tests/); do sed -e 's/from __future__ import .*/from __future__ import annotations/g' -i $i; done * Remove all UTF-8 encoding specifications for Python source files: for i in $(grep -REl '[-][*]- coding: utf-8 -[*]-' plugins/ tests/); do sed -e '/^# -\*- coding: utf-8 -\*-/d' -i $i; done * Remove __metaclass__ = type: for i in $(grep -REl '__metaclass__ = type' plugins/ tests/); do sed -e '/^__metaclass__ = type/d' -i $i; done
		
			
				
	
	
		
			462 lines
		
	
	
	
		
			16 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			462 lines
		
	
	
	
		
			16 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| #!/usr/bin/python
 | |
| 
 | |
| # 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
 | |
| 
 | |
| from __future__ import annotations
 | |
| 
 | |
| 
 | |
| DOCUMENTATION = r"""
 | |
| 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]
 | |
| extends_documentation_fragment:
 | |
|   - community.general.attributes
 | |
| attributes:
 | |
|   check_mode:
 | |
|     support: none
 | |
|   diff_mode:
 | |
|     support: none
 | |
| options:
 | |
|   api_key:
 | |
|     description:
 | |
|       - Your Datadog API key.
 | |
|     required: true
 | |
|     type: str
 | |
|   api_host:
 | |
|     description:
 | |
|       - 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.
 | |
|     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.
 | |
|       - 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.
 | |
|     choices:
 | |
|       - metric alert
 | |
|       - service check
 | |
|       - event alert
 | |
|       - event-v2 alert
 | |
|       - 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, in other words C([[) and C(]]).
 | |
|     type: str
 | |
|   silenced:
 | |
|     type: dict
 | |
|     description:
 | |
|       - Dictionary of scopes to silence, with timestamps or None.
 | |
|       - Each scope is muted until the given POSIX timestamp or forever if the value is V(None).
 | |
|   notify_no_data:
 | |
|     description:
 | |
|       - Whether this monitor notifies when data stops reporting.
 | |
|     type: bool
 | |
|     default: false
 | |
|   no_data_timeframe:
 | |
|     description:
 | |
|       - The number of minutes before a monitor notifies 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 automatically resolves from a triggered state.
 | |
|     type: str
 | |
|   renotify_interval:
 | |
|     description:
 | |
|       - The number of minutes after the last notification before a monitor re-notifies on the current status.
 | |
|       - It only re-notifies 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.
 | |
|       - Not applicable if O(renotify_interval=none).
 | |
|     type: str
 | |
|   notify_audit:
 | |
|     description:
 | |
|       - Whether tagged users are notified on changes to this monitor.
 | |
|     type: bool
 | |
|     default: false
 | |
|   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.
 | |
|       - "If not specified, it defaults to: V({'ok': 1, 'critical': 1, 'warning': 1})."
 | |
|   locked:
 | |
|     description:
 | |
|       - Whether changes to this monitor should be restricted to the creator or admins.
 | |
|     type: bool
 | |
|     default: false
 | |
|   require_full_window:
 | |
|     description:
 | |
|       - Whether this monitor needs a full window of data before it gets evaluated.
 | |
|       - We highly recommend you set this to V(false) for sparse metrics, otherwise some evaluations are 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, it is used instead of O(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
 | |
|     default: true
 | |
|     version_added: 1.3.0
 | |
|   priority:
 | |
|     description:
 | |
|       - Integer from V(1) (high) to V(5) (low) indicating alert severity.
 | |
|     type: int
 | |
|     version_added: 4.6.0
 | |
|   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
 | |
| """
 | |
| 
 | |
| EXAMPLES = r"""
 | |
| - name: Create a metric monitor
 | |
|   community.general.datadog_monitor:
 | |
|     type: "metric alert"
 | |
|     name: "Test monitor"
 | |
|     state: "present"
 | |
|     renotify_interval: 30
 | |
|     renotify_occurrences: 1
 | |
|     renotify_statuses: ["warn"]
 | |
|     notification_preset_name: "show_all"
 | |
|     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']),
 | |
|             type=dict(choices=['metric alert', 'service check', 'event alert', 'event-v2 alert', 'process alert',
 | |
|                                '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'),
 | |
|             tags=dict(type='list', elements='str'),
 | |
|             locked=dict(default=False, type='bool'),
 | |
|             require_full_window=dict(type='bool'),
 | |
|             new_host_delay=dict(),
 | |
|             evaluation_delay=dict(),
 | |
|             id=dict(),
 | |
|             include_tags=dict(default=True, type='bool'),
 | |
|             priority=dict(type='int'),
 | |
|             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']),
 | |
|         )
 | |
|     )
 | |
| 
 | |
|     # 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'],
 | |
|         "notification_preset_name": module.params['notification_preset_name'],
 | |
|         "renotify_occurrences": module.params['renotify_occurrences'],
 | |
|         "renotify_statuses": module.params['renotify_statuses'],
 | |
|     }
 | |
| 
 | |
|     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()
 |