diff options
Diffstat (limited to 'callback_plugins')
| -rw-r--r-- | callback_plugins/openshift_quick_installer.py | 291 | 
1 files changed, 291 insertions, 0 deletions
diff --git a/callback_plugins/openshift_quick_installer.py b/callback_plugins/openshift_quick_installer.py new file mode 100644 index 000000000..e2f125df9 --- /dev/null +++ b/callback_plugins/openshift_quick_installer.py @@ -0,0 +1,291 @@ +# pylint: disable=invalid-name,protected-access,import-error,line-too-long + +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program.  If not, see <http://www.gnu.org/licenses/>. + +"""This file is a stdout callback plugin for the OpenShift Quick +Installer. The purpose of this callback plugin is to reduce the amount +of produced output for customers and enable simpler progress checking. + +What's different: + +* Playbook progress is expressed as: Play <current_play>/<total_plays> (Play Name) +  Ex: Play 3/30 (Initialize Megafrobber) + +* The Tasks and Handlers in each play (and included roles) are printed +  as a series of .'s following the play progress line. + +* Many of these methods include copy and paste code from the upstream +  default.py callback. We do that to give us control over the stdout +  output while allowing Ansible to handle the file logging +  normally. The biggest changes here are that we are manually setting +  `log_only` to True in the Display.display method and we redefine the +  Display.banner method locally so we can set log_only on that call as +  well. + +""" + +from __future__ import (absolute_import, print_function) +import imp +import os +import sys +from ansible import constants as C +from ansible.utils.color import colorize, hostcolor +ANSIBLE_PATH = imp.find_module('ansible')[1] +DEFAULT_PATH = os.path.join(ANSIBLE_PATH, 'plugins/callback/default.py') +DEFAULT_MODULE = imp.load_source( +    'ansible.plugins.callback.default', +    DEFAULT_PATH +) + +try: +    from ansible.plugins.callback import CallbackBase +    BASECLASS = CallbackBase +except ImportError:  # < ansible 2.1 +    BASECLASS = DEFAULT_MODULE.CallbackModule + + +reload(sys) +sys.setdefaultencoding('utf-8') + + +class CallbackModule(DEFAULT_MODULE.CallbackModule): + +    """ +    Ansible callback plugin +    """ +    CALLBACK_VERSION = 2.2 +    CALLBACK_TYPE = 'stdout' +    CALLBACK_NAME = 'openshift_quick_installer' +    CALLBACK_NEEDS_WHITELIST = False +    plays_count = 0 +    plays_total_ran = 0 + +    def banner(self, msg, color=None): +        '''Prints a header-looking line with stars taking up to 80 columns +        of width (3 columns, minimum) + +        Overrides the upstream banner method so that display is called +        with log_only=True +        ''' +        msg = msg.strip() +        star_len = (79 - len(msg)) +        if star_len < 0: +            star_len = 3 +        stars = "*" * star_len +        self._display.display("\n%s %s" % (msg, stars), color=color, log_only=True) + +    def v2_playbook_on_start(self, playbook): +        """This is basically the start of it all""" +        self.plays_count = len(playbook.get_plays()) +        self.plays_total_ran = 0 + +        if self._display.verbosity > 1: +            from os.path import basename +            self.banner("PLAYBOOK: %s" % basename(playbook._file_name)) + +    def v2_playbook_on_play_start(self, play): +        """Each play calls this once before running any tasks + +We could print the number of tasks here as well by using +`play.get_tasks()` but that is not accurate when a play includes a +role. Only the tasks directly assigned to a play are exposed in the +`play` object. +        """ +        self.plays_total_ran += 1 +        print("") +        print("Play %s/%s (%s)" % (self.plays_total_ran, self.plays_count, play.get_name())) + +        name = play.get_name().strip() +        if not name: +            msg = "PLAY" +        else: +            msg = "PLAY [%s]" % name + +        self.banner(msg) + +    # pylint: disable=unused-argument,no-self-use +    def v2_playbook_on_task_start(self, task, is_conditional): +        """This prints out the task header. For example: + +TASK [openshift_facts : Ensure PyYaml is installed] ***... + +Rather than print out all that for every task, we print a dot +character to indicate a task has been started. +        """ +        sys.stdout.write('.') + +        args = '' +        # args can be specified as no_log in several places: in the task or in +        # the argument spec.  We can check whether the task is no_log but the +        # argument spec can't be because that is only run on the target +        # machine and we haven't run it thereyet at this time. +        # +        # So we give people a config option to affect display of the args so +        # that they can secure this if they feel that their stdout is insecure +        # (shoulder surfing, logging stdout straight to a file, etc). +        if not task.no_log and C.DISPLAY_ARGS_TO_STDOUT: +            args = ', '.join(('%s=%s' % a for a in task.args.items())) +            args = ' %s' % args +        self.banner("TASK [%s%s]" % (task.get_name().strip(), args)) +        if self._display.verbosity >= 2: +            path = task.get_path() +            if path: +                self._display.display("task path: %s" % path, color=C.COLOR_DEBUG, log_only=True) + +    # pylint: disable=unused-argument,no-self-use +    def v2_playbook_on_handler_task_start(self, task): +        """Print out task header for handlers + +Rather than print out a header for every handler, we print a dot +character to indicate a handler task has been started. +""" +        sys.stdout.write('.') + +        self.banner("RUNNING HANDLER [%s]" % task.get_name().strip()) + +    # pylint: disable=unused-argument,no-self-use +    def v2_playbook_on_cleanup_task_start(self, task): +        """Print out a task header for cleanup tasks + +Rather than print out a header for every handler, we print a dot +character to indicate a handler task has been started. +""" +        sys.stdout.write('.') + +        self.banner("CLEANUP TASK [%s]" % task.get_name().strip()) + +    def v2_playbook_on_include(self, included_file): +        """Print out paths to statically included files""" +        msg = 'included: %s for %s' % (included_file._filename, ", ".join([h.name for h in included_file._hosts])) +        self._display.display(msg, color=C.COLOR_SKIP, log_only=True) + +    def v2_runner_on_ok(self, result): +        """This prints out task results in a fancy format + +The only thing we change here is adding `log_only=True` to the +.display() call +        """ +        delegated_vars = result._result.get('_ansible_delegated_vars', None) +        self._clean_results(result._result, result._task.action) +        if result._task.action in ('include', 'include_role'): +            return +        elif result._result.get('changed', False): +            if delegated_vars: +                msg = "changed: [%s -> %s]" % (result._host.get_name(), delegated_vars['ansible_host']) +            else: +                msg = "changed: [%s]" % result._host.get_name() +            color = C.COLOR_CHANGED +        else: +            if delegated_vars: +                msg = "ok: [%s -> %s]" % (result._host.get_name(), delegated_vars['ansible_host']) +            else: +                msg = "ok: [%s]" % result._host.get_name() +            color = C.COLOR_OK + +        if result._task.loop and 'results' in result._result: +            self._process_items(result) +        else: + +            if (self._display.verbosity > 0 or '_ansible_verbose_always' in result._result) and '_ansible_verbose_override' not in result._result: +                msg += " => %s" % (self._dump_results(result._result),) +            self._display.display(msg, color=color, log_only=True) + +        self._handle_warnings(result._result) + +    def v2_runner_item_on_ok(self, result): +        """Print out task results for items you're iterating over""" +        delegated_vars = result._result.get('_ansible_delegated_vars', None) +        if result._task.action in ('include', 'include_role'): +            return +        elif result._result.get('changed', False): +            msg = 'changed' +            color = C.COLOR_CHANGED +        else: +            msg = 'ok' +            color = C.COLOR_OK + +        if delegated_vars: +            msg += ": [%s -> %s]" % (result._host.get_name(), delegated_vars['ansible_host']) +        else: +            msg += ": [%s]" % result._host.get_name() + +        msg += " => (item=%s)" % (self._get_item(result._result),) + +        if (self._display.verbosity > 0 or '_ansible_verbose_always' in result._result) and '_ansible_verbose_override' not in result._result: +            msg += " => %s" % self._dump_results(result._result) +        self._display.display(msg, color=color, log_only=True) + +    def v2_runner_item_on_skipped(self, result): +        """Print out task results when an item is skipped""" +        if C.DISPLAY_SKIPPED_HOSTS: +            msg = "skipping: [%s] => (item=%s) " % (result._host.get_name(), self._get_item(result._result)) +            if (self._display.verbosity > 0 or '_ansible_verbose_always' in result._result) and '_ansible_verbose_override' not in result._result: +                msg += " => %s" % self._dump_results(result._result) +            self._display.display(msg, color=C.COLOR_SKIP, log_only=True) + +    def v2_runner_on_skipped(self, result): +        """Print out task results when a task (or something else?) is skipped""" +        if C.DISPLAY_SKIPPED_HOSTS: +            if result._task.loop and 'results' in result._result: +                self._process_items(result) +            else: +                msg = "skipping: [%s]" % result._host.get_name() +                if (self._display.verbosity > 0 or '_ansible_verbose_always' in result._result) and '_ansible_verbose_override' not in result._result: +                    msg += " => %s" % self._dump_results(result._result) +                self._display.display(msg, color=C.COLOR_SKIP, log_only=True) + +    def v2_playbook_on_notify(self, res, handler): +        """What happens when a task result is 'changed' and the task has a +'notify' list attached. +        """ +        self._display.display("skipping: no hosts matched", color=C.COLOR_SKIP, log_only=True) + +    def v2_playbook_on_stats(self, stats): +        """Print the final playbook run stats""" +        self._display.display("", screen_only=True) +        self.banner("PLAY RECAP") + +        hosts = sorted(stats.processed.keys()) +        for h in hosts: +            t = stats.summarize(h) + +            self._display.display( +                u"%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)), +                screen_only=True +            ) + +            self._display.display( +                u"%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)), +                log_only=True +            ) + +        self._display.display("", screen_only=True) +        self._display.display("", screen_only=True) + +        # Some plays are conditional and won't run (such as load +        # balancers) if they aren't required. Let the user know about +        # this to avoid potential confusion. +        if self.plays_total_ran != self.plays_count: +            print("Installation Complete: Note: Play count is an estimate and some were skipped because your install does not require them") +            self._display.display("", screen_only=True)  | 
