From 4ae436a8cc73111586d7a40a2b5dab719992a4e5 Mon Sep 17 00:00:00 2001 From: "patchback[bot]" <45432694+patchback[bot]@users.noreply.github.com> Date: Tue, 30 Mar 2021 12:37:52 +0200 Subject: [PATCH] Callback plugin: Azure Log Analytics (#2091) (#2133) * adding plugins/callback/loganalytics.py * * fixed sanity check issues * adjusted documentation and license sections * added changelogs fragment * * added unit test * documentation updated * updated changelogs * further docuement update * minor fixes * updated unittest * suggested updates from community * remove AnsibleError section (cherry picked from commit 19db6f24f7e939dd6f46a4cecf2930b3d9d6fd49) Co-authored-by: zhcli <49675498+zhcli@users.noreply.github.com> --- plugins/callback/loganalytics.py | 234 ++++++++++++++++++ .../plugins/callback/test_loganalytics.py | 64 +++++ 2 files changed, 298 insertions(+) create mode 100644 plugins/callback/loganalytics.py create mode 100644 tests/unit/plugins/callback/test_loganalytics.py diff --git a/plugins/callback/loganalytics.py b/plugins/callback/loganalytics.py new file mode 100644 index 0000000000..507d6fccd9 --- /dev/null +++ b/plugins/callback/loganalytics.py @@ -0,0 +1,234 @@ +# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +from __future__ import (absolute_import, division, print_function) +__metaclass__ = type + +DOCUMENTATION = ''' + callback: loganalytics + type: aggregate + short_description: Posts task results to Azure Log Analytics + author: "Cyrus Li (@zhcli) " + description: + - This callback plugin will post task results in JSON formatted to an Azure Log Analytics workspace. + - Credits to authors of splunk callback plugin. + version_added: "2.4.0" + requirements: + - Whitelisting this callback plugin. + - An Azure log analytics work space has been established. + options: + workspace_id: + description: Workspace ID of the Azure log analytics workspace. + required: true + env: + - name: WORKSPACE_ID + ini: + - section: callback_loganalytics + key: workspace_id + shared_key: + description: Shared key to connect to Azure log analytics workspace. + required: true + env: + - name: WORKSPACE_SHARED_KEY + ini: + - section: callback_loganalytics + key: shared_key +''' + +EXAMPLES = ''' +examples: | + Whitelist the plugin in ansible.cfg: + [defaults] + callback_whitelist = community.general.loganalytics + Set the environment variable: + export WORKSPACE_ID=01234567-0123-0123-0123-01234567890a + export WORKSPACE_SHARED_KEY=dZD0kCbKl3ehZG6LHFMuhtE0yHiFCmetzFMc2u+roXIUQuatqU924SsAAAAPemhjbGlAemhjbGktTUJQAQIDBA== + Or configure the plugin in ansible.cfg in the callback_loganalytics block: + [callback_loganalytics] + workspace_id = 01234567-0123-0123-0123-01234567890a + shared_key = dZD0kCbKl3ehZG6LHFMuhtE0yHiFCmetzFMc2u+roXIUQuatqU924SsAAAAPemhjbGlAemhjbGktTUJQAQIDBA== +''' + +import hashlib +import hmac +import base64 +import logging +import json +import uuid +import socket +import getpass + +from datetime import datetime +from os.path import basename + +from ansible.module_utils.urls import open_url +from ansible.parsing.ajson import AnsibleJSONEncoder +from ansible.plugins.callback import CallbackBase + + +class AzureLogAnalyticsSource(object): + def __init__(self): + self.ansible_check_mode = False + self.ansible_playbook = "" + self.ansible_version = "" + self.session = str(uuid.uuid4()) + self.host = socket.gethostname() + self.user = getpass.getuser() + self.extra_vars = "" + + def __build_signature(self, date, workspace_id, shared_key, content_length): + # Build authorisation signature for Azure log analytics API call + sigs = "POST\n{0}\napplication/json\nx-ms-date:{1}\n/api/logs".format( + str(content_length), date) + utf8_sigs = sigs.encode('utf-8') + decoded_shared_key = base64.b64decode(shared_key) + hmac_sha256_sigs = hmac.new( + decoded_shared_key, utf8_sigs, digestmod=hashlib.sha256).digest() + encoded_hash = base64.b64encode(hmac_sha256_sigs).decode('utf-8') + signature = "SharedKey {0}:{1}".format(workspace_id, encoded_hash) + return signature + + def __build_workspace_url(self, workspace_id): + return "https://{0}.ods.opinsights.azure.com/api/logs?api-version=2016-04-01".format(workspace_id) + + def __rfc1123date(self): + return datetime.utcnow().strftime('%a, %d %b %Y %H:%M:%S GMT') + + def send_event(self, workspace_id, shared_key, state, result, runtime): + if result._task_fields['args'].get('_ansible_check_mode') is True: + self.ansible_check_mode = True + + if result._task_fields['args'].get('_ansible_version'): + self.ansible_version = \ + result._task_fields['args'].get('_ansible_version') + + if result._task._role: + ansible_role = str(result._task._role) + else: + ansible_role = None + + data = {} + data['uuid'] = result._task._uuid + data['session'] = self.session + data['status'] = state + data['timestamp'] = self.__rfc1123date() + data['host'] = self.host + data['user'] = self.user + data['runtime'] = runtime + data['ansible_version'] = self.ansible_version + data['ansible_check_mode'] = self.ansible_check_mode + data['ansible_host'] = result._host.name + data['ansible_playbook'] = self.ansible_playbook + data['ansible_role'] = ansible_role + data['ansible_task'] = result._task_fields + # Removing args since it can contain sensitive data + if 'args' in data['ansible_task']: + data['ansible_task'].pop('args') + data['ansible_result'] = result._result + if 'content' in data['ansible_result']: + data['ansible_result'].pop('content') + + # Adding extra vars info + data['extra_vars'] = self.extra_vars + + # Preparing the playbook logs as JSON format and send to Azure log analytics + jsondata = json.dumps({'event': data}, cls=AnsibleJSONEncoder, sort_keys=True) + content_length = len(jsondata) + rfc1123date = self.__rfc1123date() + signature = self.__build_signature(rfc1123date, workspace_id, shared_key, content_length) + workspace_url = self.__build_workspace_url(workspace_id) + + open_url( + workspace_url, + jsondata, + headers={ + 'content-type': 'application/json', + 'Authorization': signature, + 'Log-Type': 'ansible_playbook', + 'x-ms-date': rfc1123date + }, + method='POST' + ) + + +class CallbackModule(CallbackBase): + CALLBACK_VERSION = 2.0 + CALLBACK_TYPE = 'aggregate' + CALLBACK_NAME = 'loganalytics' + CALLBACK_NEEDS_WHITELIST = True + + def __init__(self, display=None): + super(CallbackModule, self).__init__(display=display) + self.start_datetimes = {} # Collect task start times + self.workspace_id = None + self.shared_key = None + self.loganalytics = AzureLogAnalyticsSource() + + def _seconds_since_start(self, result): + return ( + datetime.utcnow() - + self.start_datetimes[result._task._uuid] + ).total_seconds() + + def set_options(self, task_keys=None, var_options=None, direct=None): + super(CallbackModule, self).set_options(task_keys=task_keys, var_options=var_options, direct=direct) + self.workspace_id = self.get_option('workspace_id') + self.shared_key = self.get_option('shared_key') + + def v2_playbook_on_play_start(self, play): + vm = play.get_variable_manager() + extra_vars = vm.extra_vars + self.loganalytics.extra_vars = extra_vars + + def v2_playbook_on_start(self, playbook): + self.loganalytics.ansible_playbook = basename(playbook._file_name) + + def v2_playbook_on_task_start(self, task, is_conditional): + self.start_datetimes[task._uuid] = datetime.utcnow() + + def v2_playbook_on_handler_task_start(self, task): + self.start_datetimes[task._uuid] = datetime.utcnow() + + def v2_runner_on_ok(self, result, **kwargs): + self.loganalytics.send_event( + self.workspace_id, + self.shared_key, + 'OK', + result, + self._seconds_since_start(result) + ) + + def v2_runner_on_skipped(self, result, **kwargs): + self.loganalytics.send_event( + self.workspace_id, + self.shared_key, + 'SKIPPED', + result, + self._seconds_since_start(result) + ) + + def v2_runner_on_failed(self, result, **kwargs): + self.loganalytics.send_event( + self.workspace_id, + self.shared_key, + 'FAILED', + result, + self._seconds_since_start(result) + ) + + def runner_on_async_failed(self, result, **kwargs): + self.loganalytics.send_event( + self.workspace_id, + self.shared_key, + 'FAILED', + result, + self._seconds_since_start(result) + ) + + def v2_runner_on_unreachable(self, result, **kwargs): + self.loganalytics.send_event( + self.workspace_id, + self.shared_key, + 'UNREACHABLE', + result, + self._seconds_since_start(result) + ) diff --git a/tests/unit/plugins/callback/test_loganalytics.py b/tests/unit/plugins/callback/test_loganalytics.py new file mode 100644 index 0000000000..085e1163cb --- /dev/null +++ b/tests/unit/plugins/callback/test_loganalytics.py @@ -0,0 +1,64 @@ +# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) + +from __future__ import (absolute_import, division, print_function) +__metaclass__ = type + +from ansible.executor.task_result import TaskResult +from ansible_collections.community.general.tests.unit.compat import unittest +from ansible_collections.community.general.tests.unit.compat.mock import patch, call, MagicMock, Mock +from ansible_collections.community.general.plugins.callback.loganalytics import AzureLogAnalyticsSource +from datetime import datetime + +import json + + +class TestAzureLogAnalytics(unittest.TestCase): + @patch('ansible_collections.community.general.plugins.callback.loganalytics.socket') + def setUp(self, mock_socket): + mock_socket.gethostname.return_value = 'my-host' + mock_socket.gethostbyname.return_value = '1.2.3.4' + self.loganalytics = AzureLogAnalyticsSource() + self.mock_task = Mock('MockTask') + self.mock_task._role = 'myrole' + self.mock_task._uuid = 'myuuid' + self.task_fields = {'args': {}} + self.mock_host = Mock('MockHost') + self.mock_host.name = 'myhost' + + @patch('ansible_collections.community.general.plugins.callback.loganalytics.datetime') + @patch('ansible_collections.community.general.plugins.callback.loganalytics.open_url') + def test_overall(self, open_url_mock, mock_datetime): + mock_datetime.utcnow.return_value = datetime(2020, 12, 1) + result = TaskResult(host=self.mock_host, task=self.mock_task, return_data={}, task_fields=self.task_fields) + + self.loganalytics.send_event(workspace_id='01234567-0123-0123-0123-01234567890a', + shared_key='dZD0kCbKl3ehZG6LHFMuhtE0yHiFCmetzFMc2u+roXIUQuatqU924SsAAAAPemhjbGlAemhjbGktTUJQAQIDBA==', + state='OK', + result=result, + runtime=100) + + args, kwargs = open_url_mock.call_args + sent_data = json.loads(args[1]) + + self.assertEqual(sent_data['event']['timestamp'], 'Tue, 01 Dec 2020 00:00:00 GMT') + self.assertEqual(sent_data['event']['host'], 'my-host') + self.assertEqual(sent_data['event']['uuid'], 'myuuid') + self.assertEqual(args[0], 'https://01234567-0123-0123-0123-01234567890a.ods.opinsights.azure.com/api/logs?api-version=2016-04-01') + + @patch('ansible_collections.community.general.plugins.callback.loganalytics.datetime') + @patch('ansible_collections.community.general.plugins.callback.loganalytics.open_url') + def test_auth_headers(self, open_url_mock, mock_datetime): + mock_datetime.utcnow.return_value = datetime(2020, 12, 1) + result = TaskResult(host=self.mock_host, task=self.mock_task, return_data={}, task_fields=self.task_fields) + + self.loganalytics.send_event(workspace_id='01234567-0123-0123-0123-01234567890a', + shared_key='dZD0kCbKl3ehZG6LHFMuhtE0yHiFCmetzFMc2u+roXIUQuatqU924SsAAAAPemhjbGlAemhjbGktTUJQAQIDBA==', + state='OK', + result=result, + runtime=100) + + args, kwargs = open_url_mock.call_args + headers = kwargs['headers'] + + self.assertRegexpMatches(headers['Authorization'], r'^SharedKey 01234567-0123-0123-0123-01234567890a:.*=$') + self.assertEqual(headers['Log-Type'], 'ansible_playbook')