diff options
Diffstat (limited to 'ansible_collections/community/general/plugins/callback/unixy.py')
-rw-r--r-- | ansible_collections/community/general/plugins/callback/unixy.py | 248 |
1 files changed, 248 insertions, 0 deletions
diff --git a/ansible_collections/community/general/plugins/callback/unixy.py b/ansible_collections/community/general/plugins/callback/unixy.py new file mode 100644 index 000000000..02a2e46ba --- /dev/null +++ b/ansible_collections/community/general/plugins/callback/unixy.py @@ -0,0 +1,248 @@ +# -*- coding: utf-8 -*- +# Copyright (c) 2017, Allyson Bowles <@akatch> +# Copyright (c) 2012-2014, Michael DeHaan <michael.dehaan@gmail.com> +# 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 + +# Make coding more python3-ish +from __future__ import (absolute_import, division, print_function) +__metaclass__ = type + +DOCUMENTATION = ''' + name: unixy + type: stdout + author: Allyson Bowles (@akatch) + short_description: condensed Ansible output + description: + - Consolidated Ansible output in the style of LINUX/UNIX startup logs. + extends_documentation_fragment: + - default_callback + requirements: + - set as stdout in configuration +''' + +from os.path import basename +from ansible import constants as C +from ansible import context +from ansible.module_utils.common.text.converters import to_text +from ansible.utils.color import colorize, hostcolor +from ansible.plugins.callback.default import CallbackModule as CallbackModule_default + + +class CallbackModule(CallbackModule_default): + + ''' + Design goals: + - Print consolidated output that looks like a *NIX startup log + - Defaults should avoid displaying unnecessary information wherever possible + + TODOs: + - Only display task names if the task runs on at least one host + - Add option to display all hostnames on a single line in the appropriate result color (failures may have a separate line) + - Consolidate stats display + - Display whether run is in --check mode + - Don't show play name if no hosts found + ''' + + CALLBACK_VERSION = 2.0 + CALLBACK_TYPE = 'stdout' + CALLBACK_NAME = 'community.general.unixy' + + def _run_is_verbose(self, result): + return ((self._display.verbosity > 0 or '_ansible_verbose_always' in result._result) and '_ansible_verbose_override' not in result._result) + + def _get_task_display_name(self, task): + self.task_display_name = None + display_name = task.get_name().strip().split(" : ") + + task_display_name = display_name[-1] + if task_display_name.startswith("include"): + return + else: + self.task_display_name = task_display_name + + def _preprocess_result(self, result): + self.delegated_vars = result._result.get('_ansible_delegated_vars', None) + self._handle_exception(result._result, use_stderr=self.get_option('display_failed_stderr')) + self._handle_warnings(result._result) + + def _process_result_output(self, result, msg): + task_host = result._host.get_name() + task_result = "%s %s" % (task_host, msg) + + if self._run_is_verbose(result): + task_result = "%s %s: %s" % (task_host, msg, self._dump_results(result._result, indent=4)) + return task_result + + if self.delegated_vars: + task_delegate_host = self.delegated_vars['ansible_host'] + task_result = "%s -> %s %s" % (task_host, task_delegate_host, msg) + + if result._result.get('msg') and result._result.get('msg') != "All items completed": + task_result += " | msg: " + to_text(result._result.get('msg')) + + if result._result.get('stdout'): + task_result += " | stdout: " + result._result.get('stdout') + + if result._result.get('stderr'): + task_result += " | stderr: " + result._result.get('stderr') + + return task_result + + def v2_playbook_on_task_start(self, task, is_conditional): + self._get_task_display_name(task) + if self.task_display_name is not None: + self._display.display("%s..." % self.task_display_name) + + def v2_playbook_on_handler_task_start(self, task): + self._get_task_display_name(task) + if self.task_display_name is not None: + self._display.display("%s (via handler)... " % self.task_display_name) + + def v2_playbook_on_play_start(self, play): + name = play.get_name().strip() + if name and play.hosts: + msg = u"\n- %s on hosts: %s -" % (name, ",".join(play.hosts)) + else: + msg = u"---" + + self._display.display(msg) + + def v2_runner_on_skipped(self, result, ignore_errors=False): + if self.get_option('display_skipped_hosts'): + self._preprocess_result(result) + display_color = C.COLOR_SKIP + msg = "skipped" + + task_result = self._process_result_output(result, msg) + self._display.display(" " + task_result, display_color) + else: + return + + def v2_runner_on_failed(self, result, ignore_errors=False): + self._preprocess_result(result) + display_color = C.COLOR_ERROR + msg = "failed" + item_value = self._get_item_label(result._result) + if item_value: + msg += " | item: %s" % (item_value,) + + task_result = self._process_result_output(result, msg) + self._display.display(" " + task_result, display_color, stderr=self.get_option('display_failed_stderr')) + + def v2_runner_on_ok(self, result, msg="ok", display_color=C.COLOR_OK): + self._preprocess_result(result) + + result_was_changed = ('changed' in result._result and result._result['changed']) + if result_was_changed: + msg = "done" + item_value = self._get_item_label(result._result) + if item_value: + msg += " | item: %s" % (item_value,) + display_color = C.COLOR_CHANGED + task_result = self._process_result_output(result, msg) + self._display.display(" " + task_result, display_color) + elif self.get_option('display_ok_hosts'): + task_result = self._process_result_output(result, msg) + self._display.display(" " + task_result, display_color) + + def v2_runner_item_on_skipped(self, result): + self.v2_runner_on_skipped(result) + + def v2_runner_item_on_failed(self, result): + self.v2_runner_on_failed(result) + + def v2_runner_item_on_ok(self, result): + self.v2_runner_on_ok(result) + + def v2_runner_on_unreachable(self, result): + self._preprocess_result(result) + + msg = "unreachable" + display_color = C.COLOR_UNREACHABLE + task_result = self._process_result_output(result, msg) + + self._display.display(" " + task_result, display_color, stderr=self.get_option('display_failed_stderr')) + + def v2_on_file_diff(self, result): + if result._task.loop and 'results' in result._result: + for res in result._result['results']: + if 'diff' in res and res['diff'] and res.get('changed', False): + diff = self._get_diff(res['diff']) + if diff: + self._display.display(diff) + elif 'diff' in result._result and result._result['diff'] and result._result.get('changed', False): + diff = self._get_diff(result._result['diff']) + if diff: + self._display.display(diff) + + def v2_playbook_on_stats(self, stats): + self._display.display("\n- Play recap -", screen_only=True) + + hosts = sorted(stats.processed.keys()) + for h in hosts: + # TODO how else can we display these? + t = stats.summarize(h) + + self._display.display(u" %s : %s %s %s %s %s %s" % ( + hostcolor(h, t), + colorize(u'ok', t['ok'], C.COLOR_OK), + colorize(u'changed', t['changed'], C.COLOR_CHANGED), + colorize(u'unreachable', t['unreachable'], C.COLOR_UNREACHABLE), + colorize(u'failed', t['failures'], C.COLOR_ERROR), + colorize(u'rescued', t['rescued'], C.COLOR_OK), + colorize(u'ignored', t['ignored'], C.COLOR_WARN)), + screen_only=True + ) + + self._display.display(u" %s : %s %s %s %s %s %s" % ( + hostcolor(h, t, False), + colorize(u'ok', t['ok'], None), + colorize(u'changed', t['changed'], None), + colorize(u'unreachable', t['unreachable'], None), + colorize(u'failed', t['failures'], None), + colorize(u'rescued', t['rescued'], None), + colorize(u'ignored', t['ignored'], None)), + log_only=True + ) + if stats.custom and self.get_option('show_custom_stats'): + self._display.banner("CUSTOM STATS: ") + # per host + # TODO: come up with 'pretty format' + for k in sorted(stats.custom.keys()): + if k == '_run': + continue + self._display.display('\t%s: %s' % (k, self._dump_results(stats.custom[k], indent=1).replace('\n', ''))) + + # print per run custom stats + if '_run' in stats.custom: + self._display.display("", screen_only=True) + self._display.display('\tRUN: %s' % self._dump_results(stats.custom['_run'], indent=1).replace('\n', '')) + self._display.display("", screen_only=True) + + def v2_playbook_on_no_hosts_matched(self): + self._display.display(" No hosts found!", color=C.COLOR_DEBUG) + + def v2_playbook_on_no_hosts_remaining(self): + self._display.display(" Ran out of hosts!", color=C.COLOR_ERROR) + + def v2_playbook_on_start(self, playbook): + # TODO display whether this run is happening in check mode + self._display.display("Executing playbook %s" % basename(playbook._file_name)) + + # show CLI arguments + if self._display.verbosity > 3: + if context.CLIARGS.get('args'): + self._display.display('Positional arguments: %s' % ' '.join(context.CLIARGS['args']), + color=C.COLOR_VERBOSE, screen_only=True) + + for argument in (a for a in context.CLIARGS if a != 'args'): + val = context.CLIARGS[argument] + if val: + self._display.vvvv('%s: %s' % (argument, val)) + + def v2_runner_retry(self, result): + msg = " Retrying... (%d of %d)" % (result._result['attempts'], result._result['retries']) + if self._run_is_verbose(result): + msg += "Result was: %s" % self._dump_results(result._result) + self._display.display(msg, color=C.COLOR_DEBUG) |