Save the command line arguments into a global context

* Once cli args are parsed, they're constant.  So, save the parsed args
  into the global context for everyone else to use them from now on.
* Port cli scripts to use the CLIARGS in the context
* Refactor call to parse cli args into the run() method
* Fix unittests for changes to the internals of CLI arg parsing
* Port callback plugins to use context.CLIARGS
  * Got rid of the private self._options attribute
  * Use context.CLIARGS in the individual callback plugins instead.
  * Also output positional arguments in default and unixy plugins
  * Code has been simplified since we're now dealing with a dict rather
    than Optparse.Value
This commit is contained in:
Toshio Kuratomi 2018-12-17 18:10:59 -08:00
parent c18da65089
commit afdbb0d9d5
36 changed files with 1033 additions and 868 deletions

View file

@ -29,6 +29,7 @@ import shutil
import sys import sys
import traceback import traceback
from ansible import context
from ansible.errors import AnsibleError, AnsibleOptionsError, AnsibleParserError from ansible.errors import AnsibleError, AnsibleOptionsError, AnsibleParserError
from ansible.module_utils._text import to_text from ansible.module_utils._text import to_text
@ -106,7 +107,6 @@ if __name__ == '__main__':
exit_code = 6 exit_code = 6
else: else:
cli = mycli(args) cli = mycli(args)
cli.parse()
exit_code = cli.run() exit_code = cli.run()
except AnsibleOptionsError as e: except AnsibleOptionsError as e:
@ -134,9 +134,9 @@ if __name__ == '__main__':
# Show raw stacktraces in debug mode, It also allow pdb to # Show raw stacktraces in debug mode, It also allow pdb to
# enter post mortem mode. # enter post mortem mode.
raise raise
have_cli_options = cli is not None and cli.options is not None have_cli_options = bool(context.CLIARGS)
display.error("Unexpected Exception, this is probably a bug: %s" % to_text(e), wrap_text=False) display.error("Unexpected Exception, this is probably a bug: %s" % to_text(e), wrap_text=False)
if not have_cli_options or have_cli_options and cli.options.verbosity > 2: if not have_cli_options or have_cli_options and context.CLIARGS['verbosity'] > 2:
log_only = False log_only = False
if hasattr(e, 'orig_exc'): if hasattr(e, 'orig_exc'):
display.vvv('\nexception type: %s' % to_text(type(e.orig_exc))) display.vvv('\nexception type: %s' % to_text(type(e.orig_exc)))

View file

@ -0,0 +1,6 @@
---
minor_changes:
- Refactored the CLI code to parse the CLI arguments and then save them into
a non-mutatable global singleton. This should make it easier to modify.
- Removed the private ``_options`` attribute of ``CallbackBase``. See the porting
guide if you need access to the command line arguments in a callback plugin.

View file

@ -148,6 +148,21 @@ Plugins
* Order of enabled inventory plugins (:ref:`INVENTORY_ENABLED`) has been updated, :ref:`auto <auto_inventory>` is now before :ref:`yaml <yaml_inventory>` and :ref:`ini <ini_inventory>`. * Order of enabled inventory plugins (:ref:`INVENTORY_ENABLED`) has been updated, :ref:`auto <auto_inventory>` is now before :ref:`yaml <yaml_inventory>` and :ref:`ini <ini_inventory>`.
* The private ``_options`` attribute has been removed from the ``CallbackBase`` class of callback
plugins. If you have a third-party callback plugin which needs to access the command line arguments,
use code like the following instead of trying to use ``self._options``:
.. code-block:: python
from ansible import context
[...]
tags = context.CLIARGS['tags']
``context.CLIARGS`` is a read-only dictionary so normal dictionary retrieval methods like
``CLIARGS.get('tags')`` and ``CLIARGS['tags']`` work as expected but you won't be able to modify
the cli arguments at all.
Porting custom scripts Porting custom scripts
====================== ======================

View file

@ -1,20 +1,7 @@
# (c) 2012-2014, Michael DeHaan <michael.dehaan@gmail.com> # Copyright: (c) 2012-2014, Michael DeHaan <michael.dehaan@gmail.com>
# (c) 2016, Toshio Kuratomi <tkuratomi@ansible.com> # Copyright: (c) 2016, Toshio Kuratomi <tkuratomi@ansible.com>
# # Copyright: (c) 2018, Ansible Project
# This file is part of Ansible # GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
#
# Ansible 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.
#
# Ansible 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 Ansible. If not, see <http://www.gnu.org/licenses/>.
# Make coding more python3-ish # Make coding more python3-ish
from __future__ import (absolute_import, division, print_function) from __future__ import (absolute_import, division, print_function)
@ -34,6 +21,7 @@ from abc import ABCMeta, abstractmethod
import ansible import ansible
from ansible import constants as C from ansible import constants as C
from ansible import context
from ansible.errors import AnsibleOptionsError, AnsibleError from ansible.errors import AnsibleOptionsError, AnsibleError
from ansible.inventory.manager import InventoryManager from ansible.inventory.manager import InventoryManager
from ansible.module_utils.six import with_metaclass, string_types from ansible.module_utils.six import with_metaclass, string_types
@ -46,6 +34,7 @@ from ansible.utils.vars import load_extra_vars, load_options_vars
from ansible.vars.manager import VariableManager from ansible.vars.manager import VariableManager
from ansible.parsing.vault import PromptVaultSecret, get_file_vault_secret from ansible.parsing.vault import PromptVaultSecret, get_file_vault_secret
display = Display() display = Display()
@ -93,6 +82,156 @@ class InvalidOptsParser(SortedOptParser):
pass pass
def base_parser(usage="", output_opts=False, runas_opts=False, meta_opts=False, runtask_opts=False,
vault_opts=False, module_opts=False, async_opts=False, connect_opts=False,
subset_opts=False, check_opts=False, inventory_opts=False, epilog=None,
fork_opts=False, runas_prompt_opts=False, desc=None, basedir_opts=False,
vault_rekey_opts=False):
"""
Create an options parser for most ansible scripts
"""
# base opts
parser = SortedOptParser(usage, version=CLI.version("%prog"), description=desc, epilog=epilog)
parser.remove_option('--version')
version_help = "show program's version number, config file location, configured module search path," \
" module location, executable location and exit"
parser.add_option('--version', action="version", help=version_help)
parser.add_option('-v', '--verbose', dest='verbosity', default=C.DEFAULT_VERBOSITY, action="count",
help="verbose mode (-vvv for more, -vvvv to enable connection debugging)")
if inventory_opts:
parser.add_option('-i', '--inventory', '--inventory-file', dest='inventory', action="append",
help="specify inventory host path or comma separated host list. --inventory-file is deprecated")
parser.add_option('--list-hosts', dest='listhosts', action='store_true',
help='outputs a list of matching hosts; does not execute anything else')
parser.add_option('-l', '--limit', default=C.DEFAULT_SUBSET, dest='subset',
help='further limit selected hosts to an additional pattern')
if module_opts:
parser.add_option('-M', '--module-path', dest='module_path', default=None,
help="prepend colon-separated path(s) to module library (default=%s)" % C.DEFAULT_MODULE_PATH,
action="callback", callback=CLI.unfrack_paths, type='str')
if runtask_opts:
parser.add_option('-e', '--extra-vars', dest="extra_vars", action="append",
help="set additional variables as key=value or YAML/JSON, if filename prepend with @", default=[])
if fork_opts:
parser.add_option('-f', '--forks', dest='forks', default=C.DEFAULT_FORKS, type='int',
help="specify number of parallel processes to use (default=%s)" % C.DEFAULT_FORKS)
if vault_opts:
parser.add_option('--ask-vault-pass', default=C.DEFAULT_ASK_VAULT_PASS, dest='ask_vault_pass', action='store_true',
help='ask for vault password')
parser.add_option('--vault-password-file', default=[], dest='vault_password_files',
help="vault password file", action="callback", callback=CLI.unfrack_paths, type='string')
parser.add_option('--vault-id', default=[], dest='vault_ids', action='append', type='string',
help='the vault identity to use')
if vault_rekey_opts:
parser.add_option('--new-vault-password-file', default=None, dest='new_vault_password_file',
help="new vault password file for rekey", action="callback", callback=CLI.unfrack_path, type='string')
parser.add_option('--new-vault-id', default=None, dest='new_vault_id', type='string',
help='the new vault identity to use for rekey')
if subset_opts:
parser.add_option('-t', '--tags', dest='tags', default=C.TAGS_RUN, action='append',
help="only run plays and tasks tagged with these values")
parser.add_option('--skip-tags', dest='skip_tags', default=C.TAGS_SKIP, action='append',
help="only run plays and tasks whose tags do not match these values")
if output_opts:
parser.add_option('-o', '--one-line', dest='one_line', action='store_true',
help='condense output')
parser.add_option('-t', '--tree', dest='tree', default=None,
help='log output to this directory')
if connect_opts:
connect_group = optparse.OptionGroup(parser, "Connection Options", "control as whom and how to connect to hosts")
connect_group.add_option('-k', '--ask-pass', default=C.DEFAULT_ASK_PASS, dest='ask_pass', action='store_true',
help='ask for connection password')
connect_group.add_option('--private-key', '--key-file', default=C.DEFAULT_PRIVATE_KEY_FILE, dest='private_key_file',
help='use this file to authenticate the connection', action="callback", callback=CLI.unfrack_path, type='string')
connect_group.add_option('-u', '--user', default=C.DEFAULT_REMOTE_USER, dest='remote_user',
help='connect as this user (default=%s)' % C.DEFAULT_REMOTE_USER)
connect_group.add_option('-c', '--connection', dest='connection', default=C.DEFAULT_TRANSPORT,
help="connection type to use (default=%s)" % C.DEFAULT_TRANSPORT)
connect_group.add_option('-T', '--timeout', default=C.DEFAULT_TIMEOUT, type='int', dest='timeout',
help="override the connection timeout in seconds (default=%s)" % C.DEFAULT_TIMEOUT)
connect_group.add_option('--ssh-common-args', default='', dest='ssh_common_args',
help="specify common arguments to pass to sftp/scp/ssh (e.g. ProxyCommand)")
connect_group.add_option('--sftp-extra-args', default='', dest='sftp_extra_args',
help="specify extra arguments to pass to sftp only (e.g. -f, -l)")
connect_group.add_option('--scp-extra-args', default='', dest='scp_extra_args',
help="specify extra arguments to pass to scp only (e.g. -l)")
connect_group.add_option('--ssh-extra-args', default='', dest='ssh_extra_args',
help="specify extra arguments to pass to ssh only (e.g. -R)")
parser.add_option_group(connect_group)
runas_group = None
rg = optparse.OptionGroup(parser, "Privilege Escalation Options", "control how and which user you become as on target hosts")
if runas_opts:
runas_group = rg
# priv user defaults to root later on to enable detecting when this option was given here
runas_group.add_option("-s", "--sudo", default=C.DEFAULT_SUDO, action="store_true", dest='sudo',
help="run operations with sudo (nopasswd) (deprecated, use become)")
runas_group.add_option('-U', '--sudo-user', dest='sudo_user', default=None,
help='desired sudo user (default=root) (deprecated, use become)')
runas_group.add_option('-S', '--su', default=C.DEFAULT_SU, action='store_true',
help='run operations with su (deprecated, use become)')
runas_group.add_option('-R', '--su-user', default=None,
help='run operations with su as this user (default=%s) (deprecated, use become)' % C.DEFAULT_SU_USER)
# consolidated privilege escalation (become)
runas_group.add_option("-b", "--become", default=C.DEFAULT_BECOME, action="store_true", dest='become',
help="run operations with become (does not imply password prompting)")
runas_group.add_option('--become-method', dest='become_method', default=C.DEFAULT_BECOME_METHOD, type='choice', choices=C.BECOME_METHODS,
help="privilege escalation method to use (default=%s), valid choices: [ %s ]" %
(C.DEFAULT_BECOME_METHOD, ' | '.join(C.BECOME_METHODS)))
runas_group.add_option('--become-user', default=None, dest='become_user', type='string',
help='run operations as this user (default=%s)' % C.DEFAULT_BECOME_USER)
if runas_opts or runas_prompt_opts:
if not runas_group:
runas_group = rg
runas_group.add_option('--ask-sudo-pass', default=C.DEFAULT_ASK_SUDO_PASS, dest='ask_sudo_pass', action='store_true',
help='ask for sudo password (deprecated, use become)')
runas_group.add_option('--ask-su-pass', default=C.DEFAULT_ASK_SU_PASS, dest='ask_su_pass', action='store_true',
help='ask for su password (deprecated, use become)')
runas_group.add_option('-K', '--ask-become-pass', default=False, dest='become_ask_pass', action='store_true',
help='ask for privilege escalation password')
if runas_group:
parser.add_option_group(runas_group)
if async_opts:
parser.add_option('-P', '--poll', default=C.DEFAULT_POLL_INTERVAL, type='int', dest='poll_interval',
help="set the poll interval if using -B (default=%s)" % C.DEFAULT_POLL_INTERVAL)
parser.add_option('-B', '--background', dest='seconds', type='int', default=0,
help='run asynchronously, failing after X seconds (default=N/A)')
if check_opts:
parser.add_option("-C", "--check", default=False, dest='check', action='store_true',
help="don't make any changes; instead, try to predict some of the changes that may occur")
parser.add_option('--syntax-check', dest='syntax', action='store_true',
help="perform a syntax check on the playbook, but do not execute it")
parser.add_option("-D", "--diff", default=C.DIFF_ALWAYS, dest='diff', action='store_true',
help="when changing (small) files and templates, show the differences in those files; works great with --check")
if meta_opts:
parser.add_option('--force-handlers', default=C.DEFAULT_FORCE_HANDLERS, dest='force_handlers', action='store_true',
help="run handlers even if a task fails")
parser.add_option('--flush-cache', dest='flush_cache', action='store_true',
help="clear the fact cache for every host in inventory")
if basedir_opts:
parser.add_option('--playbook-dir', default=None, dest='basedir', action='store',
help="Since this tool does not use playbooks, use this as a subsitute playbook directory."
"This sets the relative path for many features including roles/ group_vars/ etc.")
return parser
class CLI(with_metaclass(ABCMeta, object)): class CLI(with_metaclass(ABCMeta, object)):
''' code behind bin/ansible* programs ''' ''' code behind bin/ansible* programs '''
@ -117,7 +256,6 @@ class CLI(with_metaclass(ABCMeta, object)):
""" """
self.args = args self.args = args
self.options = None
self.parser = None self.parser = None
self.action = None self.action = None
self.callback = callback self.callback = callback
@ -158,6 +296,7 @@ class CLI(with_metaclass(ABCMeta, object)):
Subclasses must implement this method. It does the actual work of Subclasses must implement this method. It does the actual work of
running an Ansible command. running an Ansible command.
""" """
self.parse()
display.vv(to_text(self.parser.get_version())) display.vv(to_text(self.parser.get_version()))
@ -308,15 +447,15 @@ class CLI(with_metaclass(ABCMeta, object)):
def ask_passwords(self): def ask_passwords(self):
''' prompt for connection and become passwords if needed ''' ''' prompt for connection and become passwords if needed '''
op = self.options op = context.CLIARGS
sshpass = None sshpass = None
becomepass = None becomepass = None
become_prompt = '' become_prompt = ''
become_prompt_method = "BECOME" if C.AGNOSTIC_BECOME_PROMPT else op.become_method.upper() become_prompt_method = "BECOME" if C.AGNOSTIC_BECOME_PROMPT else op['become_method'].upper()
try: try:
if op.ask_pass: if op['ask_pass']:
sshpass = getpass.getpass(prompt="SSH password: ") sshpass = getpass.getpass(prompt="SSH password: ")
become_prompt = "%s password[defaults to SSH password]: " % become_prompt_method become_prompt = "%s password[defaults to SSH password]: " % become_prompt_method
if sshpass: if sshpass:
@ -324,9 +463,9 @@ class CLI(with_metaclass(ABCMeta, object)):
else: else:
become_prompt = "%s password: " % become_prompt_method become_prompt = "%s password: " % become_prompt_method
if op.become_ask_pass: if op['become_ask_pass']:
becomepass = getpass.getpass(prompt=become_prompt) becomepass = getpass.getpass(prompt=become_prompt)
if op.ask_pass and becomepass == '': if op['ask_pass'] and becomepass == '':
becomepass = sshpass becomepass = sshpass
if becomepass: if becomepass:
becomepass = to_bytes(becomepass) becomepass = to_bytes(becomepass)
@ -335,43 +474,46 @@ class CLI(with_metaclass(ABCMeta, object)):
return (sshpass, becomepass) return (sshpass, becomepass)
def normalize_become_options(self): @staticmethod
''' this keeps backwards compatibility with sudo/su self.options ''' def normalize_become_options(options):
self.options.become_ask_pass = self.options.become_ask_pass or self.options.ask_sudo_pass or self.options.ask_su_pass or C.DEFAULT_BECOME_ASK_PASS ''' this keeps backwards compatibility with sudo/su command line options '''
self.options.become_user = self.options.become_user or self.options.sudo_user or self.options.su_user or C.DEFAULT_BECOME_USER if not options.become_ask_pass:
options.become_ask_pass = options.ask_sudo_pass or options.ask_su_pass or C.DEFAULT_BECOME_ASK_PASS
if not options.become_user:
options.become_user = options.sudo_user or options.su_user or C.DEFAULT_BECOME_USER
def _dep(which): def _dep(which):
display.deprecated('The %s command line option has been deprecated in favor of the "become" command line arguments' % which, '2.9') display.deprecated('The %s command line option has been deprecated in favor of the "become" command line arguments' % which, '2.9')
if self.options.become: if options.become:
pass pass
elif self.options.sudo: elif options.sudo:
self.options.become = True options.become = True
self.options.become_method = 'sudo' options.become_method = 'sudo'
_dep('sudo') _dep('sudo')
elif self.options.su: elif options.su:
self.options.become = True options.become = True
self.options.become_method = 'su' options.become_method = 'su'
_dep('su') _dep('su')
# other deprecations: # other deprecations:
if self.options.ask_sudo_pass or self.options.sudo_user: if options.ask_sudo_pass or options.sudo_user:
_dep('sudo') _dep('sudo')
if self.options.ask_su_pass or self.options.su_user: if options.ask_su_pass or options.su_user:
_dep('su') _dep('su')
def validate_conflicts(self, vault_opts=False, runas_opts=False, fork_opts=False, vault_rekey_opts=False): return options
''' check for conflicting options '''
op = self.options def validate_conflicts(self, op, vault_opts=False, runas_opts=False, fork_opts=False, vault_rekey_opts=False):
''' check for conflicting options '''
if vault_opts: if vault_opts:
# Check for vault related conflicts # Check for vault related conflicts
if (op.ask_vault_pass and op.vault_password_files): if op.ask_vault_pass and op.vault_password_files:
self.parser.error("--ask-vault-pass and --vault-password-file are mutually exclusive") self.parser.error("--ask-vault-pass and --vault-password-file are mutually exclusive")
if vault_rekey_opts: if vault_rekey_opts:
if (op.new_vault_id and op.new_vault_password_file): if op.new_vault_id and op.new_vault_password_file:
self.parser.error("--new-vault-password-file and --new-vault-id are mutually exclusive") self.parser.error("--new-vault-password-file and --new-vault-id are mutually exclusive")
if runas_opts: if runas_opts:
@ -380,13 +522,17 @@ class CLI(with_metaclass(ABCMeta, object)):
(op.su or op.su_user) and (op.become or op.become_user) or (op.su or op.su_user) and (op.become or op.become_user) or
(op.sudo or op.sudo_user) and (op.become or op.become_user)): (op.sudo or op.sudo_user) and (op.become or op.become_user)):
self.parser.error("Sudo arguments ('--sudo', '--sudo-user', and '--ask-sudo-pass') and su arguments ('--su', '--su-user', and '--ask-su-pass') " self.parser.error("Sudo arguments ('--sudo', '--sudo-user', and '--ask-sudo-pass')"
"and become arguments ('--become', '--become-user', and '--ask-become-pass') are exclusive of each other") " and su arguments ('--su', '--su-user', and '--ask-su-pass')"
" and become arguments ('--become', '--become-user', and"
" '--ask-become-pass') are exclusive of each other")
if fork_opts: if fork_opts:
if op.forks < 1: if op.forks < 1:
self.parser.error("The number of processes (--forks) must be >= 1") self.parser.error("The number of processes (--forks) must be >= 1")
return op
@staticmethod @staticmethod
def unfrack_paths(option, opt, value, parser): def unfrack_paths(option, opt, value, parser):
paths = getattr(parser.values, option.dest) paths = getattr(parser.values, option.dest)
@ -409,208 +555,104 @@ class CLI(with_metaclass(ABCMeta, object)):
else: else:
setattr(parser.values, option.dest, value) setattr(parser.values, option.dest, value)
@staticmethod @abstractmethod
def base_parser(usage="", output_opts=False, runas_opts=False, meta_opts=False, runtask_opts=False, vault_opts=False, module_opts=False, def init_parser(self, usage="", output_opts=False, runas_opts=False, meta_opts=False,
async_opts=False, connect_opts=False, subset_opts=False, check_opts=False, inventory_opts=False, epilog=None, fork_opts=False, runtask_opts=False, vault_opts=False, module_opts=False, async_opts=False,
runas_prompt_opts=False, desc=None, basedir_opts=False, vault_rekey_opts=False): connect_opts=False, subset_opts=False, check_opts=False, inventory_opts=False,
''' create an options parser for most ansible scripts ''' epilog=None, fork_opts=False, runas_prompt_opts=False, desc=None,
basedir_opts=False, vault_rekey_opts=False):
"""
Create an options parser for most ansible scripts
# base opts Subclasses need to implement this method. They will usually call the base class's
parser = SortedOptParser(usage, version=CLI.version("%prog"), description=desc, epilog=epilog) init_parser to create a basic version and then add their own options on top of that.
parser.remove_option('--version')
version_help = "show program's version number, config file location, configured module search path," \
" module location, executable location and exit"
parser.add_option('--version', action="version", help=version_help)
parser.add_option('-v', '--verbose', dest='verbosity', default=C.DEFAULT_VERBOSITY, action="count",
help="verbose mode (-vvv for more, -vvvv to enable connection debugging)")
if inventory_opts: An implementation will look something like this::
parser.add_option('-i', '--inventory', '--inventory-file', dest='inventory', action="append",
help="specify inventory host path or comma separated host list. --inventory-file is deprecated")
parser.add_option('--list-hosts', dest='listhosts', action='store_true',
help='outputs a list of matching hosts; does not execute anything else')
parser.add_option('-l', '--limit', default=C.DEFAULT_SUBSET, dest='subset',
help='further limit selected hosts to an additional pattern')
if module_opts: def init_parser(self):
parser.add_option('-M', '--module-path', dest='module_path', default=None, self.parser = super(MyCLI, self).init__parser(usage="My Ansible CLI", inventory_opts=True)
help="prepend colon-separated path(s) to module library (default=%s)" % C.DEFAULT_MODULE_PATH, self.parser.add_option('--my-option', dest='my_option', action='store')
action="callback", callback=CLI.unfrack_paths, type='str') return self.parser
if runtask_opts: """
parser.add_option('-e', '--extra-vars', dest="extra_vars", action="append", self.parser = base_parser(usage=usage, output_opts=output_opts, runas_opts=runas_opts,
help="set additional variables as key=value or YAML/JSON, if filename prepend with @", default=[]) meta_opts=meta_opts, runtask_opts=runtask_opts,
vault_opts=vault_opts, module_opts=module_opts,
if fork_opts: async_opts=async_opts, connect_opts=connect_opts,
parser.add_option('-f', '--forks', dest='forks', default=C.DEFAULT_FORKS, type='int', subset_opts=subset_opts, check_opts=check_opts,
help="specify number of parallel processes to use (default=%s)" % C.DEFAULT_FORKS) inventory_opts=inventory_opts, epilog=epilog, fork_opts=fork_opts,
runas_prompt_opts=runas_prompt_opts, desc=desc,
if vault_opts: basedir_opts=basedir_opts, vault_rekey_opts=vault_rekey_opts)
parser.add_option('--ask-vault-pass', default=C.DEFAULT_ASK_VAULT_PASS, dest='ask_vault_pass', action='store_true', return self.parser
help='ask for vault password')
parser.add_option('--vault-password-file', default=[], dest='vault_password_files',
help="vault password file", action="callback", callback=CLI.unfrack_paths, type='string')
parser.add_option('--vault-id', default=[], dest='vault_ids', action='append', type='string',
help='the vault identity to use')
if vault_rekey_opts:
parser.add_option('--new-vault-password-file', default=None, dest='new_vault_password_file',
help="new vault password file for rekey", action="callback", callback=CLI.unfrack_path, type='string')
parser.add_option('--new-vault-id', default=None, dest='new_vault_id', type='string',
help='the new vault identity to use for rekey')
if subset_opts:
parser.add_option('-t', '--tags', dest='tags', default=C.TAGS_RUN, action='append',
help="only run plays and tasks tagged with these values")
parser.add_option('--skip-tags', dest='skip_tags', default=C.TAGS_SKIP, action='append',
help="only run plays and tasks whose tags do not match these values")
if output_opts:
parser.add_option('-o', '--one-line', dest='one_line', action='store_true',
help='condense output')
parser.add_option('-t', '--tree', dest='tree', default=None,
help='log output to this directory')
if connect_opts:
connect_group = optparse.OptionGroup(parser, "Connection Options", "control as whom and how to connect to hosts")
connect_group.add_option('-k', '--ask-pass', default=C.DEFAULT_ASK_PASS, dest='ask_pass', action='store_true',
help='ask for connection password')
connect_group.add_option('--private-key', '--key-file', default=C.DEFAULT_PRIVATE_KEY_FILE, dest='private_key_file',
help='use this file to authenticate the connection', action="callback", callback=CLI.unfrack_path, type='string')
connect_group.add_option('-u', '--user', default=C.DEFAULT_REMOTE_USER, dest='remote_user',
help='connect as this user (default=%s)' % C.DEFAULT_REMOTE_USER)
connect_group.add_option('-c', '--connection', dest='connection', default=C.DEFAULT_TRANSPORT,
help="connection type to use (default=%s)" % C.DEFAULT_TRANSPORT)
connect_group.add_option('-T', '--timeout', default=C.DEFAULT_TIMEOUT, type='int', dest='timeout',
help="override the connection timeout in seconds (default=%s)" % C.DEFAULT_TIMEOUT)
connect_group.add_option('--ssh-common-args', default='', dest='ssh_common_args',
help="specify common arguments to pass to sftp/scp/ssh (e.g. ProxyCommand)")
connect_group.add_option('--sftp-extra-args', default='', dest='sftp_extra_args',
help="specify extra arguments to pass to sftp only (e.g. -f, -l)")
connect_group.add_option('--scp-extra-args', default='', dest='scp_extra_args',
help="specify extra arguments to pass to scp only (e.g. -l)")
connect_group.add_option('--ssh-extra-args', default='', dest='ssh_extra_args',
help="specify extra arguments to pass to ssh only (e.g. -R)")
parser.add_option_group(connect_group)
runas_group = None
rg = optparse.OptionGroup(parser, "Privilege Escalation Options", "control how and which user you become as on target hosts")
if runas_opts:
runas_group = rg
# priv user defaults to root later on to enable detecting when this option was given here
runas_group.add_option("-s", "--sudo", default=C.DEFAULT_SUDO, action="store_true", dest='sudo',
help="run operations with sudo (nopasswd) (deprecated, use become)")
runas_group.add_option('-U', '--sudo-user', dest='sudo_user', default=None,
help='desired sudo user (default=root) (deprecated, use become)')
runas_group.add_option('-S', '--su', default=C.DEFAULT_SU, action='store_true',
help='run operations with su (deprecated, use become)')
runas_group.add_option('-R', '--su-user', default=None,
help='run operations with su as this user (default=%s) (deprecated, use become)' % C.DEFAULT_SU_USER)
# consolidated privilege escalation (become)
runas_group.add_option("-b", "--become", default=C.DEFAULT_BECOME, action="store_true", dest='become',
help="run operations with become (does not imply password prompting)")
runas_group.add_option('--become-method', dest='become_method', default=C.DEFAULT_BECOME_METHOD, type='choice', choices=C.BECOME_METHODS,
help="privilege escalation method to use (default=%s), valid choices: [ %s ]" %
(C.DEFAULT_BECOME_METHOD, ' | '.join(C.BECOME_METHODS)))
runas_group.add_option('--become-user', default=None, dest='become_user', type='string',
help='run operations as this user (default=%s)' % C.DEFAULT_BECOME_USER)
if runas_opts or runas_prompt_opts:
if not runas_group:
runas_group = rg
runas_group.add_option('--ask-sudo-pass', default=C.DEFAULT_ASK_SUDO_PASS, dest='ask_sudo_pass', action='store_true',
help='ask for sudo password (deprecated, use become)')
runas_group.add_option('--ask-su-pass', default=C.DEFAULT_ASK_SU_PASS, dest='ask_su_pass', action='store_true',
help='ask for su password (deprecated, use become)')
runas_group.add_option('-K', '--ask-become-pass', default=False, dest='become_ask_pass', action='store_true',
help='ask for privilege escalation password')
if runas_group:
parser.add_option_group(runas_group)
if async_opts:
parser.add_option('-P', '--poll', default=C.DEFAULT_POLL_INTERVAL, type='int', dest='poll_interval',
help="set the poll interval if using -B (default=%s)" % C.DEFAULT_POLL_INTERVAL)
parser.add_option('-B', '--background', dest='seconds', type='int', default=0,
help='run asynchronously, failing after X seconds (default=N/A)')
if check_opts:
parser.add_option("-C", "--check", default=False, dest='check', action='store_true',
help="don't make any changes; instead, try to predict some of the changes that may occur")
parser.add_option('--syntax-check', dest='syntax', action='store_true',
help="perform a syntax check on the playbook, but do not execute it")
parser.add_option("-D", "--diff", default=C.DIFF_ALWAYS, dest='diff', action='store_true',
help="when changing (small) files and templates, show the differences in those files; works great with --check")
if meta_opts:
parser.add_option('--force-handlers', default=C.DEFAULT_FORCE_HANDLERS, dest='force_handlers', action='store_true',
help="run handlers even if a task fails")
parser.add_option('--flush-cache', dest='flush_cache', action='store_true',
help="clear the fact cache for every host in inventory")
if basedir_opts:
parser.add_option('--playbook-dir', default=None, dest='basedir', action='store',
help="Since this tool does not use playbooks, use this as a subsitute playbook directory."
"This sets the relative path for many features including roles/ group_vars/ etc.")
return parser
@abstractmethod @abstractmethod
def post_process_args(self, options, args):
"""Process the command line args
Subclasses need to implement this method. This method validates and transforms the command
line arguments. It can be used to check whether conflicting values were given, whether filenames
exist, etc.
An implementation will look something like this::
def post_process_args(self, options, args):
options, args = super(MyCLI, self).post_process_args(options, args)
if options.addition and options.subtraction:
raise AnsibleOptionsError('Only one of --addition and --subtraction can be specified')
if isinstance(options.listofhosts, string_types):
options.listofhosts = string_types.split(',')
return options, args
"""
# process tags
if hasattr(options, 'tags') and not options.tags:
# optparse defaults does not do what's expected
options.tags = ['all']
if hasattr(options, 'tags') and options.tags:
tags = set()
for tag_set in options.tags:
for tag in tag_set.split(u','):
tags.add(tag.strip())
options.tags = list(tags)
# process skip_tags
if hasattr(options, 'skip_tags') and options.skip_tags:
skip_tags = set()
for tag_set in options.skip_tags:
for tag in tag_set.split(u','):
skip_tags.add(tag.strip())
options.skip_tags = list(skip_tags)
# process inventory options except for CLIs that require their own processing
if hasattr(options, 'inventory') and not self.SKIP_INVENTORY_DEFAULTS:
if options.inventory:
# should always be list
if isinstance(options.inventory, string_types):
options.inventory = [options.inventory]
# Ensure full paths when needed
options.inventory = [unfrackpath(opt, follow=False) if ',' not in opt else opt for opt in options.inventory]
else:
options.inventory = C.DEFAULT_HOST_LIST
return options, args
def parse(self): def parse(self):
"""Parse the command line args """Parse the command line args
This method parses the command line arguments. It uses the parser This method parses the command line arguments. It uses the parser
stored in the self.parser attribute and saves the args and options in stored in the self.parser attribute and saves the args and options in
self.args and self.options respectively. context.CLIARGS.
Subclasses need to implement this method. They will usually create Subclasses need to implement two helper methods, init_parser() and post_process_args() which
a base_parser, add their own options to the base_parser, and then call are called from this function before and after parsing the arguments.
this method to do the actual parsing. An implementation will look
something like this::
def parse(self):
parser = super(MyCLI, self).base_parser(usage="My Ansible CLI", inventory_opts=True)
parser.add_option('--my-option', dest='my_option', action='store')
self.parser = parser
super(MyCLI, self).parse()
# If some additional transformations are needed for the
# arguments and options, do it here.
""" """
self.init_parser()
self.options, self.args = self.parser.parse_args(self.args[1:]) options, args = self.parser.parse_args(self.args[1:])
options, args = self.post_process_args(options, args)
# process tags options.args = args
if hasattr(self.options, 'tags') and not self.options.tags: context._init_global_context(options)
# optparse defaults does not do what's expected
self.options.tags = ['all']
if hasattr(self.options, 'tags') and self.options.tags:
tags = set()
for tag_set in self.options.tags:
for tag in tag_set.split(u','):
tags.add(tag.strip())
self.options.tags = list(tags)
# process skip_tags
if hasattr(self.options, 'skip_tags') and self.options.skip_tags:
skip_tags = set()
for tag_set in self.options.skip_tags:
for tag in tag_set.split(u','):
skip_tags.add(tag.strip())
self.options.skip_tags = list(skip_tags)
# process inventory options except for CLIs that require their own processing
if hasattr(self.options, 'inventory') and not self.SKIP_INVENTORY_DEFAULTS:
if self.options.inventory:
# should always be list
if isinstance(self.options.inventory, string_types):
self.options.inventory = [self.options.inventory]
# Ensure full paths when needed
self.options.inventory = [unfrackpath(opt, follow=False) if ',' not in opt else opt for opt in self.options.inventory]
else:
self.options.inventory = C.DEFAULT_HOST_LIST
@staticmethod @staticmethod
def version(prog): def version(prog):
@ -763,42 +805,45 @@ class CLI(with_metaclass(ABCMeta, object)):
return t return t
@staticmethod @staticmethod
def _play_prereqs(options): def _play_prereqs():
options = context.CLIARGS
# all needs loader # all needs loader
loader = DataLoader() loader = DataLoader()
basedir = getattr(options, 'basedir', False) basedir = options.get('basedir', False)
if basedir: if basedir:
loader.set_basedir(basedir) loader.set_basedir(basedir)
vault_ids = options.vault_ids vault_ids = list(options['vault_ids'])
default_vault_ids = C.DEFAULT_VAULT_IDENTITY_LIST default_vault_ids = C.DEFAULT_VAULT_IDENTITY_LIST
vault_ids = default_vault_ids + vault_ids vault_ids = default_vault_ids + vault_ids
vault_secrets = CLI.setup_vault_secrets(loader, vault_secrets = CLI.setup_vault_secrets(loader,
vault_ids=vault_ids, vault_ids=vault_ids,
vault_password_files=options.vault_password_files, vault_password_files=list(options['vault_password_files']),
ask_vault_pass=options.ask_vault_pass, ask_vault_pass=options['ask_vault_pass'],
auto_prompt=False) auto_prompt=False)
loader.set_vault_secrets(vault_secrets) loader.set_vault_secrets(vault_secrets)
# create the inventory, and filter it based on the subset specified (if any) # create the inventory, and filter it based on the subset specified (if any)
inventory = InventoryManager(loader=loader, sources=options.inventory) inventory = InventoryManager(loader=loader, sources=options['inventory'])
# create the variable manager, which will be shared throughout # create the variable manager, which will be shared throughout
# the code, ensuring a consistent view of global variables # the code, ensuring a consistent view of global variables
variable_manager = VariableManager(loader=loader, inventory=inventory) variable_manager = VariableManager(loader=loader, inventory=inventory)
if hasattr(options, 'basedir'): # If the basedir is specified as the empty string then it results in cwd being used. This
if options.basedir: # is not a safe location to load vars from
if options.get('basedir', False) is not False:
if basedir:
variable_manager.safe_basedir = True variable_manager.safe_basedir = True
else: else:
variable_manager.safe_basedir = True variable_manager.safe_basedir = True
# load vars from cli options # load vars from cli options
variable_manager.extra_vars = load_extra_vars(loader=loader, options=options) variable_manager.extra_vars = load_extra_vars(loader=loader)
variable_manager.options_vars = load_options_vars(options, CLI.version_info(gitinfo=False)) variable_manager.options_vars = load_options_vars(CLI.version_info(gitinfo=False))
return loader, inventory, variable_manager return loader, inventory, variable_manager

View file

@ -1,24 +1,12 @@
# (c) 2012, Michael DeHaan <michael.dehaan@gmail.com> # Copyright: (c) 2012, Michael DeHaan <michael.dehaan@gmail.com>
# # Copyright: (c) 2018, Toshio Kuratomi <tkuratomi@ansible.com>
# This file is part of Ansible # GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
#
# Ansible 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.
#
# Ansible 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 Ansible. If not, see <http://www.gnu.org/licenses/>.
from __future__ import (absolute_import, division, print_function) from __future__ import (absolute_import, division, print_function)
__metaclass__ = type __metaclass__ = type
from ansible import constants as C from ansible import constants as C
from ansible import context
from ansible.cli import CLI from ansible.cli import CLI
from ansible.errors import AnsibleError, AnsibleOptionsError from ansible.errors import AnsibleError, AnsibleOptionsError
from ansible.executor.task_queue_manager import TaskQueueManager from ansible.executor.task_queue_manager import TaskQueueManager
@ -37,10 +25,9 @@ class AdHocCLI(CLI):
this command allows you to define and run a single task 'playbook' against a set of hosts this command allows you to define and run a single task 'playbook' against a set of hosts
''' '''
def parse(self): def init_parser(self):
''' create an options parser for bin/ansible ''' ''' create an options parser for bin/ansible '''
self.parser = super(AdHocCLI, self).init_parser(
self.parser = CLI.base_parser(
usage='%prog <host-pattern> [options]', usage='%prog <host-pattern> [options]',
runas_opts=True, runas_opts=True,
inventory_opts=True, inventory_opts=True,
@ -63,24 +50,32 @@ class AdHocCLI(CLI):
self.parser.add_option('-m', '--module-name', dest='module_name', self.parser.add_option('-m', '--module-name', dest='module_name',
help="module name to execute (default=%s)" % C.DEFAULT_MODULE_NAME, help="module name to execute (default=%s)" % C.DEFAULT_MODULE_NAME,
default=C.DEFAULT_MODULE_NAME) default=C.DEFAULT_MODULE_NAME)
return self.parser
super(AdHocCLI, self).parse() def post_process_args(self, options, args):
'''Post process and validate options for bin/ansible '''
if len(self.args) < 1: options, args = super(AdHocCLI, self).post_process_args(options, args)
if len(args) < 1:
raise AnsibleOptionsError("Missing target hosts") raise AnsibleOptionsError("Missing target hosts")
elif len(self.args) > 1: elif len(args) > 1:
raise AnsibleOptionsError("Extraneous options or arguments") raise AnsibleOptionsError("Extraneous options or arguments")
display.verbosity = self.options.verbosity display.verbosity = options.verbosity
self.validate_conflicts(runas_opts=True, vault_opts=True, fork_opts=True) self.validate_conflicts(options, runas_opts=True, vault_opts=True, fork_opts=True)
options = self.normalize_become_options(options)
return options, args
def _play_ds(self, pattern, async_val, poll): def _play_ds(self, pattern, async_val, poll):
check_raw = self.options.module_name in ('command', 'win_command', 'shell', 'win_shell', 'script', 'raw') check_raw = context.CLIARGS['module_name'] in ('command', 'win_command', 'shell', 'win_shell', 'script', 'raw')
mytask = {'action': {'module': self.options.module_name, 'args': parse_kv(self.options.module_args, check_raw=check_raw)}} mytask = {'action': {'module': context.CLIARGS['module_name'], 'args': parse_kv(context.CLIARGS['module_args'], check_raw=check_raw)}}
# avoid adding to tasks that don't support it, unless set, then give user an error # avoid adding to tasks that don't support it, unless set, then give user an error
if self.options.module_name not in ('include_role', 'include_tasks') or any(frozenset((async_val, poll))): if context.CLIARGS['module_name'] not in ('include_role', 'include_tasks') or any(frozenset((async_val, poll))):
mytask['async_val'] = async_val mytask['async_val'] = async_val
mytask['poll'] = poll mytask['poll'] = poll
@ -96,46 +91,46 @@ class AdHocCLI(CLI):
super(AdHocCLI, self).run() super(AdHocCLI, self).run()
# only thing left should be host pattern # only thing left should be host pattern
pattern = to_text(self.args[0], errors='surrogate_or_strict') pattern = to_text(context.CLIARGS['args'][0], errors='surrogate_or_strict')
sshpass = None sshpass = None
becomepass = None becomepass = None
self.normalize_become_options()
(sshpass, becomepass) = self.ask_passwords() (sshpass, becomepass) = self.ask_passwords()
passwords = {'conn_pass': sshpass, 'become_pass': becomepass} passwords = {'conn_pass': sshpass, 'become_pass': becomepass}
# dynamically load any plugins # dynamically load any plugins
get_all_plugin_loaders() get_all_plugin_loaders()
loader, inventory, variable_manager = self._play_prereqs(self.options) loader, inventory, variable_manager = self._play_prereqs()
try: try:
hosts = CLI.get_host_list(inventory, self.options.subset, pattern) hosts = self.get_host_list(inventory, context.CLIARGS['subset'], pattern)
except AnsibleError: except AnsibleError:
if self.options.subset: if context.CLIARGS['subset']:
raise raise
else: else:
hosts = [] hosts = []
display.warning("No hosts matched, nothing to do") display.warning("No hosts matched, nothing to do")
if self.options.listhosts: if context.CLIARGS['listhosts']:
display.display(' hosts (%d):' % len(hosts)) display.display(' hosts (%d):' % len(hosts))
for host in hosts: for host in hosts:
display.display(' %s' % host) display.display(' %s' % host)
return 0 return 0
if self.options.module_name in C.MODULE_REQUIRE_ARGS and not self.options.module_args: if context.CLIARGS['module_name'] in C.MODULE_REQUIRE_ARGS and not context.CLIARGS['module_args']:
err = "No argument passed to %s module" % self.options.module_name err = "No argument passed to %s module" % context.CLIARGS['module_name']
if pattern.endswith(".yml"): if pattern.endswith(".yml"):
err = err + ' (did you mean to run ansible-playbook?)' err = err + ' (did you mean to run ansible-playbook?)'
raise AnsibleOptionsError(err) raise AnsibleOptionsError(err)
# Avoid modules that don't work with ad-hoc # Avoid modules that don't work with ad-hoc
if self.options.module_name in ('import_playbook',): if context.CLIARGS['module_name'] in ('import_playbook',):
raise AnsibleOptionsError("'%s' is not a valid action for ad-hoc commands" % self.options.module_name) raise AnsibleOptionsError("'%s' is not a valid action for ad-hoc commands"
% context.CLIARGS['module_name'])
play_ds = self._play_ds(pattern, self.options.seconds, self.options.poll_interval) play_ds = self._play_ds(pattern, context.CLIARGS['seconds'], context.CLIARGS['poll_interval'])
play = Play().load(play_ds, variable_manager=variable_manager, loader=loader) play = Play().load(play_ds, variable_manager=variable_manager, loader=loader)
# used in start callback # used in start callback
@ -145,7 +140,7 @@ class AdHocCLI(CLI):
if self.callback: if self.callback:
cb = self.callback cb = self.callback
elif self.options.one_line: elif context.CLIARGS['one_line']:
cb = 'oneline' cb = 'oneline'
# Respect custom 'stdout_callback' only with enabled 'bin_ansible_callbacks' # Respect custom 'stdout_callback' only with enabled 'bin_ansible_callbacks'
elif C.DEFAULT_LOAD_CALLBACK_PLUGINS and C.DEFAULT_STDOUT_CALLBACK != 'default': elif C.DEFAULT_LOAD_CALLBACK_PLUGINS and C.DEFAULT_STDOUT_CALLBACK != 'default':
@ -154,9 +149,9 @@ class AdHocCLI(CLI):
cb = 'minimal' cb = 'minimal'
run_tree = False run_tree = False
if self.options.tree: if context.CLIARGS['tree']:
C.DEFAULT_CALLBACK_WHITELIST.append('tree') C.DEFAULT_CALLBACK_WHITELIST.append('tree')
C.TREE_DIR = self.options.tree C.TREE_DIR = context.CLIARGS['tree']
run_tree = True run_tree = True
# now create a task queue manager to execute the play # now create a task queue manager to execute the play
@ -166,11 +161,11 @@ class AdHocCLI(CLI):
inventory=inventory, inventory=inventory,
variable_manager=variable_manager, variable_manager=variable_manager,
loader=loader, loader=loader,
options=self.options,
passwords=passwords, passwords=passwords,
stdout_callback=cb, stdout_callback=cb,
run_additional_callbacks=C.DEFAULT_LOAD_CALLBACK_PLUGINS, run_additional_callbacks=C.DEFAULT_LOAD_CALLBACK_PLUGINS,
run_tree=run_tree, run_tree=run_tree,
forks=context.CLIARGS['forks'],
) )
self._tqm.send_callback('v2_playbook_on_start', playbook) self._tqm.send_callback('v2_playbook_on_start', playbook)

View file

@ -1,4 +1,4 @@
# Copyright: (c) 2017, Ansible Project # Copyright: (c) 2017-2018, Ansible Project
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) # 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) from __future__ import (absolute_import, division, print_function)
@ -10,6 +10,7 @@ import subprocess
import sys import sys
import yaml import yaml
from ansible import context
from ansible.cli import CLI from ansible.cli import CLI
from ansible.config.manager import ConfigManager, Setting, find_ini_config_file from ansible.config.manager import ConfigManager, Setting, find_ini_config_file
from ansible.errors import AnsibleError, AnsibleOptionsError from ansible.errors import AnsibleError, AnsibleOptionsError
@ -33,9 +34,9 @@ class ConfigCLI(CLI):
self.config = None self.config = None
super(ConfigCLI, self).__init__(args, callback) super(ConfigCLI, self).__init__(args, callback)
def parse(self): def init_parser(self):
self.parser = CLI.base_parser( self.parser = super(ConfigCLI, self).init_parser(
usage="usage: %%prog [%s] [--help] [options] [ansible.cfg]" % "|".join(sorted(self.VALID_ACTIONS)), usage="usage: %%prog [%s] [--help] [options] [ansible.cfg]" % "|".join(sorted(self.VALID_ACTIONS)),
epilog="\nSee '%s <command> --help' for more information on a specific command.\n\n" % os.path.basename(sys.argv[0]), epilog="\nSee '%s <command> --help' for more information on a specific command.\n\n" % os.path.basename(sys.argv[0]),
desc="View, edit, and manage ansible configuration.", desc="View, edit, and manage ansible configuration.",
@ -56,15 +57,20 @@ class ConfigCLI(CLI):
elif self.action == "search": elif self.action == "search":
self.parser.set_usage("usage: %prog update [options] [-c ansible.cfg] <search term>") self.parser.set_usage("usage: %prog update [options] [-c ansible.cfg] <search term>")
self.options, self.args = self.parser.parse_args() return self.parser
display.verbosity = self.options.verbosity
def post_process_args(self, options, args):
super(ConfigCLI, self).post_process_args(options, args)
display.verbosity = options.verbosity
return options, args
def run(self): def run(self):
super(ConfigCLI, self).run() super(ConfigCLI, self).run()
if self.options.config_file: if context.CLIARGS['config_file']:
self.config_file = unfrackpath(self.options.config_file, follow=False) self.config_file = unfrackpath(context.CLIARGS['config_file'], follow=False)
self.config = ConfigManager(self.config_file) self.config = ConfigManager(self.config_file)
else: else:
self.config = ConfigManager() self.config = ConfigManager()
@ -96,10 +102,10 @@ class ConfigCLI(CLI):
raise AnsibleError("Option not implemented yet") raise AnsibleError("Option not implemented yet")
# pylint: disable=unreachable # pylint: disable=unreachable
if self.options.setting is None: if context.CLIARGS['setting'] is None:
raise AnsibleOptionsError("update option requires a setting to update") raise AnsibleOptionsError("update option requires a setting to update")
(entry, value) = self.options.setting.split('=') (entry, value) = context.CLIARGS['setting'].split('=')
if '.' in entry: if '.' in entry:
(section, option) = entry.split('.') (section, option) = entry.split('.')
else: else:
@ -164,7 +170,7 @@ class ConfigCLI(CLI):
else: else:
color = 'green' color = 'green'
msg = "%s(%s) = %s" % (setting, 'default', defaults[setting].get('default')) msg = "%s(%s) = %s" % (setting, 'default', defaults[setting].get('default'))
if not self.options.only_changed or color == 'yellow': if not context.CLIARGS['only_changed'] or color == 'yellow':
text.append(stringc(msg, color)) text.append(stringc(msg, color))
self.pager(to_text('\n'.join(text), errors='surrogate_or_strict')) self.pager(to_text('\n'.join(text), errors='surrogate_or_strict'))

View file

@ -1,19 +1,7 @@
# (c) 2014, Nandor Sivok <dominis@haxor.hu> # Copyright: (c) 2014, Nandor Sivok <dominis@haxor.hu>
# (c) 2016, Redhat Inc # Copyright: (c) 2016, Redhat Inc
# # Copyright: (c) 2018, Ansible Project
# ansible-console is free software: you can redistribute it and/or modify # GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
# 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.
#
# ansible-console 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 Ansible. If not, see <http://www.gnu.org/licenses/>.
#
from __future__ import (absolute_import, division, print_function) from __future__ import (absolute_import, division, print_function)
__metaclass__ = type __metaclass__ = type
@ -37,6 +25,7 @@ import os
import sys import sys
from ansible import constants as C from ansible import constants as C
from ansible import context
from ansible.cli import CLI from ansible.cli import CLI
from ansible.executor.task_queue_manager import TaskQueueManager from ansible.executor.task_queue_manager import TaskQueueManager
from ansible.module_utils._text import to_native, to_text from ansible.module_utils._text import to_native, to_text
@ -75,10 +64,21 @@ class ConsoleCLI(CLI, cmd.Cmd):
self.passwords = dict() self.passwords = dict()
self.modules = None self.modules = None
self.cwd = '*'
# Defaults for these are set from the CLI in run()
self.remote_user = None
self.become = None
self.become_user = None
self.become_method = None
self.check_mode = None
self.diff = None
self.forks = None
cmd.Cmd.__init__(self) cmd.Cmd.__init__(self)
def parse(self): def init_parser(self):
self.parser = CLI.base_parser( super(ConsoleCLI, self).init_parser(
usage='%prog [<host-pattern>] [options]', usage='%prog [<host-pattern>] [options]',
runas_opts=True, runas_opts=True,
inventory_opts=True, inventory_opts=True,
@ -96,12 +96,14 @@ class ConsoleCLI(CLI, cmd.Cmd):
self.parser.add_option('--step', dest='step', action='store_true', self.parser.add_option('--step', dest='step', action='store_true',
help="one-step-at-a-time: confirm each task before running") help="one-step-at-a-time: confirm each task before running")
self.parser.set_defaults(cwd='*') return self.parser
super(ConsoleCLI, self).parse() def post_process_args(self, options, args):
options, args = super(ConsoleCLI, self).post_process_args(options, args)
display.verbosity = self.options.verbosity display.verbosity = options.verbosity
self.validate_conflicts(runas_opts=True, vault_opts=True, fork_opts=True) options = self.normalize_become_options(options)
self.validate_conflicts(options, runas_opts=True, vault_opts=True, fork_opts=True)
return options, args
def get_names(self): def get_names(self):
return dir(self) return dir(self)
@ -113,10 +115,10 @@ class ConsoleCLI(CLI, cmd.Cmd):
self.do_exit(self) self.do_exit(self)
def set_prompt(self): def set_prompt(self):
login_user = self.options.remote_user or getpass.getuser() login_user = self.remote_user or getpass.getuser()
self.selected = self.inventory.list_hosts(self.options.cwd) self.selected = self.inventory.list_hosts(self.cwd)
prompt = "%s@%s (%d)[f:%s]" % (login_user, self.options.cwd, len(self.selected), self.options.forks) prompt = "%s@%s (%d)[f:%s]" % (login_user, self.cwd, len(self.selected), self.forks)
if self.options.become and self.options.become_user in [None, 'root']: if self.become and self.become_user in [None, 'root']:
prompt += "# " prompt += "# "
color = C.COLOR_ERROR color = C.COLOR_ERROR
else: else:
@ -126,8 +128,8 @@ class ConsoleCLI(CLI, cmd.Cmd):
def list_modules(self): def list_modules(self):
modules = set() modules = set()
if self.options.module_path: if context.CLIARGS['module_path']:
for path in self.options.module_path: for path in context.CLIARGS['module_path']:
if path: if path:
module_loader.add_directory(path) module_loader.add_directory(path)
@ -165,7 +167,7 @@ class ConsoleCLI(CLI, cmd.Cmd):
if arg.startswith("#"): if arg.startswith("#"):
return False return False
if not self.options.cwd: if not self.cwd:
display.error("No host found") display.error("No host found")
return False return False
@ -180,16 +182,20 @@ class ConsoleCLI(CLI, cmd.Cmd):
module = 'shell' module = 'shell'
module_args = arg module_args = arg
self.options.module_name = module
result = None result = None
try: try:
check_raw = self.options.module_name in ('command', 'shell', 'script', 'raw') check_raw = module in ('command', 'shell', 'script', 'raw')
play_ds = dict( play_ds = dict(
name="Ansible Shell", name="Ansible Shell",
hosts=self.options.cwd, hosts=self.cwd,
gather_facts='no', gather_facts='no',
tasks=[dict(action=dict(module=module, args=parse_kv(module_args, check_raw=check_raw)))] tasks=[dict(action=dict(module=module, args=parse_kv(module_args, check_raw=check_raw)))],
remote_user=self.remote_user,
become=self.become,
become_user=self.become_user,
become_method=self.become_method,
check_mode=self.check_mode,
diff=self.diff,
) )
play = Play().load(play_ds, variable_manager=self.variable_manager, loader=self.loader) play = Play().load(play_ds, variable_manager=self.variable_manager, loader=self.loader)
except Exception as e: except Exception as e:
@ -205,11 +211,11 @@ class ConsoleCLI(CLI, cmd.Cmd):
inventory=self.inventory, inventory=self.inventory,
variable_manager=self.variable_manager, variable_manager=self.variable_manager,
loader=self.loader, loader=self.loader,
options=self.options,
passwords=self.passwords, passwords=self.passwords,
stdout_callback=cb, stdout_callback=cb,
run_additional_callbacks=C.DEFAULT_LOAD_CALLBACK_PLUGINS, run_additional_callbacks=C.DEFAULT_LOAD_CALLBACK_PLUGINS,
run_tree=False, run_tree=False,
forks=self.forks,
) )
result = self._tqm.run(play) result = self._tqm.run(play)
@ -252,7 +258,13 @@ class ConsoleCLI(CLI, cmd.Cmd):
if not arg: if not arg:
display.display('Usage: forks <number>') display.display('Usage: forks <number>')
return return
self.options.forks = int(arg)
forks = int(arg)
if forks <= 0:
display.display('forks must be greater than or equal to 1')
return
self.forks = forks
self.set_prompt() self.set_prompt()
do_serial = do_forks do_serial = do_forks
@ -275,11 +287,11 @@ class ConsoleCLI(CLI, cmd.Cmd):
cd webservers:dbservers:&staging:!phoenix cd webservers:dbservers:&staging:!phoenix
""" """
if not arg: if not arg:
self.options.cwd = '*' self.cwd = '*'
elif arg in '/*': elif arg in '/*':
self.options.cwd = 'all' self.cwd = 'all'
elif self.inventory.get_hosts(arg): elif self.inventory.get_hosts(arg):
self.options.cwd = arg self.cwd = arg
else: else:
display.display("no host matched") display.display("no host matched")
@ -297,8 +309,8 @@ class ConsoleCLI(CLI, cmd.Cmd):
def do_become(self, arg): def do_become(self, arg):
"""Toggle whether plays run with become""" """Toggle whether plays run with become"""
if arg: if arg:
self.options.become = boolean(arg, strict=False) self.become = boolean(arg, strict=False)
display.v("become changed to %s" % self.options.become) display.v("become changed to %s" % self.become)
self.set_prompt() self.set_prompt()
else: else:
display.display("Please specify become value, e.g. `become yes`") display.display("Please specify become value, e.g. `become yes`")
@ -306,7 +318,7 @@ class ConsoleCLI(CLI, cmd.Cmd):
def do_remote_user(self, arg): def do_remote_user(self, arg):
"""Given a username, set the remote user plays are run by""" """Given a username, set the remote user plays are run by"""
if arg: if arg:
self.options.remote_user = arg self.remote_user = arg
self.set_prompt() self.set_prompt()
else: else:
display.display("Please specify a remote user, e.g. `remote_user root`") display.display("Please specify a remote user, e.g. `remote_user root`")
@ -314,33 +326,33 @@ class ConsoleCLI(CLI, cmd.Cmd):
def do_become_user(self, arg): def do_become_user(self, arg):
"""Given a username, set the user that plays are run by when using become""" """Given a username, set the user that plays are run by when using become"""
if arg: if arg:
self.options.become_user = arg self.become_user = arg
else: else:
display.display("Please specify a user, e.g. `become_user jenkins`") display.display("Please specify a user, e.g. `become_user jenkins`")
display.v("Current user is %s" % self.options.become_user) display.v("Current user is %s" % self.become_user)
self.set_prompt() self.set_prompt()
def do_become_method(self, arg): def do_become_method(self, arg):
"""Given a become_method, set the privilege escalation method when using become""" """Given a become_method, set the privilege escalation method when using become"""
if arg: if arg:
self.options.become_method = arg self.become_method = arg
display.v("become_method changed to %s" % self.options.become_method) display.v("become_method changed to %s" % self.become_method)
else: else:
display.display("Please specify a become_method, e.g. `become_method su`") display.display("Please specify a become_method, e.g. `become_method su`")
def do_check(self, arg): def do_check(self, arg):
"""Toggle whether plays run with check mode""" """Toggle whether plays run with check mode"""
if arg: if arg:
self.options.check = boolean(arg, strict=False) self.check_mode = boolean(arg, strict=False)
display.v("check mode changed to %s" % self.options.check) display.v("check mode changed to %s" % self.check_mode)
else: else:
display.display("Please specify check mode value, e.g. `check yes`") display.display("Please specify check mode value, e.g. `check yes`")
def do_diff(self, arg): def do_diff(self, arg):
"""Toggle whether plays run with diff""" """Toggle whether plays run with diff"""
if arg: if arg:
self.options.diff = boolean(arg, strict=False) self.diff = boolean(arg, strict=False)
display.v("diff mode changed to %s" % self.options.diff) display.v("diff mode changed to %s" % self.diff)
else: else:
display.display("Please specify a diff value , e.g. `diff yes`") display.display("Please specify a diff value , e.g. `diff yes`")
@ -370,10 +382,10 @@ class ConsoleCLI(CLI, cmd.Cmd):
mline = line.partition(' ')[2] mline = line.partition(' ')[2]
offs = len(mline) - len(text) offs = len(mline) - len(text)
if self.options.cwd in ('all', '*', '\\'): if self.cwd in ('all', '*', '\\'):
completions = self.hosts + self.groups completions = self.hosts + self.groups
else: else:
completions = [x.name for x in self.inventory.list_hosts(self.options.cwd)] completions = [x.name for x in self.inventory.list_hosts(self.cwd)]
return [to_native(s)[offs:] for s in completions if to_native(s).startswith(to_native(mline))] return [to_native(s)[offs:] for s in completions if to_native(s).startswith(to_native(mline))]
@ -398,11 +410,20 @@ class ConsoleCLI(CLI, cmd.Cmd):
becomepass = None becomepass = None
# hosts # hosts
if len(self.args) != 1: if len(context.CLIARGS['args']) != 1:
self.pattern = 'all' self.pattern = 'all'
else: else:
self.pattern = self.args[0] self.pattern = context.CLIARGS['args'][0]
self.options.cwd = self.pattern self.cwd = self.pattern
# Defaults from the command line
self.remote_user = context.CLIARGS['remote_user']
self.become = context.CLIARGS['become']
self.become_user = context.CLIARGS['become_user']
self.become_method = context.CLIARGS['become_method']
self.check_mode = context.CLIARGS['check']
self.diff = context.CLIARGS['diff']
self.forks = context.CLIARGS['forks']
# dynamically add modules as commands # dynamically add modules as commands
self.modules = self.list_modules() self.modules = self.list_modules()
@ -410,13 +431,12 @@ class ConsoleCLI(CLI, cmd.Cmd):
setattr(self, 'do_' + module, lambda arg, module=module: self.default(module + ' ' + arg)) setattr(self, 'do_' + module, lambda arg, module=module: self.default(module + ' ' + arg))
setattr(self, 'help_' + module, lambda module=module: self.helpdefault(module)) setattr(self, 'help_' + module, lambda module=module: self.helpdefault(module))
self.normalize_become_options()
(sshpass, becomepass) = self.ask_passwords() (sshpass, becomepass) = self.ask_passwords()
self.passwords = {'conn_pass': sshpass, 'become_pass': becomepass} self.passwords = {'conn_pass': sshpass, 'become_pass': becomepass}
self.loader, self.inventory, self.variable_manager = self._play_prereqs(self.options) self.loader, self.inventory, self.variable_manager = self._play_prereqs()
hosts = CLI.get_host_list(self.inventory, self.options.subset, self.pattern) hosts = self.get_host_list(self.inventory, context.CLIARGS['subset'], self.pattern)
self.groups = self.inventory.list_groups() self.groups = self.inventory.list_groups()
self.hosts = [x.name for x in hosts] self.hosts = [x.name for x in hosts]

View file

@ -1,4 +1,5 @@
# Copyright: (c) 2014, James Tanner <tanner.jc@gmail.com> # Copyright: (c) 2014, James Tanner <tanner.jc@gmail.com>
# Copyright: (c) 2018, Ansible Project
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) # 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) from __future__ import (absolute_import, division, print_function)
@ -14,6 +15,7 @@ import yaml
import ansible.plugins.loader as plugin_loader import ansible.plugins.loader as plugin_loader
from ansible import constants as C from ansible import constants as C
from ansible import context
from ansible.cli import CLI from ansible.cli import CLI
from ansible.errors import AnsibleError, AnsibleOptionsError from ansible.errors import AnsibleError, AnsibleOptionsError
from ansible.module_utils._text import to_native from ansible.module_utils._text import to_native
@ -43,9 +45,9 @@ class DocCLI(CLI):
super(DocCLI, self).__init__(args) super(DocCLI, self).__init__(args)
self.plugin_list = set() self.plugin_list = set()
def parse(self): def init_parser(self):
self.parser = CLI.base_parser( self.parser = super(DocCLI, self).init_parser(
usage='usage: %prog [-l|-F|-s] [options] [-t <plugin type> ] [plugin]', usage='usage: %prog [-l|-F|-s] [options] [-t <plugin type> ] [plugin]',
module_opts=True, module_opts=True,
desc="plugin documentation tool", desc="plugin documentation tool",
@ -66,18 +68,27 @@ class DocCLI(CLI):
help='Choose which plugin type (defaults to "module"). ' help='Choose which plugin type (defaults to "module"). '
'Available plugin types are : {0}'.format(C.DOCUMENTABLE_PLUGINS), 'Available plugin types are : {0}'.format(C.DOCUMENTABLE_PLUGINS),
choices=C.DOCUMENTABLE_PLUGINS) choices=C.DOCUMENTABLE_PLUGINS)
super(DocCLI, self).parse() return self.parser
if [self.options.all_plugins, self.options.json_dump, self.options.list_dir, self.options.list_files, self.options.show_snippet].count(True) > 1: def post_process_args(self, options, args):
if [options.all_plugins, options.json_dump, options.list_dir, options.list_files, options.show_snippet].count(True) > 1:
raise AnsibleOptionsError("Only one of -l, -F, -s, -j or -a can be used at the same time.") raise AnsibleOptionsError("Only one of -l, -F, -s, -j or -a can be used at the same time.")
display.verbosity = self.options.verbosity display.verbosity = options.verbosity
# process all plugins of type
if options.all_plugins:
args = self.get_all_plugins_of_type(options['type'])
if options.module_path:
display.warning('Ignoring "--module-path/-M" option as "--all/-a" only displays builtins')
return options, args
def run(self): def run(self):
super(DocCLI, self).run() super(DocCLI, self).run()
plugin_type = self.options.type plugin_type = context.CLIARGS['type']
if plugin_type in C.DOCUMENTABLE_PLUGINS: if plugin_type in C.DOCUMENTABLE_PLUGINS:
loader = getattr(plugin_loader, '%s_loader' % plugin_type) loader = getattr(plugin_loader, '%s_loader' % plugin_type)
@ -85,17 +96,17 @@ class DocCLI(CLI):
raise AnsibleOptionsError("Unknown or undocumentable plugin type: %s" % plugin_type) raise AnsibleOptionsError("Unknown or undocumentable plugin type: %s" % plugin_type)
# add to plugin path from command line # add to plugin path from command line
if self.options.module_path: if context.CLIARGS['module_path']:
for path in self.options.module_path: for path in context.CLIARGS['module_path']:
if path: if path:
loader.add_directory(path) loader.add_directory(path)
# save only top level paths for errors # save only top level paths for errors
search_paths = DocCLI.print_paths(loader) search_paths = self.print_paths(loader)
loader._paths = None # reset so we can use subdirs below loader._paths = None # reset so we can use subdirs below
# list plugins names and filepath for type # list plugins names and filepath for type
if self.options.list_files: if context.CLIARGS['list_files']:
paths = loader._get_paths() paths = loader._get_paths()
for path in paths: for path in paths:
self.plugin_list.update(self.find_plugins(path, plugin_type)) self.plugin_list.update(self.find_plugins(path, plugin_type))
@ -105,7 +116,7 @@ class DocCLI(CLI):
return 0 return 0
# list plugins for type # list plugins for type
if self.options.list_dir: if context.CLIARGS['list_dir']:
paths = loader._get_paths() paths = loader._get_paths()
for path in paths: for path in paths:
self.plugin_list.update(self.find_plugins(path, plugin_type)) self.plugin_list.update(self.find_plugins(path, plugin_type))
@ -113,14 +124,8 @@ class DocCLI(CLI):
self.pager(self.get_plugin_list_text(loader)) self.pager(self.get_plugin_list_text(loader))
return 0 return 0
# process all plugins of type
if self.options.all_plugins:
self.args = self.get_all_plugins_of_type(plugin_type)
if self.options.module_path:
display.warning('Ignoring "--module-path/-M" option as "--all/-a" only displays builtins')
# dump plugin desc/metadata as JSON # dump plugin desc/metadata as JSON
if self.options.json_dump: if context.CLIARGS['json_dump']:
plugin_data = {} plugin_data = {}
plugin_names = self.get_all_plugins_of_type(plugin_type) plugin_names = self.get_all_plugins_of_type(plugin_type)
for plugin_name in plugin_names: for plugin_name in plugin_names:
@ -132,12 +137,12 @@ class DocCLI(CLI):
return 0 return 0
if len(self.args) == 0: if len(context.CLIARGS['args']) == 0:
raise AnsibleOptionsError("Incorrect options passed") raise AnsibleOptionsError("Incorrect options passed")
# process command line list # process command line list
text = '' text = ''
for plugin in self.args: for plugin in context.CLIARGS['args']:
textret = self.format_plugin_doc(plugin, loader, plugin_type, search_paths) textret = self.format_plugin_doc(plugin, loader, plugin_type, search_paths)
if textret: if textret:
@ -165,7 +170,7 @@ class DocCLI(CLI):
raise AnsibleError("unable to load {0} plugin named {1} ".format(plugin_type, plugin_name)) raise AnsibleError("unable to load {0} plugin named {1} ".format(plugin_type, plugin_name))
try: try:
doc, __, __, metadata = get_docstring(filename, fragment_loader, verbose=(self.options.verbosity > 0)) doc, __, __, metadata = get_docstring(filename, fragment_loader, verbose=(context.CLIARGS['verbosity'] > 0))
except Exception: except Exception:
display.vvv(traceback.format_exc()) display.vvv(traceback.format_exc())
raise AnsibleError( raise AnsibleError(
@ -215,7 +220,7 @@ class DocCLI(CLI):
try: try:
doc, plainexamples, returndocs, metadata = get_docstring(filename, fragment_loader, doc, plainexamples, returndocs, metadata = get_docstring(filename, fragment_loader,
verbose=(self.options.verbosity > 0)) verbose=(context.CLIARGS['verbosity'] > 0))
except Exception: except Exception:
display.vvv(traceback.format_exc()) display.vvv(traceback.format_exc())
display.error( display.error(
@ -242,7 +247,7 @@ class DocCLI(CLI):
if 'docuri' in doc: if 'docuri' in doc:
doc['docuri'] = doc[plugin_type].replace('_', '-') doc['docuri'] = doc[plugin_type].replace('_', '-')
if self.options.show_snippet and plugin_type == 'module': if context.CLIARGS['show_snippet'] and plugin_type == 'module':
text += self.get_snippet_text(doc) text += self.get_snippet_text(doc)
else: else:
text += self.get_man_text(doc) text += self.get_man_text(doc)
@ -516,13 +521,13 @@ class DocCLI(CLI):
def get_man_text(self, doc): def get_man_text(self, doc):
self.IGNORE = self.IGNORE + (self.options.type,) self.IGNORE = self.IGNORE + (context.CLIARGS['type'],)
opt_indent = " " opt_indent = " "
text = [] text = []
pad = display.columns * 0.20 pad = display.columns * 0.20
limit = max(display.columns - int(pad), 70) limit = max(display.columns - int(pad), 70)
text.append("> %s (%s)\n" % (doc.get(self.options.type, doc.get('plugin_type')).upper(), doc.pop('filename'))) text.append("> %s (%s)\n" % (doc.get(context.CLIARGS['type'], doc.get('plugin_type')).upper(), doc.pop('filename')))
if isinstance(doc['description'], list): if isinstance(doc['description'], list):
desc = " ".join(doc.pop('description')) desc = " ".join(doc.pop('description'))

View file

@ -1,23 +1,6 @@
######################################################################## # Copyright: (c) 2013, James Cammarata <jcammarata@ansible.com>
# # Copyright: (c) 2018, Ansible Project
# (C) 2013, James Cammarata <jcammarata@ansible.com> # GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
#
# This file is part of Ansible
#
# Ansible 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.
#
# Ansible 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 Ansible. If not, see <http://www.gnu.org/licenses/>.
#
########################################################################
from __future__ import (absolute_import, division, print_function) from __future__ import (absolute_import, division, print_function)
__metaclass__ = type __metaclass__ = type
@ -31,6 +14,7 @@ import yaml
from jinja2 import Environment, FileSystemLoader from jinja2 import Environment, FileSystemLoader
from ansible import context
import ansible.constants as C import ansible.constants as C
from ansible.cli import CLI from ansible.cli import CLI
from ansible.errors import AnsibleError, AnsibleOptionsError from ansible.errors import AnsibleError, AnsibleOptionsError
@ -131,10 +115,10 @@ class GalaxyCLI(CLI):
if self.action in ("init", "install"): if self.action in ("init", "install"):
self.parser.add_option('-f', '--force', dest='force', action='store_true', default=False, help='Force overwriting an existing role') self.parser.add_option('-f', '--force', dest='force', action='store_true', default=False, help='Force overwriting an existing role')
def parse(self): def init_parser(self):
''' create an options parser for bin/ansible ''' ''' create an options parser for bin/ansible '''
self.parser = CLI.base_parser( self.parser = super(GalaxyCLI, self).init_parser(
usage="usage: %%prog [%s] [--help] [options] ..." % "|".join(sorted(self.VALID_ACTIONS)), usage="usage: %%prog [%s] [--help] [options] ..." % "|".join(sorted(self.VALID_ACTIONS)),
epilog="\nSee '%s <command> --help' for more information on a specific command.\n\n" % os.path.basename(sys.argv[0]), epilog="\nSee '%s <command> --help' for more information on a specific command.\n\n" % os.path.basename(sys.argv[0]),
desc="Perform various Role related operations.", desc="Perform various Role related operations.",
@ -146,15 +130,19 @@ class GalaxyCLI(CLI):
help='Ignore SSL certificate validation errors.') help='Ignore SSL certificate validation errors.')
self.set_action() self.set_action()
super(GalaxyCLI, self).parse() return self.parser
display.verbosity = self.options.verbosity def post_process_args(self, options, args):
self.galaxy = Galaxy(self.options) options, args = super(GalaxyCLI, self).post_process_args(options, args)
display.verbosity = options.verbosity
return options, args
def run(self): def run(self):
super(GalaxyCLI, self).run() super(GalaxyCLI, self).run()
self.galaxy = Galaxy()
self.api = GalaxyAPI(self.galaxy) self.api = GalaxyAPI(self.galaxy)
self.execute() self.execute()
@ -163,7 +151,7 @@ class GalaxyCLI(CLI):
Exits with the specified return code unless the Exits with the specified return code unless the
option --ignore-errors was specified option --ignore-errors was specified
""" """
if not self.options.ignore_errors: if not context.CLIARGS['ignore_errors']:
raise AnsibleError('- you can use --ignore-errors to skip failed roles and finish processing the list.') raise AnsibleError('- you can use --ignore-errors to skip failed roles and finish processing the list.')
def _display_role_info(self, role_info): def _display_role_info(self, role_info):
@ -196,11 +184,11 @@ class GalaxyCLI(CLI):
creates the skeleton framework of a role that complies with the galaxy metadata format. creates the skeleton framework of a role that complies with the galaxy metadata format.
""" """
init_path = self.options.init_path init_path = context.CLIARGS['init_path']
force = self.options.force force = context.CLIARGS['force']
role_skeleton = self.options.role_skeleton role_skeleton = context.CLIARGS['role_skeleton']
role_name = self.args.pop(0).strip() if self.args else None role_name = context.CLIARGS['args'][0].strip() if context.CLIARGS['args'] else None
if not role_name: if not role_name:
raise AnsibleOptionsError("- no role name specified for init") raise AnsibleOptionsError("- no role name specified for init")
role_path = os.path.join(init_path, role_name) role_path = os.path.join(init_path, role_name)
@ -221,7 +209,7 @@ class GalaxyCLI(CLI):
license='license (GPLv2, CC-BY, etc)', license='license (GPLv2, CC-BY, etc)',
issue_tracker_url='http://example.com/issue/tracker', issue_tracker_url='http://example.com/issue/tracker',
min_ansible_version='2.4', min_ansible_version='2.4',
role_type=self.options.role_type role_type=context.CLIARGS['role_type']
) )
# create role directory # create role directory
@ -268,14 +256,14 @@ class GalaxyCLI(CLI):
prints out detailed information about an installed role as well as info available from the galaxy API. prints out detailed information about an installed role as well as info available from the galaxy API.
""" """
if len(self.args) == 0: if not context.CLIARGS['args']:
# the user needs to specify a role # the user needs to specify a role
raise AnsibleOptionsError("- you must specify a user/role name") raise AnsibleOptionsError("- you must specify a user/role name")
roles_path = self.options.roles_path roles_path = context.CLIARGS['roles_path']
data = '' data = ''
for role in self.args: for role in context.CLIARGS['args']:
role_info = {'path': roles_path} role_info = {'path': roles_path}
gr = GalaxyRole(self.galaxy, role) gr = GalaxyRole(self.galaxy, role)
@ -288,7 +276,7 @@ class GalaxyCLI(CLI):
role_info.update(install_info) role_info.update(install_info)
remote_data = False remote_data = False
if not self.options.offline: if not context.CLIARGS['offline']:
remote_data = self.api.lookup_role_by_name(role, False) remote_data = self.api.lookup_role_by_name(role, False)
if remote_data: if remote_data:
@ -315,14 +303,14 @@ class GalaxyCLI(CLI):
uses the args list of roles to be installed, unless -f was specified. The list of roles uses the args list of roles to be installed, unless -f was specified. The list of roles
can be a name (which will be downloaded via the galaxy API and github), or it can be a local .tar.gz file. can be a name (which will be downloaded via the galaxy API and github), or it can be a local .tar.gz file.
""" """
role_file = self.options.role_file role_file = context.CLIARGS['role_file']
if len(self.args) == 0 and role_file is None: if not context.CLIARGS['args'] and role_file is None:
# the user needs to specify one of either --role-file or specify a single user/role name # the user needs to specify one of either --role-file or specify a single user/role name
raise AnsibleOptionsError("- you must specify a user/role name or a roles file") raise AnsibleOptionsError("- you must specify a user/role name or a roles file")
no_deps = self.options.no_deps no_deps = context.CLIARGS['no_deps']
force = self.options.force force = context.CLIARGS['force']
roles_left = [] roles_left = []
if role_file: if role_file:
@ -362,13 +350,13 @@ class GalaxyCLI(CLI):
else: else:
# roles were specified directly, so we'll just go out grab them # roles were specified directly, so we'll just go out grab them
# (and their dependencies, unless the user doesn't want us to). # (and their dependencies, unless the user doesn't want us to).
for rname in self.args: for rname in context.CLIARGS['args']:
role = RoleRequirement.role_yaml_parse(rname.strip()) role = RoleRequirement.role_yaml_parse(rname.strip())
roles_left.append(GalaxyRole(self.galaxy, **role)) roles_left.append(GalaxyRole(self.galaxy, **role))
for role in roles_left: for role in roles_left:
# only process roles in roles files when names matches if given # only process roles in roles files when names matches if given
if role_file and self.args and role.name not in self.args: if role_file and context.CLIARGS['args'] and role.name not in context.CLIARGS['args']:
display.vvv('Skipping role %s' % role.name) display.vvv('Skipping role %s' % role.name)
continue continue
@ -437,10 +425,10 @@ class GalaxyCLI(CLI):
removes the list of roles passed as arguments from the local system. removes the list of roles passed as arguments from the local system.
""" """
if len(self.args) == 0: if not context.CLIARGS['args']:
raise AnsibleOptionsError('- you must specify at least one role to remove.') raise AnsibleOptionsError('- you must specify at least one role to remove.')
for role_name in self.args: for role_name in context.CLIARGS['args']:
role = GalaxyRole(self.galaxy, role_name) role = GalaxyRole(self.galaxy, role_name)
try: try:
if role.remove(): if role.remove():
@ -457,7 +445,7 @@ class GalaxyCLI(CLI):
lists the roles installed on the local system or matches a single role passed as an argument. lists the roles installed on the local system or matches a single role passed as an argument.
""" """
if len(self.args) > 1: if len(context.CLIARGS['args']) > 1:
raise AnsibleOptionsError("- please specify only one role to list, or specify no roles to see a full list") raise AnsibleOptionsError("- please specify only one role to list, or specify no roles to see a full list")
def _display_role(gr): def _display_role(gr):
@ -469,9 +457,9 @@ class GalaxyCLI(CLI):
version = "(unknown version)" version = "(unknown version)"
display.display("- %s, %s" % (gr.name, version)) display.display("- %s, %s" % (gr.name, version))
if len(self.args) == 1: if context.CLIARGS['args']:
# show the requested role, if it exists # show the requested role, if it exists
name = self.args.pop() name = context.CLIARGS['args'][0]
gr = GalaxyRole(self.galaxy, name) gr = GalaxyRole(self.galaxy, name)
if gr.metadata: if gr.metadata:
display.display('# %s' % os.path.dirname(gr.path)) display.display('# %s' % os.path.dirname(gr.path))
@ -480,7 +468,7 @@ class GalaxyCLI(CLI):
display.display("- the role %s was not found" % name) display.display("- the role %s was not found" % name)
else: else:
# show all valid roles in the roles_path directory # show all valid roles in the roles_path directory
roles_path = self.options.roles_path roles_path = context.CLIARGS['roles_path']
path_found = False path_found = False
warnings = [] warnings = []
for path in roles_path: for path in roles_path:
@ -509,17 +497,14 @@ class GalaxyCLI(CLI):
page_size = 1000 page_size = 1000
search = None search = None
if len(self.args): if context.CLIARGS['args']:
terms = [] search = '+'.join(context.CLIARGS['args'])
for i in range(len(self.args)):
terms.append(self.args.pop())
search = '+'.join(terms[::-1])
if not search and not self.options.platforms and not self.options.galaxy_tags and not self.options.author: if not search and not context.CLIARGS['platforms'] and not context.CLIARGS['galaxy_tags'] and not context.CLIARGS['author']:
raise AnsibleError("Invalid query. At least one search term, platform, galaxy tag or author must be provided.") raise AnsibleError("Invalid query. At least one search term, platform, galaxy tag or author must be provided.")
response = self.api.search_roles(search, platforms=self.options.platforms, response = self.api.search_roles(search, platforms=context.CLIARGS['platforms'],
tags=self.options.galaxy_tags, author=self.options.author, page_size=page_size) tags=context.CLIARGS['galaxy_tags'], author=context.CLIARGS['author'], page_size=page_size)
if response['count'] == 0: if response['count'] == 0:
display.display("No roles match your search.", color=C.COLOR_ERROR) display.display("No roles match your search.", color=C.COLOR_ERROR)
@ -553,18 +538,18 @@ class GalaxyCLI(CLI):
verify user's identify via Github and retrieve an auth token from Ansible Galaxy. verify user's identify via Github and retrieve an auth token from Ansible Galaxy.
""" """
# Authenticate with github and retrieve a token # Authenticate with github and retrieve a token
if self.options.token is None: if context.CLIARGS['token'] is None:
if C.GALAXY_TOKEN: if C.GALAXY_TOKEN:
github_token = C.GALAXY_TOKEN github_token = C.GALAXY_TOKEN
else: else:
login = GalaxyLogin(self.galaxy) login = GalaxyLogin(self.galaxy)
github_token = login.create_github_token() github_token = login.create_github_token()
else: else:
github_token = self.options.token github_token = context.CLIARGS['token']
galaxy_response = self.api.authenticate(github_token) galaxy_response = self.api.authenticate(github_token)
if self.options.token is None and C.GALAXY_TOKEN is None: if context.CLIARGS['token'] is None and C.GALAXY_TOKEN is None:
# Remove the token we created # Remove the token we created
login.remove_github_token() login.remove_github_token()
@ -586,17 +571,19 @@ class GalaxyCLI(CLI):
'FAILED': C.COLOR_ERROR, 'FAILED': C.COLOR_ERROR,
} }
if len(self.args) < 2: if len(context.CLIARGS['args']) < 2:
raise AnsibleError("Expected a github_username and github_repository. Use --help.") raise AnsibleError("Expected a github_username and github_repository. Use --help.")
github_repo = to_text(self.args.pop(), errors='surrogate_or_strict') github_user = to_text(context.CLIARGS['args'][0], errors='surrogate_or_strict')
github_user = to_text(self.args.pop(), errors='surrogate_or_strict') github_repo = to_text(context.CLIARGS['args'][1], errors='surrogate_or_strict')
if self.options.check_status: if context.CLIARGS['check_status']:
task = self.api.get_import_task(github_user=github_user, github_repo=github_repo) task = self.api.get_import_task(github_user=github_user, github_repo=github_repo)
else: else:
# Submit an import request # Submit an import request
task = self.api.create_import_task(github_user, github_repo, reference=self.options.reference, role_name=self.options.role_name) task = self.api.create_import_task(github_user, github_repo,
reference=context.CLIARGS['reference'],
role_name=context.CLIARGS['role_name'])
if len(task) > 1: if len(task) > 1:
# found multiple roles associated with github_user/github_repo # found multiple roles associated with github_user/github_repo
@ -610,11 +597,11 @@ class GalaxyCLI(CLI):
return 0 return 0
# found a single role as expected # found a single role as expected
display.display("Successfully submitted import request %d" % task[0]['id']) display.display("Successfully submitted import request %d" % task[0]['id'])
if not self.options.wait: if not context.CLIARGS['wait']:
display.display("Role name: %s" % task[0]['summary_fields']['role']['name']) display.display("Role name: %s" % task[0]['summary_fields']['role']['name'])
display.display("Repo: %s/%s" % (task[0]['github_user'], task[0]['github_repo'])) display.display("Repo: %s/%s" % (task[0]['github_user'], task[0]['github_repo']))
if self.options.check_status or self.options.wait: if context.CLIARGS['check_status'] or context.CLIARGS['wait']:
# Get the status of the import # Get the status of the import
msg_list = [] msg_list = []
finished = False finished = False
@ -634,7 +621,7 @@ class GalaxyCLI(CLI):
def execute_setup(self): def execute_setup(self):
""" Setup an integration from Github or Travis for Ansible Galaxy roles""" """ Setup an integration from Github or Travis for Ansible Galaxy roles"""
if self.options.setup_list: if context.CLIARGS['setup_list']:
# List existing integration secrets # List existing integration secrets
secrets = self.api.list_secrets() secrets = self.api.list_secrets()
if len(secrets) == 0: if len(secrets) == 0:
@ -648,19 +635,19 @@ class GalaxyCLI(CLI):
secret['github_repo']), color=C.COLOR_OK) secret['github_repo']), color=C.COLOR_OK)
return 0 return 0
if self.options.remove_id: if context.CLIARGS['remove_id']:
# Remove a secret # Remove a secret
self.api.remove_secret(self.options.remove_id) self.api.remove_secret(context.CLIARGS['remove_id'])
display.display("Secret removed. Integrations using this secret will not longer work.", color=C.COLOR_OK) display.display("Secret removed. Integrations using this secret will not longer work.", color=C.COLOR_OK)
return 0 return 0
if len(self.args) < 4: if len(context.CLIARGS['args']) < 4:
raise AnsibleError("Missing one or more arguments. Expecting: source github_user github_repo secret") raise AnsibleError("Missing one or more arguments. Expecting: source github_user github_repo secret")
secret = self.args.pop() source = context.CLIARGS['args'][0]
github_repo = self.args.pop() github_user = context.CLIARGS['args'][1]
github_user = self.args.pop() github_repo = context.CLIARGS['args'][2]
source = self.args.pop() secret = context.CLIARGS['args'][3]
resp = self.api.add_secret(source, github_user, github_repo, secret) resp = self.api.add_secret(source, github_user, github_repo, secret)
display.display("Added integration for %s %s/%s" % (resp['source'], resp['github_user'], resp['github_repo'])) display.display("Added integration for %s %s/%s" % (resp['source'], resp['github_user'], resp['github_repo']))
@ -670,11 +657,11 @@ class GalaxyCLI(CLI):
def execute_delete(self): def execute_delete(self):
""" Delete a role from Ansible Galaxy. """ """ Delete a role from Ansible Galaxy. """
if len(self.args) < 2: if len(context.CLIARGS['args']) < 2:
raise AnsibleError("Missing one or more arguments. Expected: github_user github_repo") raise AnsibleError("Missing one or more arguments. Expected: github_user github_repo")
github_repo = self.args.pop() github_user = context.CLIARGS['args'][0]
github_user = self.args.pop() github_repo = context.CLIARGS['args'][1]
resp = self.api.delete_role(github_user, github_repo) resp = self.api.delete_role(github_user, github_repo)
if len(resp['deleted_roles']) > 1: if len(resp['deleted_roles']) > 1:

View file

@ -1,18 +1,6 @@
# (c) 2017, Brian Coca <bcoca@ansible.com> # Copyright: (c) 2017, Brian Coca <bcoca@ansible.com>
# # Copyright: (c) 2018, Ansible Project
# Ansible is free software: you can redistribute it and/or modify # GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
# 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.
#
# Ansible 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 Ansible. If not, see <http://www.gnu.org/licenses/>.
#
from __future__ import (absolute_import, division, print_function) from __future__ import (absolute_import, division, print_function)
__metaclass__ = type __metaclass__ = type
@ -21,6 +9,7 @@ import optparse
from operator import attrgetter from operator import attrgetter
from ansible import constants as C from ansible import constants as C
from ansible import context
from ansible.cli import CLI from ansible.cli import CLI
from ansible.errors import AnsibleError, AnsibleOptionsError from ansible.errors import AnsibleError, AnsibleOptionsError
from ansible.inventory.host import Host from ansible.inventory.host import Host
@ -66,9 +55,9 @@ class InventoryCLI(CLI):
self._new_api = True self._new_api = True
def parse(self): def init_parser(self):
self.parser = CLI.base_parser( self.parser = super(InventoryCLI, self).init_parser(
usage='usage: %prog [options] [host|group]', usage='usage: %prog [options] [host|group]',
epilog='Show Ansible inventory information, by default it uses the inventory script JSON format', epilog='Show Ansible inventory information, by default it uses the inventory script JSON format',
inventory_opts=True, inventory_opts=True,
@ -103,15 +92,15 @@ class InventoryCLI(CLI):
# self.parser.add_option("--ignore-vars-plugins", action="store_true", default=False, dest='ignore_vars_plugins', # self.parser.add_option("--ignore-vars-plugins", action="store_true", default=False, dest='ignore_vars_plugins',
# help="When doing an --list, skip vars data from vars plugins, by default, this would include group_vars/ and host_vars/") # help="When doing an --list, skip vars data from vars plugins, by default, this would include group_vars/ and host_vars/")
super(InventoryCLI, self).parse() return self.parser
display.verbosity = self.options.verbosity def post_process_args(self, options, args):
display.verbosity = options.verbosity
self.validate_conflicts(vault_opts=True) self.validate_conflicts(options, vault_opts=True)
# there can be only one! and, at least, one! # there can be only one! and, at least, one!
used = 0 used = 0
for opt in (self.options.list, self.options.host, self.options.graph): for opt in (options.list, options.host, options.graph):
if opt: if opt:
used += 1 used += 1
if used == 0: if used == 0:
@ -120,22 +109,23 @@ class InventoryCLI(CLI):
raise AnsibleOptionsError("Conflicting options used, only one of --host, --graph or --list can be used at the same time.") raise AnsibleOptionsError("Conflicting options used, only one of --host, --graph or --list can be used at the same time.")
# set host pattern to default if not supplied # set host pattern to default if not supplied
if len(self.args) > 0: if len(args) > 0:
self.options.pattern = self.args[0] options.pattern = args[0]
else: else:
self.options.pattern = 'all' options.pattern = 'all'
return options, args
def run(self): def run(self):
super(InventoryCLI, self).run() super(InventoryCLI, self).run()
results = None
# Initialize needed objects # Initialize needed objects
self.loader, self.inventory, self.vm = self._play_prereqs(self.options) self.loader, self.inventory, self.vm = self._play_prereqs()
if self.options.host: results = None
hosts = self.inventory.get_hosts(self.options.host) if context.CLIARGS['host']:
hosts = self.inventory.get_hosts(context.CLIARGS['host'])
if len(hosts) != 1: if len(hosts) != 1:
raise AnsibleOptionsError("You must pass a single valid host to --host parameter") raise AnsibleOptionsError("You must pass a single valid host to --host parameter")
@ -145,13 +135,13 @@ class InventoryCLI(CLI):
# FIXME: should we template first? # FIXME: should we template first?
results = self.dump(myvars) results = self.dump(myvars)
elif self.options.graph: elif context.CLIARGS['graph']:
results = self.inventory_graph() results = self.inventory_graph()
elif self.options.list: elif context.CLIARGS['list']:
top = self._get_group('all') top = self._get_group('all')
if self.options.yaml: if context.CLIARGS['yaml']:
results = self.yaml_inventory(top) results = self.yaml_inventory(top)
elif self.options.toml: elif context.CLIARGS['toml']:
results = self.toml_inventory(top) results = self.toml_inventory(top)
else: else:
results = self.json_inventory(top) results = self.json_inventory(top)
@ -166,11 +156,11 @@ class InventoryCLI(CLI):
def dump(self, stuff): def dump(self, stuff):
if self.options.yaml: if context.CLIARGS['yaml']:
import yaml import yaml
from ansible.parsing.yaml.dumper import AnsibleDumper from ansible.parsing.yaml.dumper import AnsibleDumper
results = yaml.dump(stuff, Dumper=AnsibleDumper, default_flow_style=False) results = yaml.dump(stuff, Dumper=AnsibleDumper, default_flow_style=False)
elif self.options.toml: elif context.CLIARGS['toml']:
from ansible.plugins.inventory.toml import toml_dumps, HAS_TOML from ansible.plugins.inventory.toml import toml_dumps, HAS_TOML
if not HAS_TOML: if not HAS_TOML:
raise AnsibleError( raise AnsibleError(
@ -227,7 +217,7 @@ class InventoryCLI(CLI):
def _get_host_variables(self, host): def _get_host_variables(self, host):
if self.options.export: if context.CLIARGS['export']:
hostvars = host.get_vars() hostvars = host.get_vars()
# FIXME: add switch to skip vars plugins # FIXME: add switch to skip vars plugins
@ -264,7 +254,7 @@ class InventoryCLI(CLI):
def _show_vars(self, dump, depth): def _show_vars(self, dump, depth):
result = [] result = []
self._remove_internal(dump) self._remove_internal(dump)
if self.options.show_vars: if context.CLIARGS['show_vars']:
for (name, val) in sorted(dump.items()): for (name, val) in sorted(dump.items()):
result.append(self._graph_name('{%s = %s}' % (name, val), depth)) result.append(self._graph_name('{%s = %s}' % (name, val), depth))
return result return result
@ -292,7 +282,7 @@ class InventoryCLI(CLI):
def inventory_graph(self): def inventory_graph(self):
start_at = self._get_group(self.options.pattern) start_at = self._get_group(context.CLIARGS['pattern'])
if start_at: if start_at:
return '\n'.join(self._graph_group(start_at)) return '\n'.join(self._graph_group(start_at))
else: else:
@ -313,7 +303,7 @@ class InventoryCLI(CLI):
if subgroup.name not in seen: if subgroup.name not in seen:
results.update(format_group(subgroup)) results.update(format_group(subgroup))
seen.add(subgroup.name) seen.add(subgroup.name)
if self.options.export: if context.CLIARGS['export']:
results[group.name]['vars'] = self._get_group_variables(group) results[group.name]['vars'] = self._get_group_variables(group)
self._remove_empty(results[group.name]) self._remove_empty(results[group.name])
@ -362,8 +352,7 @@ class InventoryCLI(CLI):
self._remove_internal(myvars) self._remove_internal(myvars)
results[group.name]['hosts'][h.name] = myvars results[group.name]['hosts'][h.name] = myvars
if self.options.export: if context.CLIARGS['export']:
gvars = self._get_group_variables(group) gvars = self._get_group_variables(group)
if gvars: if gvars:
results[group.name]['vars'] = gvars results[group.name]['vars'] = gvars
@ -403,7 +392,7 @@ class InventoryCLI(CLI):
except KeyError: except KeyError:
results[group.name]['hosts'] = {host.name: host_vars} results[group.name]['hosts'] = {host.name: host_vars}
if self.options.export: if context.CLIARGS['export']:
results[group.name]['vars'] = self._get_group_variables(group) results[group.name]['vars'] = self._get_group_variables(group)
self._remove_empty(results[group.name]) self._remove_empty(results[group.name])

View file

@ -21,6 +21,7 @@ __metaclass__ = type
import os import os
import stat import stat
from ansible import context
from ansible.cli import CLI from ansible.cli import CLI
from ansible.errors import AnsibleError, AnsibleOptionsError from ansible.errors import AnsibleError, AnsibleOptionsError
from ansible.executor.playbook_executor import PlaybookExecutor from ansible.executor.playbook_executor import PlaybookExecutor
@ -35,10 +36,10 @@ class PlaybookCLI(CLI):
''' the tool to run *Ansible playbooks*, which are a configuration and multinode deployment system. ''' the tool to run *Ansible playbooks*, which are a configuration and multinode deployment system.
See the project home page (https://docs.ansible.com) for more information. ''' See the project home page (https://docs.ansible.com) for more information. '''
def parse(self): def init_parser(self):
# create parser for CLI options # create parser for CLI options
parser = CLI.base_parser( super(PlaybookCLI, self).init_parser(
usage="%prog [options] playbook.yml [playbook2 ...]", usage="%prog [options] playbook.yml [playbook2 ...]",
connect_opts=True, connect_opts=True,
meta_opts=True, meta_opts=True,
@ -54,49 +55,55 @@ class PlaybookCLI(CLI):
) )
# ansible playbook specific opts # ansible playbook specific opts
parser.add_option('--list-tasks', dest='listtasks', action='store_true', self.parser.add_option('--list-tasks', dest='listtasks', action='store_true',
help="list all tasks that would be executed") help="list all tasks that would be executed")
parser.add_option('--list-tags', dest='listtags', action='store_true', self.parser.add_option('--list-tags', dest='listtags', action='store_true',
help="list all available tags") help="list all available tags")
parser.add_option('--step', dest='step', action='store_true', self.parser.add_option('--step', dest='step', action='store_true',
help="one-step-at-a-time: confirm each task before running") help="one-step-at-a-time: confirm each task before running")
parser.add_option('--start-at-task', dest='start_at_task', self.parser.add_option('--start-at-task', dest='start_at_task',
help="start the playbook at the task matching this name") help="start the playbook at the task matching this name")
self.parser = parser return self.parser
super(PlaybookCLI, self).parse()
if len(self.args) == 0: def post_process_args(self, options, args):
options, args = super(PlaybookCLI, self).post_process_args(options, args)
if len(args) == 0:
raise AnsibleOptionsError("You must specify a playbook file to run") raise AnsibleOptionsError("You must specify a playbook file to run")
display.verbosity = self.options.verbosity display.verbosity = options.verbosity
self.validate_conflicts(runas_opts=True, vault_opts=True, fork_opts=True) self.validate_conflicts(options, runas_opts=True, vault_opts=True, fork_opts=True)
options = self.normalize_become_options(options)
return options, args
def run(self): def run(self):
super(PlaybookCLI, self).run() super(PlaybookCLI, self).run()
# Note: slightly wrong, this is written so that implicit localhost # Note: slightly wrong, this is written so that implicit localhost
# Manage passwords # manages passwords
sshpass = None sshpass = None
becomepass = None becomepass = None
passwords = {} passwords = {}
# initial error check, to make sure all specified playbooks are accessible # initial error check, to make sure all specified playbooks are accessible
# before we start running anything through the playbook executor # before we start running anything through the playbook executor
for playbook in self.args: for playbook in context.CLIARGS['args']:
if not os.path.exists(playbook): if not os.path.exists(playbook):
raise AnsibleError("the playbook: %s could not be found" % playbook) raise AnsibleError("the playbook: %s could not be found" % playbook)
if not (os.path.isfile(playbook) or stat.S_ISFIFO(os.stat(playbook).st_mode)): if not (os.path.isfile(playbook) or stat.S_ISFIFO(os.stat(playbook).st_mode)):
raise AnsibleError("the playbook: %s does not appear to be a file" % playbook) raise AnsibleError("the playbook: %s does not appear to be a file" % playbook)
# don't deal with privilege escalation or passwords when we don't need to # don't deal with privilege escalation or passwords when we don't need to
if not self.options.listhosts and not self.options.listtasks and not self.options.listtags and not self.options.syntax: if not (context.CLIARGS['listhosts'] or context.CLIARGS['listtasks'] or
self.normalize_become_options() context.CLIARGS['listtags'] or context.CLIARGS['syntax']):
(sshpass, becomepass) = self.ask_passwords() (sshpass, becomepass) = self.ask_passwords()
passwords = {'conn_pass': sshpass, 'become_pass': becomepass} passwords = {'conn_pass': sshpass, 'become_pass': becomepass}
loader, inventory, variable_manager = self._play_prereqs(self.options) loader, inventory, variable_manager = self._play_prereqs()
# (which is not returned in list_hosts()) is taken into account for # (which is not returned in list_hosts()) is taken into account for
# warning if inventory is empty. But it can't be taken into account for # warning if inventory is empty. But it can't be taken into account for
@ -104,14 +111,15 @@ class PlaybookCLI(CLI):
# limit if only implicit localhost was in inventory to start with. # limit if only implicit localhost was in inventory to start with.
# #
# Fix this when we rewrite inventory by making localhost a real host (and thus show up in list_hosts()) # Fix this when we rewrite inventory by making localhost a real host (and thus show up in list_hosts())
hosts = CLI.get_host_list(inventory, self.options.subset) hosts = super(PlaybookCLI, self).get_host_list(inventory, context.CLIARGS['subset'])
# flush fact cache if requested # flush fact cache if requested
if self.options.flush_cache: if context.CLIARGS['flush_cache']:
self._flush_cache(inventory, variable_manager) self._flush_cache(inventory, variable_manager)
# create the playbook executor, which manages running the plays via a task queue manager # create the playbook executor, which manages running the plays via a task queue manager
pbex = PlaybookExecutor(playbooks=self.args, inventory=inventory, variable_manager=variable_manager, loader=loader, options=self.options, pbex = PlaybookExecutor(playbooks=context.CLIARGS['args'], inventory=inventory,
variable_manager=variable_manager, loader=loader,
passwords=passwords) passwords=passwords)
results = pbex.run() results = pbex.run()
@ -131,7 +139,7 @@ class PlaybookCLI(CLI):
mytags = set(play.tags) mytags = set(play.tags)
msg += '\tTAGS: [%s]' % (','.join(mytags)) msg += '\tTAGS: [%s]' % (','.join(mytags))
if self.options.listhosts: if context.CLIARGS['listhosts']:
playhosts = set(inventory.get_hosts(play.hosts)) playhosts = set(inventory.get_hosts(play.hosts))
msg += "\n pattern: %s\n hosts (%d):" % (play.hosts, len(playhosts)) msg += "\n pattern: %s\n hosts (%d):" % (play.hosts, len(playhosts))
for host in playhosts: for host in playhosts:
@ -140,9 +148,9 @@ class PlaybookCLI(CLI):
display.display(msg) display.display(msg)
all_tags = set() all_tags = set()
if self.options.listtags or self.options.listtasks: if context.CLIARGS['listtags'] or context.CLIARGS['listtasks']:
taskmsg = '' taskmsg = ''
if self.options.listtasks: if context.CLIARGS['listtasks']:
taskmsg = ' tasks:\n' taskmsg = ' tasks:\n'
def _process_block(b): def _process_block(b):
@ -155,7 +163,7 @@ class PlaybookCLI(CLI):
continue continue
all_tags.update(task.tags) all_tags.update(task.tags)
if self.options.listtasks: if context.CLIARGS['listtasks']:
cur_tags = list(mytags.union(set(task.tags))) cur_tags = list(mytags.union(set(task.tags)))
cur_tags.sort() cur_tags.sort()
if task.name: if task.name:
@ -167,14 +175,14 @@ class PlaybookCLI(CLI):
return taskmsg return taskmsg
all_vars = variable_manager.get_vars(play=play) all_vars = variable_manager.get_vars(play=play)
play_context = PlayContext(play=play, options=self.options) play_context = PlayContext(play=play)
for block in play.compile(): for block in play.compile():
block = block.filter_tagged_tasks(play_context, all_vars) block = block.filter_tagged_tasks(play_context, all_vars)
if not block.has_tasks(): if not block.has_tasks():
continue continue
taskmsg += _process_block(block) taskmsg += _process_block(block)
if self.options.listtags: if context.CLIARGS['listtags']:
cur_tags = list(mytags.union(all_tags)) cur_tags = list(mytags.union(all_tags))
cur_tags.sort() cur_tags.sort()
taskmsg += " TASK TAGS: [%s]\n" % ', '.join(cur_tags) taskmsg += " TASK TAGS: [%s]\n" % ', '.join(cur_tags)

View file

@ -1,19 +1,6 @@
# (c) 2012, Michael DeHaan <michael.dehaan@gmail.com> # Copyright: (c) 2012, Michael DeHaan <michael.dehaan@gmail.com>
# # Copyright: (c) 2018, Ansible Project
# This file is part of Ansible # GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
#
# Ansible 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.
#
# Ansible 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 Ansible. If not, see <http://www.gnu.org/licenses/>.
from __future__ import (absolute_import, division, print_function) from __future__ import (absolute_import, division, print_function)
__metaclass__ = type __metaclass__ = type
@ -27,8 +14,9 @@ import socket
import sys import sys
import time import time
from ansible.cli import CLI
from ansible import constants as C from ansible import constants as C
from ansible import context
from ansible.cli import CLI
from ansible.errors import AnsibleOptionsError from ansible.errors import AnsibleOptionsError
from ansible.module_utils._text import to_native, to_text from ansible.module_utils._text import to_native, to_text
from ansible.plugins.loader import module_loader from ansible.plugins.loader import module_loader
@ -68,8 +56,8 @@ class PullCLI(CLI):
def _get_inv_cli(self): def _get_inv_cli(self):
inv_opts = '' inv_opts = ''
if getattr(self.options, 'inventory'): if context.CLIARGS.get('inventory', False):
for inv in self.options.inventory: for inv in context.CLIARGS['inventory']:
if isinstance(inv, list): if isinstance(inv, list):
inv_opts += " -i '%s' " % ','.join(inv) inv_opts += " -i '%s' " % ','.join(inv)
elif ',' in inv or os.path.exists(inv): elif ',' in inv or os.path.exists(inv):
@ -77,10 +65,10 @@ class PullCLI(CLI):
return inv_opts return inv_opts
def parse(self): def init_parser(self):
''' create an options parser for bin/ansible ''' ''' create an options parser for bin/ansible '''
self.parser = CLI.base_parser( self.parser = super(PullCLI, self).init_parser(
usage='%prog -U <repository> [options] [<playbook.yml>]', usage='%prog -U <repository> [options] [<playbook.yml>]',
connect_opts=True, connect_opts=True,
vault_opts=True, vault_opts=True,
@ -126,32 +114,37 @@ class PullCLI(CLI):
self.parser.add_option("--diff", default=C.DIFF_ALWAYS, dest='diff', action='store_true', self.parser.add_option("--diff", default=C.DIFF_ALWAYS, dest='diff', action='store_true',
help="when changing (small) files and templates, show the differences in those files; works great with --check") help="when changing (small) files and templates, show the differences in those files; works great with --check")
super(PullCLI, self).parse() return self.parser
if not self.options.dest: def post_process_args(self, options, args):
options, args = super(PullCLI, self).post_process_args(options, args)
if not options.dest:
hostname = socket.getfqdn() hostname = socket.getfqdn()
# use a hostname dependent directory, in case of $HOME on nfs # use a hostname dependent directory, in case of $HOME on nfs
self.options.dest = os.path.join('~/.ansible/pull', hostname) options.dest = os.path.join('~/.ansible/pull', hostname)
self.options.dest = os.path.expandvars(os.path.expanduser(self.options.dest)) options.dest = os.path.expandvars(os.path.expanduser(options.dest))
if os.path.exists(self.options.dest) and not os.path.isdir(self.options.dest): if os.path.exists(options.dest) and not os.path.isdir(options.dest):
raise AnsibleOptionsError("%s is not a valid or accessible directory." % self.options.dest) raise AnsibleOptionsError("%s is not a valid or accessible directory." % options.dest)
if self.options.sleep: if options.sleep:
try: try:
secs = random.randint(0, int(self.options.sleep)) secs = random.randint(0, int(options.sleep))
self.options.sleep = secs options.sleep = secs
except ValueError: except ValueError:
raise AnsibleOptionsError("%s is not a number." % self.options.sleep) raise AnsibleOptionsError("%s is not a number." % options.sleep)
if not self.options.url: if not options.url:
raise AnsibleOptionsError("URL for repository not specified, use -h for help") raise AnsibleOptionsError("URL for repository not specified, use -h for help")
if self.options.module_name not in self.SUPPORTED_REPO_MODULES: if options.module_name not in self.SUPPORTED_REPO_MODULES:
raise AnsibleOptionsError("Unsupported repo module %s, choices are %s" % (self.options.module_name, ','.join(self.SUPPORTED_REPO_MODULES))) raise AnsibleOptionsError("Unsupported repo module %s, choices are %s" % (options.module_name, ','.join(self.SUPPORTED_REPO_MODULES)))
display.verbosity = self.options.verbosity display.verbosity = options.verbosity
self.validate_conflicts(vault_opts=True) self.validate_conflicts(options, vault_opts=True)
return options, args
def run(self): def run(self):
''' use Runner lib to do SSH things ''' ''' use Runner lib to do SSH things '''
@ -169,8 +162,8 @@ class PullCLI(CLI):
host = socket.getfqdn() host = socket.getfqdn()
limit_opts = 'localhost,%s,127.0.0.1' % ','.join(set([host, node, host.split('.')[0], node.split('.')[0]])) limit_opts = 'localhost,%s,127.0.0.1' % ','.join(set([host, node, host.split('.')[0], node.split('.')[0]]))
base_opts = '-c local ' base_opts = '-c local '
if self.options.verbosity > 0: if context.CLIARGS['verbosity'] > 0:
base_opts += ' -%s' % ''.join(["v" for x in range(0, self.options.verbosity)]) base_opts += ' -%s' % ''.join(["v" for x in range(0, context.CLIARGS['verbosity'])])
# Attempt to use the inventory passed in as an argument # Attempt to use the inventory passed in as an argument
# It might not yet have been downloaded so use localhost as default # It might not yet have been downloaded so use localhost as default
@ -179,61 +172,65 @@ class PullCLI(CLI):
inv_opts = " -i localhost, " inv_opts = " -i localhost, "
# SCM specific options # SCM specific options
if self.options.module_name == 'git': if context.CLIARGS['module_name'] == 'git':
repo_opts = "name=%s dest=%s" % (self.options.url, self.options.dest) repo_opts = "name=%s dest=%s" % (context.CLIARGS['url'], context.CLIARGS['dest'])
if self.options.checkout: if context.CLIARGS['checkout']:
repo_opts += ' version=%s' % self.options.checkout repo_opts += ' version=%s' % context.CLIARGS['checkout']
if self.options.accept_host_key: if context.CLIARGS['accept_host_key']:
repo_opts += ' accept_hostkey=yes' repo_opts += ' accept_hostkey=yes'
if self.options.private_key_file: if context.CLIARGS['private_key_file']:
repo_opts += ' key_file=%s' % self.options.private_key_file repo_opts += ' key_file=%s' % context.CLIARGS['private_key_file']
if self.options.verify: if context.CLIARGS['verify']:
repo_opts += ' verify_commit=yes' repo_opts += ' verify_commit=yes'
if self.options.tracksubs: if context.CLIARGS['tracksubs']:
repo_opts += ' track_submodules=yes' repo_opts += ' track_submodules=yes'
if not self.options.fullclone: if not context.CLIARGS['fullclone']:
repo_opts += ' depth=1' repo_opts += ' depth=1'
elif self.options.module_name == 'subversion': elif context.CLIARGS['module_name'] == 'subversion':
repo_opts = "repo=%s dest=%s" % (self.options.url, self.options.dest) repo_opts = "repo=%s dest=%s" % (context.CLIARGS['url'], context.CLIARGS['dest'])
if self.options.checkout: if context.CLIARGS['checkout']:
repo_opts += ' revision=%s' % self.options.checkout repo_opts += ' revision=%s' % context.CLIARGS['checkout']
if not self.options.fullclone: if not context.CLIARGS['fullclone']:
repo_opts += ' export=yes' repo_opts += ' export=yes'
elif self.options.module_name == 'hg': elif context.CLIARGS['module_name'] == 'hg':
repo_opts = "repo=%s dest=%s" % (self.options.url, self.options.dest) repo_opts = "repo=%s dest=%s" % (context.CLIARGS['url'], context.CLIARGS['dest'])
if self.options.checkout: if context.CLIARGS['checkout']:
repo_opts += ' revision=%s' % self.options.checkout repo_opts += ' revision=%s' % context.CLIARGS['checkout']
elif self.options.module_name == 'bzr': elif context.CLIARGS['module_name'] == 'bzr':
repo_opts = "name=%s dest=%s" % (self.options.url, self.options.dest) repo_opts = "name=%s dest=%s" % (context.CLIARGS['url'], context.CLIARGS['dest'])
if self.options.checkout: if context.CLIARGS['checkout']:
repo_opts += ' version=%s' % self.options.checkout repo_opts += ' version=%s' % context.CLIARGS['checkout']
else: else:
raise AnsibleOptionsError('Unsupported (%s) SCM module for pull, choices are: %s' % (self.options.module_name, ','.join(self.REPO_CHOICES))) raise AnsibleOptionsError('Unsupported (%s) SCM module for pull, choices are: %s'
% (context.CLIARGS['module_name'],
','.join(self.REPO_CHOICES)))
# options common to all supported SCMS # options common to all supported SCMS
if self.options.clean: if context.CLIARGS['clean']:
repo_opts += ' force=yes' repo_opts += ' force=yes'
path = module_loader.find_plugin(self.options.module_name) path = module_loader.find_plugin(context.CLIARGS['module_name'])
if path is None: if path is None:
raise AnsibleOptionsError(("module '%s' not found.\n" % self.options.module_name)) raise AnsibleOptionsError(("module '%s' not found.\n" % context.CLIARGS['module_name']))
bin_path = os.path.dirname(os.path.abspath(sys.argv[0])) bin_path = os.path.dirname(os.path.abspath(sys.argv[0]))
# hardcode local and inventory/host as this is just meant to fetch the repo # hardcode local and inventory/host as this is just meant to fetch the repo
cmd = '%s/ansible %s %s -m %s -a "%s" all -l "%s"' % (bin_path, inv_opts, base_opts, self.options.module_name, repo_opts, limit_opts) cmd = '%s/ansible %s %s -m %s -a "%s" all -l "%s"' % (bin_path, inv_opts, base_opts,
context.CLIARGS['module_name'],
repo_opts, limit_opts)
for ev in self.options.extra_vars: for ev in context.CLIARGS['extra_vars']:
cmd += ' -e "%s"' % ev cmd += ' -e "%s"' % ev
# Nap? # Nap?
if self.options.sleep: if context.CLIARGS['sleep']:
display.display("Sleeping for %d seconds..." % self.options.sleep) display.display("Sleeping for %d seconds..." % context.CLIARGS['sleep'])
time.sleep(self.options.sleep) time.sleep(context.CLIARGS['sleep'])
# RUN the Checkout command # RUN the Checkout command
display.debug("running ansible with VCS module to checkout repo") display.debug("running ansible with VCS module to checkout repo")
@ -241,45 +238,45 @@ class PullCLI(CLI):
rc, b_out, b_err = run_cmd(cmd, live=True) rc, b_out, b_err = run_cmd(cmd, live=True)
if rc != 0: if rc != 0:
if self.options.force: if context.CLIARGS['force']:
display.warning("Unable to update repository. Continuing with (forced) run of playbook.") display.warning("Unable to update repository. Continuing with (forced) run of playbook.")
else: else:
return rc return rc
elif self.options.ifchanged and b'"changed": true' not in b_out: elif context.CLIARGS['ifchanged'] and b'"changed": true' not in b_out:
display.display("Repository has not changed, quitting.") display.display("Repository has not changed, quitting.")
return 0 return 0
playbook = self.select_playbook(self.options.dest) playbook = self.select_playbook(context.CLIARGS['dest'])
if playbook is None: if playbook is None:
raise AnsibleOptionsError("Could not find a playbook to run.") raise AnsibleOptionsError("Could not find a playbook to run.")
# Build playbook command # Build playbook command
cmd = '%s/ansible-playbook %s %s' % (bin_path, base_opts, playbook) cmd = '%s/ansible-playbook %s %s' % (bin_path, base_opts, playbook)
if self.options.vault_password_files: if context.CLIARGS['vault_password_files']:
for vault_password_file in self.options.vault_password_files: for vault_password_file in context.CLIARGS['vault_password_files']:
cmd += " --vault-password-file=%s" % vault_password_file cmd += " --vault-password-file=%s" % vault_password_file
if self.options.vault_ids: if context.CLIARGS['vault_ids']:
for vault_id in self.options.vault_ids: for vault_id in context.CLIARGS['vault_ids']:
cmd += " --vault-id=%s" % vault_id cmd += " --vault-id=%s" % vault_id
for ev in self.options.extra_vars: for ev in context.CLIARGS['extra_vars']:
cmd += ' -e "%s"' % ev cmd += ' -e "%s"' % ev
if self.options.ask_sudo_pass or self.options.ask_su_pass or self.options.become_ask_pass: if context.CLIARGS['ask_sudo_pass'] or context.CLIARGS['ask_su_pass'] or context.CLIARGS['become_ask_pass']:
cmd += ' --ask-become-pass' cmd += ' --ask-become-pass'
if self.options.skip_tags: if context.CLIARGS['skip_tags']:
cmd += ' --skip-tags "%s"' % to_native(u','.join(self.options.skip_tags)) cmd += ' --skip-tags "%s"' % to_native(u','.join(context.CLIARGS['skip_tags']))
if self.options.tags: if context.CLIARGS['tags']:
cmd += ' -t "%s"' % to_native(u','.join(self.options.tags)) cmd += ' -t "%s"' % to_native(u','.join(context.CLIARGS['tags']))
if self.options.subset: if context.CLIARGS['subset']:
cmd += ' -l "%s"' % self.options.subset cmd += ' -l "%s"' % context.CLIARGS['subset']
else: else:
cmd += ' -l "%s"' % limit_opts cmd += ' -l "%s"' % limit_opts
if self.options.check: if context.CLIARGS['check']:
cmd += ' -C' cmd += ' -C'
if self.options.diff: if context.CLIARGS['diff']:
cmd += ' -D' cmd += ' -D'
os.chdir(self.options.dest) os.chdir(context.CLIARGS['dest'])
# redo inventory options as new files might exist now # redo inventory options as new files might exist now
inv_opts = self._get_inv_cli() inv_opts = self._get_inv_cli()
@ -291,12 +288,12 @@ class PullCLI(CLI):
display.debug('EXEC: %s' % cmd) display.debug('EXEC: %s' % cmd)
rc, b_out, b_err = run_cmd(cmd, live=True) rc, b_out, b_err = run_cmd(cmd, live=True)
if self.options.purge: if context.CLIARGS['purge']:
os.chdir('/') os.chdir('/')
try: try:
shutil.rmtree(self.options.dest) shutil.rmtree(context.CLIARGS['dest'])
except Exception as e: except Exception as e:
display.error(u"Failed to remove %s: %s" % (self.options.dest, to_text(e))) display.error(u"Failed to remove %s: %s" % (context.CLIARGS['dest'], to_text(e)))
return rc return rc
@ -309,8 +306,8 @@ class PullCLI(CLI):
def select_playbook(self, path): def select_playbook(self, path):
playbook = None playbook = None
if len(self.args) > 0 and self.args[0] is not None: if context.CLIARGS['args'] and context.CLIARGS['args'][0] is not None:
playbook = os.path.join(path, self.args[0]) playbook = os.path.join(path, context.CLIARGS['args'][0])
rc = self.try_playbook(playbook) rc = self.try_playbook(playbook)
if rc != 0: if rc != 0:
display.warning("%s: %s" % (playbook, self.PLAYBOOK_ERRORS[rc])) display.warning("%s: %s" % (playbook, self.PLAYBOOK_ERRORS[rc]))

View file

@ -1,20 +1,6 @@
# (c) 2014, James Tanner <tanner.jc@gmail.com> # (c) 2014, James Tanner <tanner.jc@gmail.com>
# # Copyright: (c) 2018, Ansible Project
# Ansible is free software: you can redistribute it and/or modify # GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
# 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.
#
# Ansible 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 Ansible. If not, see <http://www.gnu.org/licenses/>.
#
# ansible-vault is a script that encrypts/decrypts YAML files. See
# https://docs.ansible.com/playbooks_vault.html for more details.
from __future__ import (absolute_import, division, print_function) from __future__ import (absolute_import, division, print_function)
__metaclass__ = type __metaclass__ = type
@ -22,8 +8,9 @@ __metaclass__ = type
import os import os
import sys import sys
from ansible.cli import CLI
from ansible import constants as C from ansible import constants as C
from ansible import context
from ansible.cli import CLI
from ansible.errors import AnsibleOptionsError from ansible.errors import AnsibleOptionsError
from ansible.module_utils._text import to_text, to_bytes from ansible.module_utils._text import to_text, to_bytes
from ansible.parsing.dataloader import DataLoader from ansible.parsing.dataloader import DataLoader
@ -75,7 +62,7 @@ class VaultCLI(CLI):
if self.action in self.can_output: if self.action in self.can_output:
self.parser.add_option('--output', default=None, dest='output_file', self.parser.add_option('--output', default=None, dest='output_file',
help='output file name for encrypt or decrypt; use - for stdout', help='output file name for encrypt or decrypt; use - for stdout',
action="callback", callback=CLI.unfrack_path, type='string') action="callback", callback=self.unfrack_path, type='string')
# options specific to self.actions # options specific to self.actions
if self.action == "create": if self.action == "create":
@ -109,9 +96,9 @@ class VaultCLI(CLI):
action='store', type='string', action='store', type='string',
help='the vault id used to encrypt (required if more than vault-id is provided)') help='the vault id used to encrypt (required if more than vault-id is provided)')
def parse(self): def init_parser(self):
self.parser = CLI.base_parser( self.parser = super(VaultCLI, self).init_parser(
vault_opts=True, vault_opts=True,
vault_rekey_opts=True, vault_rekey_opts=True,
usage="usage: %%prog [%s] [options] [vaultfile.yml]" % "|".join(sorted(self.VALID_ACTIONS)), usage="usage: %%prog [%s] [options] [vaultfile.yml]" % "|".join(sorted(self.VALID_ACTIONS)),
@ -121,18 +108,21 @@ class VaultCLI(CLI):
self.set_action() self.set_action()
super(VaultCLI, self).parse() return self.parser
self.validate_conflicts(vault_opts=True, vault_rekey_opts=True)
display.verbosity = self.options.verbosity def post_process_args(self, options, args):
options, args = super(VaultCLI, self).post_process_args(options, args)
self.validate_conflicts(options, vault_opts=True, vault_rekey_opts=True)
if self.options.vault_ids: display.verbosity = options.verbosity
for vault_id in self.options.vault_ids:
if options.vault_ids:
for vault_id in options.vault_ids:
if u';' in vault_id: if u';' in vault_id:
raise AnsibleOptionsError("'%s' is not a valid vault id. The character ';' is not allowed in vault ids" % vault_id) raise AnsibleOptionsError("'%s' is not a valid vault id. The character ';' is not allowed in vault ids" % vault_id)
if self.action not in self.can_output: if self.action not in self.can_output:
if len(self.args) == 0: if not args:
raise AnsibleOptionsError("Vault requires at least one filename as a parameter") raise AnsibleOptionsError("Vault requires at least one filename as a parameter")
else: else:
# This restriction should remain in place until it's possible to # This restriction should remain in place until it's possible to
@ -140,17 +130,19 @@ class VaultCLI(CLI):
# to create an encrypted file that can't be read back in. But in # to create an encrypted file that can't be read back in. But in
# the meanwhile, "cat a b c|ansible-vault encrypt --output x" is # the meanwhile, "cat a b c|ansible-vault encrypt --output x" is
# a workaround. # a workaround.
if self.options.output_file and len(self.args) > 1: if options.output_file and len(args) > 1:
raise AnsibleOptionsError("At most one input file may be used with the --output option") raise AnsibleOptionsError("At most one input file may be used with the --output option")
if self.action == 'encrypt_string': if self.action == 'encrypt_string':
if '-' in self.args or len(self.args) == 0 or self.options.encrypt_string_stdin_name: if '-' in args or not args or options.encrypt_string_stdin_name:
self.encrypt_string_read_stdin = True self.encrypt_string_read_stdin = True
# TODO: prompting from stdin and reading from stdin seem mutually exclusive, but verify that. # TODO: prompting from stdin and reading from stdin seem mutually exclusive, but verify that.
if self.options.encrypt_string_prompt and self.encrypt_string_read_stdin: if options.encrypt_string_prompt and self.encrypt_string_read_stdin:
raise AnsibleOptionsError('The --prompt option is not supported if also reading input from stdin') raise AnsibleOptionsError('The --prompt option is not supported if also reading input from stdin')
return options, args
def run(self): def run(self):
super(VaultCLI, self).run() super(VaultCLI, self).run()
loader = DataLoader() loader = DataLoader()
@ -158,7 +150,7 @@ class VaultCLI(CLI):
# set default restrictive umask # set default restrictive umask
old_umask = os.umask(0o077) old_umask = os.umask(0o077)
vault_ids = self.options.vault_ids vault_ids = list(context.CLIARGS['vault_ids'])
# there are 3 types of actions, those that just 'read' (decrypt, view) and only # there are 3 types of actions, those that just 'read' (decrypt, view) and only
# need to ask for a password once, and those that 'write' (create, encrypt) that # need to ask for a password once, and those that 'write' (create, encrypt) that
@ -171,26 +163,25 @@ class VaultCLI(CLI):
# TODO: instead of prompting for these before, we could let VaultEditor # TODO: instead of prompting for these before, we could let VaultEditor
# call a callback when it needs it. # call a callback when it needs it.
if self.action in ['decrypt', 'view', 'rekey', 'edit']: if self.action in ['decrypt', 'view', 'rekey', 'edit']:
vault_secrets = self.setup_vault_secrets(loader, vault_secrets = self.setup_vault_secrets(loader, vault_ids=vault_ids,
vault_ids=vault_ids, vault_password_files=list(context.CLIARGS['vault_password_files']),
vault_password_files=self.options.vault_password_files, ask_vault_pass=context.CLIARGS['ask_vault_pass'])
ask_vault_pass=self.options.ask_vault_pass)
if not vault_secrets: if not vault_secrets:
raise AnsibleOptionsError("A vault password is required to use Ansible's Vault") raise AnsibleOptionsError("A vault password is required to use Ansible's Vault")
if self.action in ['encrypt', 'encrypt_string', 'create']: if self.action in ['encrypt', 'encrypt_string', 'create']:
encrypt_vault_id = None encrypt_vault_id = None
# no --encrypt-vault-id self.options.encrypt_vault_id for 'edit' # no --encrypt-vault-id context.CLIARGS['encrypt_vault_id'] for 'edit'
if self.action not in ['edit']: if self.action not in ['edit']:
encrypt_vault_id = self.options.encrypt_vault_id or C.DEFAULT_VAULT_ENCRYPT_IDENTITY encrypt_vault_id = context.CLIARGS['encrypt_vault_id'] or C.DEFAULT_VAULT_ENCRYPT_IDENTITY
vault_secrets = None vault_secrets = None
vault_secrets = \ vault_secrets = \
self.setup_vault_secrets(loader, self.setup_vault_secrets(loader,
vault_ids=vault_ids, vault_ids=vault_ids,
vault_password_files=self.options.vault_password_files, vault_password_files=context.CLIARGS['vault_password_files'],
ask_vault_pass=self.options.ask_vault_pass, ask_vault_pass=context.CLIARGS['ask_vault_pass'],
create_new_password=True) create_new_password=True)
if len(vault_secrets) > 1 and not encrypt_vault_id: if len(vault_secrets) > 1 and not encrypt_vault_id:
@ -209,7 +200,7 @@ class VaultCLI(CLI):
self.encrypt_secret = encrypt_secret[1] self.encrypt_secret = encrypt_secret[1]
if self.action in ['rekey']: if self.action in ['rekey']:
encrypt_vault_id = self.options.encrypt_vault_id or C.DEFAULT_VAULT_ENCRYPT_IDENTITY encrypt_vault_id = context.CLIARGS['encrypt_vault_id'] or C.DEFAULT_VAULT_ENCRYPT_IDENTITY
# print('encrypt_vault_id: %s' % encrypt_vault_id) # print('encrypt_vault_id: %s' % encrypt_vault_id)
# print('default_encrypt_vault_id: %s' % default_encrypt_vault_id) # print('default_encrypt_vault_id: %s' % default_encrypt_vault_id)
@ -218,18 +209,18 @@ class VaultCLI(CLI):
new_vault_ids = [] new_vault_ids = []
if encrypt_vault_id: if encrypt_vault_id:
new_vault_ids = default_vault_ids new_vault_ids = default_vault_ids
if self.options.new_vault_id: if context.CLIARGS['new_vault_id']:
new_vault_ids.append(self.options.new_vault_id) new_vault_ids.append(context.CLIARGS['new_vault_id'])
new_vault_password_files = [] new_vault_password_files = []
if self.options.new_vault_password_file: if context.CLIARGS['new_vault_password_file']:
new_vault_password_files.append(self.options.new_vault_password_file) new_vault_password_files.append(context.CLIARGS['new_vault_password_file'])
new_vault_secrets = \ new_vault_secrets = \
self.setup_vault_secrets(loader, self.setup_vault_secrets(loader,
vault_ids=new_vault_ids, vault_ids=new_vault_ids,
vault_password_files=new_vault_password_files, vault_password_files=new_vault_password_files,
ask_vault_pass=self.options.ask_vault_pass, ask_vault_pass=context.CLIARGS['ask_vault_pass'],
create_new_password=True) create_new_password=True)
if not new_vault_secrets: if not new_vault_secrets:
@ -257,14 +248,14 @@ class VaultCLI(CLI):
def execute_encrypt(self): def execute_encrypt(self):
''' encrypt the supplied file using the provided vault secret ''' ''' encrypt the supplied file using the provided vault secret '''
if len(self.args) == 0 and sys.stdin.isatty(): if not context.CLIARGS['args'] and sys.stdin.isatty():
display.display("Reading plaintext input from stdin", stderr=True) display.display("Reading plaintext input from stdin", stderr=True)
for f in self.args or ['-']: for f in context.CLIARGS['args'] or ['-']:
# Fixme: use the correct vau # Fixme: use the correct vau
self.editor.encrypt_file(f, self.encrypt_secret, self.editor.encrypt_file(f, self.encrypt_secret,
vault_id=self.encrypt_vault_id, vault_id=self.encrypt_vault_id,
output_file=self.options.output_file) output_file=context.CLIARGS['output_file'])
if sys.stdout.isatty(): if sys.stdout.isatty():
display.display("Encryption successful", stderr=True) display.display("Encryption successful", stderr=True)
@ -296,10 +287,10 @@ class VaultCLI(CLI):
# remove the non-option '-' arg (used to indicate 'read from stdin') from the candidate args so # remove the non-option '-' arg (used to indicate 'read from stdin') from the candidate args so
# we don't add it to the plaintext list # we don't add it to the plaintext list
args = [x for x in self.args if x != '-'] args = [x for x in context.CLIARGS['args'] if x != '-']
# We can prompt and read input, or read from stdin, but not both. # We can prompt and read input, or read from stdin, but not both.
if self.options.encrypt_string_prompt: if context.CLIARGS['encrypt_string_prompt']:
msg = "String to encrypt: " msg = "String to encrypt: "
name = None name = None
@ -332,20 +323,21 @@ class VaultCLI(CLI):
b_plaintext = to_bytes(stdin_text) b_plaintext = to_bytes(stdin_text)
# defaults to None # defaults to None
name = self.options.encrypt_string_stdin_name name = context.CLIARGS['encrypt_string_stdin_name']
b_plaintext_list.append((b_plaintext, self.FROM_STDIN, name)) b_plaintext_list.append((b_plaintext, self.FROM_STDIN, name))
# use any leftover args as strings to encrypt # use any leftover args as strings to encrypt
# Try to match args up to --name options # Try to match args up to --name options
if hasattr(self.options, 'encrypt_string_names') and self.options.encrypt_string_names: if context.CLIARGS.get('encrypt_string_names', False):
name_and_text_list = list(zip(self.options.encrypt_string_names, args)) name_and_text_list = list(zip(context.CLIARGS['encrypt_string_names'], args))
# Some but not enough --name's to name each var # Some but not enough --name's to name each var
if len(args) > len(name_and_text_list): if len(args) > len(name_and_text_list):
# Trying to avoid ever showing the plaintext in the output, so this warning is vague to avoid that. # Trying to avoid ever showing the plaintext in the output, so this warning is vague to avoid that.
display.display('The number of --name options do not match the number of args.', display.display('The number of --name options do not match the number of args.',
stderr=True) stderr=True)
display.display('The last named variable will be "%s". The rest will not have names.' % self.options.encrypt_string_names[-1], display.display('The last named variable will be "%s". The rest will not have'
' names.' % context.CLIARGS['encrypt_string_names'][-1],
stderr=True) stderr=True)
# Add the rest of the args without specifying a name # Add the rest of the args without specifying a name
@ -419,11 +411,11 @@ class VaultCLI(CLI):
def execute_decrypt(self): def execute_decrypt(self):
''' decrypt the supplied file using the provided vault secret ''' ''' decrypt the supplied file using the provided vault secret '''
if len(self.args) == 0 and sys.stdin.isatty(): if not context.CLIARGS['args'] and sys.stdin.isatty():
display.display("Reading ciphertext input from stdin", stderr=True) display.display("Reading ciphertext input from stdin", stderr=True)
for f in self.args or ['-']: for f in context.CLIARGS['args'] or ['-']:
self.editor.decrypt_file(f, output_file=self.options.output_file) self.editor.decrypt_file(f, output_file=context.CLIARGS['output_file'])
if sys.stdout.isatty(): if sys.stdout.isatty():
display.display("Decryption successful", stderr=True) display.display("Decryption successful", stderr=True)
@ -431,21 +423,21 @@ class VaultCLI(CLI):
def execute_create(self): def execute_create(self):
''' create and open a file in an editor that will be encrypted with the provided vault secret when closed''' ''' create and open a file in an editor that will be encrypted with the provided vault secret when closed'''
if len(self.args) > 1: if len(context.CLIARGS['args']) > 1:
raise AnsibleOptionsError("ansible-vault create can take only one filename argument") raise AnsibleOptionsError("ansible-vault create can take only one filename argument")
self.editor.create_file(self.args[0], self.encrypt_secret, self.editor.create_file(context.CLIARGS['args'][0], self.encrypt_secret,
vault_id=self.encrypt_vault_id) vault_id=self.encrypt_vault_id)
def execute_edit(self): def execute_edit(self):
''' open and decrypt an existing vaulted file in an editor, that will be encrypted again when closed''' ''' open and decrypt an existing vaulted file in an editor, that will be encrypted again when closed'''
for f in self.args: for f in context.CLIARGS['args']:
self.editor.edit_file(f) self.editor.edit_file(f)
def execute_view(self): def execute_view(self):
''' open, decrypt and view an existing vaulted file using a pager using the supplied vault secret ''' ''' open, decrypt and view an existing vaulted file using a pager using the supplied vault secret '''
for f in self.args: for f in context.CLIARGS['args']:
# Note: vault should return byte strings because it could encrypt # Note: vault should return byte strings because it could encrypt
# and decrypt binary files. We are responsible for changing it to # and decrypt binary files. We are responsible for changing it to
# unicode here because we are displaying it and therefore can make # unicode here because we are displaying it and therefore can make
@ -456,7 +448,7 @@ class VaultCLI(CLI):
def execute_rekey(self): def execute_rekey(self):
''' re-encrypt a vaulted file with a new secret, the previous secret is required ''' ''' re-encrypt a vaulted file with a new secret, the previous secret is required '''
for f in self.args: for f in context.CLIARGS['args']:
# FIXME: plumb in vault_id, use the default new_vault_secret for now # FIXME: plumb in vault_id, use the default new_vault_secret for now
self.editor.rekey_file(f, self.new_encrypt_secret, self.editor.rekey_file(f, self.new_encrypt_secret,
self.new_encrypt_vault_id) self.new_encrypt_vault_id)

53
lib/ansible/context.py Normal file
View file

@ -0,0 +1,53 @@
# Copyright: (c) 2018, Toshio Kuratomi <tkuratomi@ansible.com>
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
# Make coding more python3-ish
from __future__ import (absolute_import, division, print_function)
__metaclass__ = type
"""
Context of the running Ansible.
In the future we *may* create Context objects to allow running multiple Ansible plays in parallel
with different contexts but that is currently out of scope as the Ansible library is just for
running the ansible command line tools.
These APIs are still in flux so do not use them unless you are willing to update them with every Ansible release
"""
from ansible import arguments
# Note: this is not the singleton version. That is only created once the program has actually
# parsed the args
CLIARGS = arguments.CLIArgs({})
class _Context:
"""
Not yet ready for Prime Time
Eventually this may allow for code which needs to run under different contexts (for instance, as
if they were run with different command line args or from different current working directories)
to exist in the same process. But at the moment, we don't need that so this code has not been
tested for suitability.
"""
def __init__(self):
global CLIARGS
self._CLIARGS = arguments.CLIArgs(CLIARGS)
@property
def CLIARGS(self):
return self._CLIARGS
@CLIARGS.setter
def CLIARGS_set(self, new_cli_args):
if not isinstance(new_cli_args, arguments.CLIArgs):
raise TypeError('CLIARGS must be of type (ansible.arguments.CLIArgs)')
self._CLIARGS = new_cli_args
def _init_global_context(cli_args):
"""Initialize the global context objects"""
global CLIARGS
CLIARGS = arguments.GlobalCLIArgs.from_options(cli_args)

View file

@ -22,6 +22,7 @@ __metaclass__ = type
import os import os
from ansible import constants as C from ansible import constants as C
from ansible import context
from ansible.executor.task_queue_manager import TaskQueueManager from ansible.executor.task_queue_manager import TaskQueueManager
from ansible.module_utils._text import to_native, to_text from ansible.module_utils._text import to_native, to_text
from ansible.playbook import Playbook from ansible.playbook import Playbook
@ -42,19 +43,20 @@ class PlaybookExecutor:
basis for bin/ansible-playbook operation. basis for bin/ansible-playbook operation.
''' '''
def __init__(self, playbooks, inventory, variable_manager, loader, options, passwords): def __init__(self, playbooks, inventory, variable_manager, loader, passwords):
self._playbooks = playbooks self._playbooks = playbooks
self._inventory = inventory self._inventory = inventory
self._variable_manager = variable_manager self._variable_manager = variable_manager
self._loader = loader self._loader = loader
self._options = options
self.passwords = passwords self.passwords = passwords
self._unreachable_hosts = dict() self._unreachable_hosts = dict()
if options.listhosts or options.listtasks or options.listtags or options.syntax: if context.CLIARGS.get('listhosts') or context.CLIARGS.get('listtasks') or \
context.CLIARGS.get('listtags') or context.CLIARGS.get('syntax'):
self._tqm = None self._tqm = None
else: else:
self._tqm = TaskQueueManager(inventory=inventory, variable_manager=variable_manager, loader=loader, options=options, passwords=self.passwords) self._tqm = TaskQueueManager(inventory=inventory, variable_manager=variable_manager,
loader=loader, passwords=self.passwords)
# Note: We run this here to cache whether the default ansible ssh # Note: We run this here to cache whether the default ansible ssh
# executable supports control persist. Sometime in the future we may # executable supports control persist. Sometime in the future we may
@ -127,7 +129,7 @@ class PlaybookExecutor:
templar = Templar(loader=self._loader, variables=all_vars) templar = Templar(loader=self._loader, variables=all_vars)
play.post_validate(templar) play.post_validate(templar)
if self._options.syntax: if context.CLIARGS['syntax']:
continue continue
if self._tqm is None: if self._tqm is None:
@ -218,15 +220,15 @@ class PlaybookExecutor:
if self._loader: if self._loader:
self._loader.cleanup_all_tmp_files() self._loader.cleanup_all_tmp_files()
if self._options.syntax: if context.CLIARGS['syntax']:
display.display("No issues encountered") display.display("No issues encountered")
return result return result
if self._options.start_at_task and not self._tqm._start_at_done: if context.CLIARGS['start_at_task'] and not self._tqm._start_at_done:
display.error( display.error(
"No matching task \"%s\" found." "No matching task \"%s\" found."
" Note: --start-at-task can only follow static includes." " Note: --start-at-task can only follow static includes."
% self._options.start_at_task % context.CLIARGS['start_at_task']
) )
return result return result

View file

@ -24,6 +24,7 @@ import os
import tempfile import tempfile
from ansible import constants as C from ansible import constants as C
from ansible import context
from ansible.errors import AnsibleError from ansible.errors import AnsibleError
from ansible.executor.play_iterator import PlayIterator from ansible.executor.play_iterator import PlayIterator
from ansible.executor.stats import AggregateStats from ansible.executor.stats import AggregateStats
@ -65,25 +66,25 @@ class TaskQueueManager:
RUN_FAILED_BREAK_PLAY = 8 RUN_FAILED_BREAK_PLAY = 8
RUN_UNKNOWN_ERROR = 255 RUN_UNKNOWN_ERROR = 255
def __init__(self, inventory, variable_manager, loader, options, passwords, stdout_callback=None, run_additional_callbacks=True, run_tree=False): def __init__(self, inventory, variable_manager, loader, passwords, stdout_callback=None, run_additional_callbacks=True, run_tree=False, forks=None):
self._inventory = inventory self._inventory = inventory
self._variable_manager = variable_manager self._variable_manager = variable_manager
self._loader = loader self._loader = loader
self._options = options
self._stats = AggregateStats() self._stats = AggregateStats()
self.passwords = passwords self.passwords = passwords
self._stdout_callback = stdout_callback self._stdout_callback = stdout_callback
self._run_additional_callbacks = run_additional_callbacks self._run_additional_callbacks = run_additional_callbacks
self._run_tree = run_tree self._run_tree = run_tree
self._forks = forks or 5
self._callbacks_loaded = False self._callbacks_loaded = False
self._callback_plugins = [] self._callback_plugins = []
self._start_at_done = False self._start_at_done = False
# make sure any module paths (if specified) are added to the module_loader # make sure any module paths (if specified) are added to the module_loader
if options.module_path: if context.CLIARGS.get('module_path', False):
for path in options.module_path: for path in context.CLIARGS['module_path']:
if path: if path:
module_loader.add_directory(path) module_loader.add_directory(path)
@ -214,7 +215,7 @@ class TaskQueueManager:
loader=self._loader, loader=self._loader,
) )
play_context = PlayContext(new_play, self._options, self.passwords, self._connection_lockfile.fileno()) play_context = PlayContext(new_play, self.passwords, self._connection_lockfile.fileno())
if (self._stdout_callback and if (self._stdout_callback and
hasattr(self._stdout_callback, 'set_play_context')): hasattr(self._stdout_callback, 'set_play_context')):
self._stdout_callback.set_play_context(play_context) self._stdout_callback.set_play_context(play_context)
@ -239,7 +240,7 @@ class TaskQueueManager:
) )
# adjust to # of workers to configured forks or size of batch, whatever is lower # adjust to # of workers to configured forks or size of batch, whatever is lower
self._initialize_processes(min(self._options.forks, iterator.batch_size)) self._initialize_processes(min(self._forks, iterator.batch_size))
# load the specified strategy (or the default linear one) # load the specified strategy (or the default linear one)
strategy = strategy_loader.get(new_play.strategy, self) strategy = strategy_loader.get(new_play.strategy, self)
@ -259,7 +260,7 @@ class TaskQueueManager:
# during initialization, the PlayContext will clear the start_at_task # during initialization, the PlayContext will clear the start_at_task
# field to signal that a matching task was found, so check that here # field to signal that a matching task was found, so check that here
# and remember it so we don't try to skip tasks on future plays # and remember it so we don't try to skip tasks on future plays
if getattr(self._options, 'start_at_task', None) is not None and play_context.start_at_task is None: if context.CLIARGS.get('start_at_task') is not None and play_context.start_at_task is None:
self._start_at_done = True self._start_at_done = True
# and run the play using the strategy and cleanup on way out # and run the play using the strategy and cleanup on way out

View file

@ -25,6 +25,7 @@ __metaclass__ = type
import os import os
from ansible import context
from ansible.errors import AnsibleError from ansible.errors import AnsibleError
from ansible.module_utils.six import string_types from ansible.module_utils.six import string_types
@ -35,19 +36,18 @@ from ansible.module_utils.six import string_types
class Galaxy(object): class Galaxy(object):
''' Keeps global galaxy info ''' ''' Keeps global galaxy info '''
def __init__(self, options): def __init__(self):
self.options = options # roles_path needs to be a list and will be by default
# self.options.roles_path needs to be a list and will be by default roles_path = context.CLIARGS.get('roles_path', tuple())
roles_path = getattr(self.options, 'roles_path', []) # cli option handling is responsible for splitting roles_path
# cli option handling is responsible for making roles_path a list
self.roles_paths = roles_path self.roles_paths = roles_path
self.roles = {} self.roles = {}
# load data path for resource usage # load data path for resource usage
this_dir, this_filename = os.path.split(__file__) this_dir, this_filename = os.path.split(__file__)
type_path = getattr(self.options, 'role_type', "default") type_path = context.CLIARGS.get('role_type', "default")
self.DATA_PATH = os.path.join(this_dir, 'data', type_path) self.DATA_PATH = os.path.join(this_dir, 'data', type_path)
@property @property

View file

@ -24,6 +24,7 @@ __metaclass__ = type
import json import json
from ansible import context
import ansible.constants as C import ansible.constants as C
from ansible.errors import AnsibleError from ansible.errors import AnsibleError
from ansible.galaxy.token import GalaxyToken from ansible.galaxy.token import GalaxyToken
@ -63,7 +64,7 @@ class GalaxyAPI(object):
self.galaxy = galaxy self.galaxy = galaxy
self.token = GalaxyToken() self.token = GalaxyToken()
self._api_server = C.GALAXY_SERVER self._api_server = C.GALAXY_SERVER
self._validate_certs = not galaxy.options.ignore_certs self._validate_certs = not context.CLIARGS['ignore_certs']
self.baseurl = None self.baseurl = None
self.version = None self.version = None
self.initialized = False self.initialized = False
@ -71,8 +72,8 @@ class GalaxyAPI(object):
display.debug('Validate TLS certificates: %s' % self._validate_certs) display.debug('Validate TLS certificates: %s' % self._validate_certs)
# set the API server # set the API server
if galaxy.options.api_server != C.GALAXY_SERVER: if context.CLIARGS['api_server'] != C.GALAXY_SERVER:
self._api_server = galaxy.options.api_server self._api_server = context.CLIARGS['api_server']
def __auth_header(self): def __auth_header(self):
token = self.token.get() token = self.token.get()

View file

@ -31,6 +31,7 @@ import yaml
from distutils.version import LooseVersion from distutils.version import LooseVersion
from shutil import rmtree from shutil import rmtree
from ansible import context
from ansible.errors import AnsibleError from ansible.errors import AnsibleError
from ansible.module_utils._text import to_native, to_text from ansible.module_utils._text import to_native, to_text
from ansible.module_utils.urls import open_url from ansible.module_utils.urls import open_url
@ -52,11 +53,10 @@ class GalaxyRole(object):
self._metadata = None self._metadata = None
self._install_info = None self._install_info = None
self._validate_certs = not galaxy.options.ignore_certs self._validate_certs = not context.CLIARGS['ignore_certs']
display.debug('Validate TLS certificates: %s' % self._validate_certs) display.debug('Validate TLS certificates: %s' % self._validate_certs)
self.options = galaxy.options
self.galaxy = galaxy self.galaxy = galaxy
self.name = name self.name = name
@ -196,7 +196,7 @@ class GalaxyRole(object):
if self.scm: if self.scm:
# create tar file from scm url # create tar file from scm url
tmp_file = RoleRequirement.scm_archive_role(keep_scm_meta=self.options.keep_scm_meta, **self.spec) tmp_file = RoleRequirement.scm_archive_role(keep_scm_meta=context.CLIARGS['keep_scm_meta'], **self.spec)
elif self.src: elif self.src:
if os.path.isfile(self.src): if os.path.isfile(self.src):
tmp_file = self.src tmp_file = self.src
@ -298,7 +298,7 @@ class GalaxyRole(object):
if os.path.exists(self.path): if os.path.exists(self.path):
if not os.path.isdir(self.path): if not os.path.isdir(self.path):
raise AnsibleError("the specified roles path exists and is not a directory.") raise AnsibleError("the specified roles path exists and is not a directory.")
elif not getattr(self.options, "force", False): elif not context.CLIARGS.get("force", False):
raise AnsibleError("the specified role %s appears to already exist. Use --force to replace it." % self.name) raise AnsibleError("the specified role %s appears to already exist. Use --force to replace it." % self.name)
else: else:
# using --force, remove the old path # using --force, remove the old path

View file

@ -29,6 +29,7 @@ import string
import sys import sys
from ansible import constants as C from ansible import constants as C
from ansible import context
from ansible.errors import AnsibleError from ansible.errors import AnsibleError
from ansible.module_utils.six import iteritems from ansible.module_utils.six import iteritems
from ansible.module_utils.six.moves import shlex_quote from ansible.module_utils.six.moves import shlex_quote
@ -186,7 +187,7 @@ class PlayContext(Base):
_gather_timeout = FieldAttribute(isa='string', default=C.DEFAULT_GATHER_TIMEOUT) _gather_timeout = FieldAttribute(isa='string', default=C.DEFAULT_GATHER_TIMEOUT)
_fact_path = FieldAttribute(isa='string', default=C.DEFAULT_FACT_PATH) _fact_path = FieldAttribute(isa='string', default=C.DEFAULT_FACT_PATH)
def __init__(self, play=None, options=None, passwords=None, connection_lockfd=None): def __init__(self, play=None, passwords=None, connection_lockfd=None):
super(PlayContext, self).__init__() super(PlayContext, self).__init__()
@ -203,8 +204,8 @@ class PlayContext(Base):
self.connection_lockfd = connection_lockfd self.connection_lockfd = connection_lockfd
# set options before play to allow play to override them # set options before play to allow play to override them
if options: if context.CLIARGS:
self.set_options(options) self.set_options()
if play: if play:
self.set_play(play) self.set_play(play)
@ -250,7 +251,7 @@ class PlayContext(Base):
# for flag in ('ssh_common_args', 'docker_extra_args', 'sftp_extra_args', 'scp_extra_args', 'ssh_extra_args'): # for flag in ('ssh_common_args', 'docker_extra_args', 'sftp_extra_args', 'scp_extra_args', 'ssh_extra_args'):
# setattr(self, flag, getattr(options, flag, '')) # setattr(self, flag, getattr(options, flag, ''))
def set_options(self, options): def set_options(self):
''' '''
Configures this connection information instance with data from Configures this connection information instance with data from
options specified by the user on the command line. These have a options specified by the user on the command line. These have a
@ -258,33 +259,33 @@ class PlayContext(Base):
''' '''
# privilege escalation # privilege escalation
self.become = options.become self.become = context.CLIARGS['become']
self.become_method = options.become_method self.become_method = context.CLIARGS['become_method']
self.become_user = options.become_user self.become_user = context.CLIARGS['become_user']
self.check_mode = boolean(options.check, strict=False) self.check_mode = boolean(context.CLIARGS['check'], strict=False)
self.diff = boolean(options.diff, strict=False) self.diff = boolean(context.CLIARGS['diff'], strict=False)
# general flags (should we move out?) # general flags (should we move out?)
# should only be 'non plugin' flags # should only be 'non plugin' flags
for flag in OPTION_FLAGS: for flag in OPTION_FLAGS:
attribute = getattr(options, flag, False) attribute = context.CLIARGS.get(flag, False)
if attribute: if attribute:
setattr(self, flag, attribute) setattr(self, flag, attribute)
if hasattr(options, 'timeout') and options.timeout: if context.CLIARGS.get('timeout', False):
self.timeout = int(options.timeout) self.timeout = context.CLIARGS['timeout']
# get the tag info from options. We check to see if the options have # get the tag info from options. We check to see if the options have
# the attribute, as it is not always added via the CLI # the attribute, as it is not always added via the CLI
if hasattr(options, 'tags'): if context.CLIARGS.get('tags', False):
self.only_tags.update(options.tags) self.only_tags.update(context.CLIARGS['tags'])
if len(self.only_tags) == 0: if len(self.only_tags) == 0:
self.only_tags = set(['all']) self.only_tags = set(['all'])
if hasattr(options, 'skip_tags'): if context.CLIARGS.get('skip_tags', False):
self.skip_tags.update(options.skip_tags) self.skip_tags.update(context.CLIARGS['skip_tags'])
def set_task_and_variable_override(self, task, variables, templar): def set_task_and_variable_override(self, task, variables, templar):
''' '''

View file

@ -46,11 +46,6 @@ else:
global_display = Display() global_display = Display()
try:
from __main__ import cli
except ImportError:
# using API w/o cli
cli = False
__all__ = ["CallbackBase"] __all__ = ["CallbackBase"]
@ -72,11 +67,6 @@ class CallbackBase(AnsiblePlugin):
else: else:
self._display = global_display self._display = global_display
if cli:
self._options = cli.options
else:
self._options = None
if self._display.verbosity >= 4: if self._display.verbosity >= 4:
name = getattr(self, 'CALLBACK_NAME', 'unnamed') name = getattr(self, 'CALLBACK_NAME', 'unnamed')
ctype = getattr(self, 'CALLBACK_TYPE', 'old') ctype = getattr(self, 'CALLBACK_TYPE', 'old')

View file

@ -19,6 +19,7 @@ DOCUMENTATION = '''
''' '''
from ansible import constants as C from ansible import constants as C
from ansible import context
from ansible.playbook.task_include import TaskInclude from ansible.playbook.task_include import TaskInclude
from ansible.plugins.callback import CallbackBase from ansible.plugins.callback import CallbackBase
from ansible.utils.color import colorize, hostcolor from ansible.utils.color import colorize, hostcolor
@ -370,15 +371,16 @@ class CallbackModule(CallbackBase):
from os.path import basename from os.path import basename
self._display.banner("PLAYBOOK: %s" % basename(playbook._file_name)) self._display.banner("PLAYBOOK: %s" % basename(playbook._file_name))
# show CLI arguments
if self._display.verbosity > 3: if self._display.verbosity > 3:
# show CLI options if context.CLIARGS.get('args'):
if self._options is not None: self._display.display('Positional arguments: %s' % ' '.join(context.CLIARGS['args']),
for option in dir(self._options): color=C.COLOR_VERBOSE, screen_only=True)
if option.startswith('_') or option in ['read_file', 'ensure_value', 'read_module']:
continue for argument in (a for a in context.CLIARGS if a != 'args'):
val = getattr(self._options, option) val = context.CLIARGS[argument]
if val and self._display.verbosity > 3: if val:
self._display.display('%s: %s' % (option, val), color=C.COLOR_VERBOSE, screen_only=True) self._display.display('%s: %s' % (argument, val), color=C.COLOR_VERBOSE, screen_only=True)
def v2_runner_retry(self, result): def v2_runner_retry(self, result):
task_name = result.task_name or result._task task_name = result.task_name or result._task

View file

@ -58,11 +58,7 @@ import json
import os import os
import uuid import uuid
try: from ansible import context
from __main__ import cli
except ImportError:
cli = None
from ansible.module_utils._text import to_text from ansible.module_utils._text import to_text
from ansible.module_utils.urls import open_url from ansible.module_utils.urls import open_url
from ansible.plugins.callback import CallbackBase from ansible.plugins.callback import CallbackBase
@ -87,8 +83,6 @@ class CallbackModule(CallbackBase):
super(CallbackModule, self).__init__(display=display) super(CallbackModule, self).__init__(display=display)
self._options = cli.options
if not HAS_PRETTYTABLE: if not HAS_PRETTYTABLE:
self.disabled = True self.disabled = True
self._display.warning('The `prettytable` python module is not ' self._display.warning('The `prettytable` python module is not '
@ -145,13 +139,14 @@ class CallbackModule(CallbackBase):
title = [ title = [
'*Playbook initiated* (_%s_)' % self.guid '*Playbook initiated* (_%s_)' % self.guid
] ]
invocation_items = [] invocation_items = []
if self._options and self.show_invocation: if context.CLIARGS and self.show_invocation:
tags = self._options.tags tags = context.CLIARGS['tags']
skip_tags = self._options.skip_tags skip_tags = context.CLIARGS['skip_tags']
extra_vars = self._options.extra_vars extra_vars = context.CLIARGS['extra_vars']
subset = self._options.subset subset = context.CLIARGS['subset']
inventory = [os.path.abspath(i) for i in self._options.inventory] inventory = [os.path.abspath(i) for i in context.CLIARGS['inventory']]
invocation_items.append('Inventory: %s' % ', '.join(inventory)) invocation_items.append('Inventory: %s' % ', '.join(inventory))
if tags and tags != ['all']: if tags and tags != ['all']:
@ -164,7 +159,7 @@ class CallbackModule(CallbackBase):
invocation_items.append('Extra Vars: %s' % invocation_items.append('Extra Vars: %s' %
' '.join(extra_vars)) ' '.join(extra_vars))
title.append('by *%s*' % self._options.remote_user) title.append('by *%s*' % context.CLIARGS['remote_user'])
title.append('\n\n*%s*' % self.playbook_name) title.append('\n\n*%s*' % self.playbook_name)
msg_items = [' '.join(title)] msg_items = [' '.join(title)]

View file

@ -23,6 +23,7 @@ DOCUMENTATION = '''
from os.path import basename from os.path import basename
from ansible import constants as C from ansible import constants as C
from ansible import context
from ansible.module_utils._text import to_text from ansible.module_utils._text import to_text
from ansible.plugins.callback import CallbackBase from ansible.plugins.callback import CallbackBase
from ansible.utils.color import colorize, hostcolor from ansible.utils.color import colorize, hostcolor
@ -200,14 +201,16 @@ class CallbackModule(CallbackBase):
# TODO display whether this run is happening in check mode # TODO display whether this run is happening in check mode
self._display.display("Executing playbook %s" % basename(playbook._file_name)) self._display.display("Executing playbook %s" % basename(playbook._file_name))
# show CLI arguments
if self._display.verbosity > 3: if self._display.verbosity > 3:
if self._options is not None: if context.CLIARGS.get('args'):
for option in dir(self._options): self._display.display('Positional arguments: %s' % ' '.join(context.CLIARGS['args']),
if option.startswith('_') or option in ['read_file', 'ensure_value', 'read_module']: color=C.COLOR_VERBOSE, screen_only=True)
continue
val = getattr(self._options, option) for argument in (a for a in context.CLIARGS if a != 'args'):
val = context.CLIARGS[argument]
if val: if val:
self._display.vvvv('%s: %s' % (option, val)) self._display.vvvv('%s: %s' % (argument, val))
def v2_runner_retry(self, result): def v2_runner_retry(self, result):
msg = " Retrying... (%d of %d)" % (result._result['attempts'], result._result['retries']) msg = " Retrying... (%d of %d)" % (result._result['attempts'], result._result['retries'])

View file

@ -32,6 +32,7 @@ from multiprocessing import Lock
from jinja2.exceptions import UndefinedError from jinja2.exceptions import UndefinedError
from ansible import constants as C from ansible import constants as C
from ansible import context
from ansible.errors import AnsibleError, AnsibleFileNotFound, AnsibleParserError, AnsibleUndefinedVariable from ansible.errors import AnsibleError, AnsibleFileNotFound, AnsibleParserError, AnsibleUndefinedVariable
from ansible.executor import action_write_locks from ansible.executor import action_write_locks
from ansible.executor.process.worker import WorkerProcess from ansible.executor.process.worker import WorkerProcess
@ -170,9 +171,9 @@ class StrategyBase:
self._variable_manager = tqm.get_variable_manager() self._variable_manager = tqm.get_variable_manager()
self._loader = tqm.get_loader() self._loader = tqm.get_loader()
self._final_q = tqm._final_q self._final_q = tqm._final_q
self._step = getattr(tqm._options, 'step', False) self._step = context.CLIARGS.get('step', False)
self._diff = getattr(tqm._options, 'diff', False) self._diff = context.CLIARGS.get('diff', False)
self.flush_cache = getattr(tqm._options, 'flush_cache', False) self.flush_cache = context.CLIARGS.get('flush_cache', False)
# the task cache is a dictionary of tuples of (host.name, task._uuid) # the task cache is a dictionary of tuples of (host.name, task._uuid)
# used to find the original task object of in-flight tasks and to store # used to find the original task object of in-flight tasks and to store

View file

@ -27,6 +27,7 @@ from json import dumps
from ansible import constants as C from ansible import constants as C
from ansible import context
from ansible.errors import AnsibleError, AnsibleOptionsError from ansible.errors import AnsibleError, AnsibleOptionsError
from ansible.module_utils.six import iteritems, string_types from ansible.module_utils.six import iteritems, string_types
from ansible.module_utils._text import to_native, to_text from ansible.module_utils._text import to_native, to_text
@ -119,10 +120,9 @@ def merge_hash(a, b):
return result return result
def load_extra_vars(loader, options): def load_extra_vars(loader):
extra_vars = {} extra_vars = {}
if hasattr(options, 'extra_vars'): for extra_vars_opt in context.CLIARGS.get('extra_vars', tuple()):
for extra_vars_opt in options.extra_vars:
data = None data = None
extra_vars_opt = to_text(extra_vars_opt, errors='surrogate_or_strict') extra_vars_opt = to_text(extra_vars_opt, errors='surrogate_or_strict')
if extra_vars_opt.startswith(u"@"): if extra_vars_opt.startswith(u"@"):
@ -143,7 +143,7 @@ def load_extra_vars(loader, options):
return extra_vars return extra_vars
def load_options_vars(options, version): def load_options_vars(version):
options_vars = {'ansible_version': version} options_vars = {'ansible_version': version}
attrs = {'check': 'check_mode', attrs = {'check': 'check_mode',
@ -156,7 +156,7 @@ def load_options_vars(options, version):
'verbosity': 'verbosity'} 'verbosity': 'verbosity'}
for attr, alias in attrs.items(): for attr, alias in attrs.items():
opt = getattr(options, attr, None) opt = context.CLIARGS.get(attr)
if opt is not None: if opt is not None:
options_vars['ansible_%s' % alias] = opt options_vars['ansible_%s' % alias] = opt

View file

@ -5,6 +5,8 @@ from __future__ import (absolute_import, division, print_function)
__metaclass__ = type __metaclass__ = type
import pytest import pytest
from ansible import context
from ansible.cli.adhoc import AdHocCLI, display from ansible.cli.adhoc import AdHocCLI, display
from ansible.errors import AnsibleOptionsError from ansible.errors import AnsibleOptionsError
@ -22,7 +24,7 @@ def test_with_command():
module_name = 'command' module_name = 'command'
adhoc_cli = AdHocCLI(args=['-m', module_name, '-vv']) adhoc_cli = AdHocCLI(args=['-m', module_name, '-vv'])
adhoc_cli.parse() adhoc_cli.parse()
assert adhoc_cli.options.module_name == module_name assert context.CLIARGS['module_name'] == module_name
assert display.verbosity == 2 assert display.verbosity == 2
@ -36,9 +38,8 @@ def test_with_extra_parameters():
def test_simple_command(): def test_simple_command():
""" Test valid command and its run""" """ Test valid command and its run"""
adhoc_cli = AdHocCLI(['/bin/ansible', '-m', 'command', 'localhost']) adhoc_cli = AdHocCLI(['/bin/ansible', '-m', 'command', 'localhost', '-a', 'echo "hi"'])
adhoc_cli.parse() adhoc_cli.parse()
adhoc_cli.options.module_args = "echo 'hi'"
ret = adhoc_cli.run() ret = adhoc_cli.run()
assert ret == 0 assert ret == 0
@ -63,9 +64,8 @@ def test_did_you_mean_playbook():
def test_play_ds_positive(): def test_play_ds_positive():
""" Test _play_ds""" """ Test _play_ds"""
adhoc_cli = AdHocCLI(args=['/bin/ansible', 'localhost']) adhoc_cli = AdHocCLI(args=['/bin/ansible', 'localhost', '-m', 'command'])
adhoc_cli.parse() adhoc_cli.parse()
adhoc_cli.options.module_name = 'command'
ret = adhoc_cli._play_ds('command', 10, 2) ret = adhoc_cli._play_ds('command', 10, 2)
assert ret['name'] == 'Ansible Ad-Hoc' assert ret['name'] == 'Ansible Ad-Hoc'
assert ret['tasks'] == [{'action': {'module': 'command', 'args': {}}, 'async_val': 10, 'poll': 2}] assert ret['tasks'] == [{'action': {'module': 'command', 'args': {}}, 'async_val': 10, 'poll': 2}]
@ -73,9 +73,8 @@ def test_play_ds_positive():
def test_play_ds_with_include_role(): def test_play_ds_with_include_role():
""" Test include_role command with poll""" """ Test include_role command with poll"""
adhoc_cli = AdHocCLI(args=['/bin/ansible', 'localhost']) adhoc_cli = AdHocCLI(args=['/bin/ansible', 'localhost', '-m', 'include_role'])
adhoc_cli.parse() adhoc_cli.parse()
adhoc_cli.options.module_name = 'include_role'
ret = adhoc_cli._play_ds('include_role', None, 2) ret = adhoc_cli._play_ds('include_role', None, 2)
assert ret['name'] == 'Ansible Ad-Hoc' assert ret['name'] == 'Ansible Ad-Hoc'
assert ret['gather_facts'] == 'no' assert ret['gather_facts'] == 'no'
@ -88,5 +87,5 @@ def test_run_import_playbook():
adhoc_cli.parse() adhoc_cli.parse()
with pytest.raises(AnsibleOptionsError) as exec_info: with pytest.raises(AnsibleOptionsError) as exec_info:
adhoc_cli.run() adhoc_cli.run()
assert adhoc_cli.options.module_name == import_playbook assert context.CLIARGS['module_name'] == import_playbook
assert "'%s' is not a valid action for ad-hoc commands" % import_playbook == str(exec_info.value) assert "'%s' is not a valid action for ad-hoc commands" % import_playbook == str(exec_info.value)

View file

@ -26,6 +26,8 @@ import tarfile
import tempfile import tempfile
import yaml import yaml
from ansible import arguments
from ansible import context
from ansible.cli.galaxy import GalaxyCLI from ansible.cli.galaxy import GalaxyCLI
from units.compat import unittest from units.compat import unittest
from units.compat.mock import call, patch from units.compat.mock import call, patch
@ -47,7 +49,6 @@ class TestGalaxy(unittest.TestCase):
# creating framework for a role # creating framework for a role
gc = GalaxyCLI(args=["ansible-galaxy", "init", "--offline", "delete_me"]) gc = GalaxyCLI(args=["ansible-galaxy", "init", "--offline", "delete_me"])
gc.parse()
gc.run() gc.run()
cls.role_dir = "./delete_me" cls.role_dir = "./delete_me"
cls.role_name = "delete_me" cls.role_name = "delete_me"
@ -96,8 +97,14 @@ class TestGalaxy(unittest.TestCase):
shutil.rmtree(cls.temp_dir) shutil.rmtree(cls.temp_dir)
def setUp(self): def setUp(self):
# Reset the stored command line args
arguments.GlobalCLIArgs._Singleton__instance = None
self.default_args = ['ansible-galaxy'] self.default_args = ['ansible-galaxy']
def tearDown(self):
# Reset the stored command line args
arguments.GlobalCLIArgs._Singleton__instance = None
def test_init(self): def test_init(self):
galaxy_cli = GalaxyCLI(args=self.default_args) galaxy_cli = GalaxyCLI(args=self.default_args)
self.assertTrue(isinstance(galaxy_cli, GalaxyCLI)) self.assertTrue(isinstance(galaxy_cli, GalaxyCLI))
@ -120,12 +127,11 @@ class TestGalaxy(unittest.TestCase):
def test_run(self): def test_run(self):
''' verifies that the GalaxyCLI object's api is created and that execute() is called. ''' ''' verifies that the GalaxyCLI object's api is created and that execute() is called. '''
gc = GalaxyCLI(args=["ansible-galaxy", "install", "--ignore-errors", "imaginary_role"]) gc = GalaxyCLI(args=["ansible-galaxy", "install", "--ignore-errors", "imaginary_role"])
gc.parse()
with patch.object(ansible.cli.CLI, "execute", return_value=None) as mock_ex: with patch.object(ansible.cli.CLI, "execute", return_value=None) as mock_ex:
with patch.object(ansible.cli.CLI, "run", return_value=None) as mock_run: with patch.object(ansible.cli.CLI, "run", return_value=None) as mock_run:
gc.run() gc.run()
# testing # testing
self.assertIsInstance(gc.galaxy, ansible.galaxy.Galaxy)
self.assertEqual(mock_run.call_count, 1) self.assertEqual(mock_run.call_count, 1)
self.assertTrue(isinstance(gc.api, ansible.galaxy.api.GalaxyAPI)) self.assertTrue(isinstance(gc.api, ansible.galaxy.api.GalaxyAPI))
self.assertEqual(mock_ex.call_count, 1) self.assertEqual(mock_ex.call_count, 1)
@ -133,15 +139,16 @@ class TestGalaxy(unittest.TestCase):
def test_execute_remove(self): def test_execute_remove(self):
# installing role # installing role
gc = GalaxyCLI(args=["ansible-galaxy", "install", "-p", self.role_path, "-r", self.role_req, '--force']) gc = GalaxyCLI(args=["ansible-galaxy", "install", "-p", self.role_path, "-r", self.role_req, '--force'])
gc.parse()
gc.run() gc.run()
# location where the role was installed # location where the role was installed
role_file = os.path.join(self.role_path, self.role_name) role_file = os.path.join(self.role_path, self.role_name)
# removing role # removing role
# Have to reset the arguments in the context object manually since we're doing the
# equivalent of running the command line program twice
arguments.GlobalCLIArgs._Singleton__instance = None
gc = GalaxyCLI(args=["ansible-galaxy", "remove", role_file, self.role_name]) gc = GalaxyCLI(args=["ansible-galaxy", "remove", role_file, self.role_name])
gc.parse()
gc.run() gc.run()
# testing role was removed # testing role was removed
@ -151,7 +158,6 @@ class TestGalaxy(unittest.TestCase):
def test_exit_without_ignore_without_flag(self): def test_exit_without_ignore_without_flag(self):
''' tests that GalaxyCLI exits with the error specified if the --ignore-errors flag is not used ''' ''' tests that GalaxyCLI exits with the error specified if the --ignore-errors flag is not used '''
gc = GalaxyCLI(args=["ansible-galaxy", "install", "--server=None", "fake_role_name"]) gc = GalaxyCLI(args=["ansible-galaxy", "install", "--server=None", "fake_role_name"])
gc.parse()
with patch.object(ansible.utils.display.Display, "display", return_value=None) as mocked_display: with patch.object(ansible.utils.display.Display, "display", return_value=None) as mocked_display:
# testing that error expected is raised # testing that error expected is raised
self.assertRaises(AnsibleError, gc.run) self.assertRaises(AnsibleError, gc.run)
@ -161,7 +167,6 @@ class TestGalaxy(unittest.TestCase):
''' tests that GalaxyCLI exits without the error specified if the --ignore-errors flag is used ''' ''' tests that GalaxyCLI exits without the error specified if the --ignore-errors flag is used '''
# testing with --ignore-errors flag # testing with --ignore-errors flag
gc = GalaxyCLI(args=["ansible-galaxy", "install", "--server=None", "fake_role_name", "--ignore-errors"]) gc = GalaxyCLI(args=["ansible-galaxy", "install", "--server=None", "fake_role_name", "--ignore-errors"])
gc.parse()
with patch.object(ansible.utils.display.Display, "display", return_value=None) as mocked_display: with patch.object(ansible.utils.display.Display, "display", return_value=None) as mocked_display:
gc.run() gc.run()
self.assertTrue(mocked_display.called_once_with("- downloading role 'fake_role_name', owned by ")) self.assertTrue(mocked_display.called_once_with("- downloading role 'fake_role_name', owned by "))
@ -172,7 +177,6 @@ class TestGalaxy(unittest.TestCase):
# checking that the common results of parse() for all possible actions have been created/called # checking that the common results of parse() for all possible actions have been created/called
self.assertIsInstance(galaxycli_obj.parser, ansible.cli.SortedOptParser) self.assertIsInstance(galaxycli_obj.parser, ansible.cli.SortedOptParser)
self.assertIsInstance(galaxycli_obj.galaxy, ansible.galaxy.Galaxy)
formatted_call = { formatted_call = {
'import': 'usage: %prog import [options] github_user github_repo', 'import': 'usage: %prog import [options] github_user github_repo',
'delete': 'usage: %prog delete [options] github_user github_repo', 'delete': 'usage: %prog delete [options] github_user github_repo',
@ -206,74 +210,74 @@ class TestGalaxy(unittest.TestCase):
''' testing the options parser when the action 'delete' is given ''' ''' testing the options parser when the action 'delete' is given '''
gc = GalaxyCLI(args=["ansible-galaxy", "delete"]) gc = GalaxyCLI(args=["ansible-galaxy", "delete"])
self.run_parse_common(gc, "delete") self.run_parse_common(gc, "delete")
self.assertEqual(gc.options.verbosity, 0) self.assertEqual(context.CLIARGS['verbosity'], 0)
def test_parse_import(self): def test_parse_import(self):
''' testing the options parser when the action 'import' is given ''' ''' testing the options parser when the action 'import' is given '''
gc = GalaxyCLI(args=["ansible-galaxy", "import"]) gc = GalaxyCLI(args=["ansible-galaxy", "import"])
self.run_parse_common(gc, "import") self.run_parse_common(gc, "import")
self.assertEqual(gc.options.wait, True) self.assertEqual(context.CLIARGS['wait'], True)
self.assertEqual(gc.options.reference, None) self.assertEqual(context.CLIARGS['reference'], None)
self.assertEqual(gc.options.check_status, False) self.assertEqual(context.CLIARGS['check_status'], False)
self.assertEqual(gc.options.verbosity, 0) self.assertEqual(context.CLIARGS['verbosity'], 0)
def test_parse_info(self): def test_parse_info(self):
''' testing the options parser when the action 'info' is given ''' ''' testing the options parser when the action 'info' is given '''
gc = GalaxyCLI(args=["ansible-galaxy", "info"]) gc = GalaxyCLI(args=["ansible-galaxy", "info"])
self.run_parse_common(gc, "info") self.run_parse_common(gc, "info")
self.assertEqual(gc.options.offline, False) self.assertEqual(context.CLIARGS['offline'], False)
def test_parse_init(self): def test_parse_init(self):
''' testing the options parser when the action 'init' is given ''' ''' testing the options parser when the action 'init' is given '''
gc = GalaxyCLI(args=["ansible-galaxy", "init"]) gc = GalaxyCLI(args=["ansible-galaxy", "init"])
self.run_parse_common(gc, "init") self.run_parse_common(gc, "init")
self.assertEqual(gc.options.offline, False) self.assertEqual(context.CLIARGS['offline'], False)
self.assertEqual(gc.options.force, False) self.assertEqual(context.CLIARGS['force'], False)
def test_parse_install(self): def test_parse_install(self):
''' testing the options parser when the action 'install' is given ''' ''' testing the options parser when the action 'install' is given '''
gc = GalaxyCLI(args=["ansible-galaxy", "install"]) gc = GalaxyCLI(args=["ansible-galaxy", "install"])
self.run_parse_common(gc, "install") self.run_parse_common(gc, "install")
self.assertEqual(gc.options.ignore_errors, False) self.assertEqual(context.CLIARGS['ignore_errors'], False)
self.assertEqual(gc.options.no_deps, False) self.assertEqual(context.CLIARGS['no_deps'], False)
self.assertEqual(gc.options.role_file, None) self.assertEqual(context.CLIARGS['role_file'], None)
self.assertEqual(gc.options.force, False) self.assertEqual(context.CLIARGS['force'], False)
def test_parse_list(self): def test_parse_list(self):
''' testing the options parser when the action 'list' is given ''' ''' testing the options parser when the action 'list' is given '''
gc = GalaxyCLI(args=["ansible-galaxy", "list"]) gc = GalaxyCLI(args=["ansible-galaxy", "list"])
self.run_parse_common(gc, "list") self.run_parse_common(gc, "list")
self.assertEqual(gc.options.verbosity, 0) self.assertEqual(context.CLIARGS['verbosity'], 0)
def test_parse_login(self): def test_parse_login(self):
''' testing the options parser when the action 'login' is given ''' ''' testing the options parser when the action 'login' is given '''
gc = GalaxyCLI(args=["ansible-galaxy", "login"]) gc = GalaxyCLI(args=["ansible-galaxy", "login"])
self.run_parse_common(gc, "login") self.run_parse_common(gc, "login")
self.assertEqual(gc.options.verbosity, 0) self.assertEqual(context.CLIARGS['verbosity'], 0)
self.assertEqual(gc.options.token, None) self.assertEqual(context.CLIARGS['token'], None)
def test_parse_remove(self): def test_parse_remove(self):
''' testing the options parser when the action 'remove' is given ''' ''' testing the options parser when the action 'remove' is given '''
gc = GalaxyCLI(args=["ansible-galaxy", "remove"]) gc = GalaxyCLI(args=["ansible-galaxy", "remove"])
self.run_parse_common(gc, "remove") self.run_parse_common(gc, "remove")
self.assertEqual(gc.options.verbosity, 0) self.assertEqual(context.CLIARGS['verbosity'], 0)
def test_parse_search(self): def test_parse_search(self):
''' testing the options parswer when the action 'search' is given ''' ''' testing the options parswer when the action 'search' is given '''
gc = GalaxyCLI(args=["ansible-galaxy", "search"]) gc = GalaxyCLI(args=["ansible-galaxy", "search"])
self.run_parse_common(gc, "search") self.run_parse_common(gc, "search")
self.assertEqual(gc.options.platforms, None) self.assertEqual(context.CLIARGS['platforms'], None)
self.assertEqual(gc.options.galaxy_tags, None) self.assertEqual(context.CLIARGS['galaxy_tags'], None)
self.assertEqual(gc.options.author, None) self.assertEqual(context.CLIARGS['author'], None)
def test_parse_setup(self): def test_parse_setup(self):
''' testing the options parser when the action 'setup' is given ''' ''' testing the options parser when the action 'setup' is given '''
gc = GalaxyCLI(args=["ansible-galaxy", "setup"]) gc = GalaxyCLI(args=["ansible-galaxy", "setup"])
self.run_parse_common(gc, "setup") self.run_parse_common(gc, "setup")
self.assertEqual(gc.options.verbosity, 0) self.assertEqual(context.CLIARGS['verbosity'], 0)
self.assertEqual(gc.options.remove_id, None) self.assertEqual(context.CLIARGS['remove_id'], None)
self.assertEqual(gc.options.setup_list, False) self.assertEqual(context.CLIARGS['setup_list'], False)
class ValidRoleTests(object): class ValidRoleTests(object):
@ -299,7 +303,6 @@ class ValidRoleTests(object):
# create role using default skeleton # create role using default skeleton
gc = GalaxyCLI(args=['ansible-galaxy', 'init', '-c', '--offline'] + galaxy_args + ['--init-path', cls.test_dir, cls.role_name]) gc = GalaxyCLI(args=['ansible-galaxy', 'init', '-c', '--offline'] + galaxy_args + ['--init-path', cls.test_dir, cls.role_name])
gc.parse()
gc.run() gc.run()
cls.gc = gc cls.gc = gc
@ -466,4 +469,4 @@ class TestGalaxyInitSkeleton(unittest.TestCase, ValidRoleTests):
self.assertTrue(os.path.exists(os.path.join(self.role_dir, 'templates_extra', 'templates.txt'))) self.assertTrue(os.path.exists(os.path.join(self.role_dir, 'templates_extra', 'templates.txt')))
def test_skeleton_option(self): def test_skeleton_option(self):
self.assertEquals(self.role_skeleton_path, self.gc.options.role_skeleton, msg='Skeleton path was not parsed properly from the command line') self.assertEquals(self.role_skeleton_path, context.CLIARGS['role_skeleton'], msg='Skeleton path was not parsed properly from the command line')

View file

@ -22,6 +22,7 @@ __metaclass__ = type
from units.compat import unittest from units.compat import unittest
from units.mock.loader import DictDataLoader from units.mock.loader import DictDataLoader
from ansible import context
from ansible.inventory.manager import InventoryManager from ansible.inventory.manager import InventoryManager
from ansible.vars.manager import VariableManager from ansible.vars.manager import VariableManager
@ -32,7 +33,7 @@ class TestPlaybookCLI(unittest.TestCase):
def test_flush_cache(self): def test_flush_cache(self):
cli = PlaybookCLI(args=["ansible-playbook", "--flush-cache", "foobar.yml"]) cli = PlaybookCLI(args=["ansible-playbook", "--flush-cache", "foobar.yml"])
cli.parse() cli.parse()
self.assertTrue(cli.options.flush_cache) self.assertTrue(context.CLIARGS['flush_cache'])
variable_manager = VariableManager() variable_manager = VariableManager()
fake_loader = DictDataLoader({'foobar.yml': ""}) fake_loader = DictDataLoader({'foobar.yml': ""})

View file

@ -21,6 +21,8 @@ __metaclass__ = type
from units.compat import unittest from units.compat import unittest
from units.compat.mock import MagicMock from units.compat.mock import MagicMock
from ansible import arguments
from ansible.executor.playbook_executor import PlaybookExecutor from ansible.executor.playbook_executor import PlaybookExecutor
from ansible.playbook import Playbook from ansible.playbook import Playbook
from ansible.template import Templar from ansible.template import Templar
@ -31,10 +33,12 @@ from units.mock.loader import DictDataLoader
class TestPlaybookExecutor(unittest.TestCase): class TestPlaybookExecutor(unittest.TestCase):
def setUp(self): def setUp(self):
pass # Reset command line args for every test
arguments.CLIArgs._Singleton__instance = None
def tearDown(self): def tearDown(self):
pass # And cleanup after ourselves too
arguments.CLIArgs._Singleton__instance = None
def test_get_serialized_batches(self): def test_get_serialized_batches(self):
fake_loader = DictDataLoader({ fake_loader = DictDataLoader({
@ -77,11 +81,6 @@ class TestPlaybookExecutor(unittest.TestCase):
mock_inventory = MagicMock() mock_inventory = MagicMock()
mock_var_manager = MagicMock() mock_var_manager = MagicMock()
# fake out options to use the syntax CLI switch, which will ensure
# the PlaybookExecutor doesn't create a TaskQueueManager
mock_options = MagicMock()
mock_options.syntax.value = True
templar = Templar(loader=fake_loader) templar = Templar(loader=fake_loader)
pbe = PlaybookExecutor( pbe = PlaybookExecutor(
@ -89,7 +88,6 @@ class TestPlaybookExecutor(unittest.TestCase):
inventory=mock_inventory, inventory=mock_inventory,
variable_manager=mock_var_manager, variable_manager=mock_var_manager,
loader=fake_loader, loader=fake_loader,
options=mock_options,
passwords=[], passwords=[],
) )

View file

@ -20,6 +20,9 @@ from __future__ import (absolute_import, division, print_function)
from units.compat import unittest from units.compat import unittest
from units.compat.mock import MagicMock from units.compat.mock import MagicMock
from ansible import arguments
from ansible import context
from ansible.executor.task_queue_manager import TaskQueueManager from ansible.executor.task_queue_manager import TaskQueueManager
from ansible.playbook import Playbook from ansible.playbook import Playbook
from ansible.plugins.callback import CallbackBase from ansible.plugins.callback import CallbackBase
@ -32,10 +35,11 @@ class TestTaskQueueManagerCallbacks(unittest.TestCase):
inventory = MagicMock() inventory = MagicMock()
variable_manager = MagicMock() variable_manager = MagicMock()
loader = MagicMock() loader = MagicMock()
options = MagicMock()
passwords = [] passwords = []
self._tqm = TaskQueueManager(inventory, variable_manager, loader, options, passwords) # Reset the stored command line args
arguments.GlobalCLIArgs._Singleton__instance = None
self._tqm = TaskQueueManager(inventory, variable_manager, loader, passwords)
self._playbook = Playbook(loader) self._playbook = Playbook(loader)
# we use a MagicMock to register the result of the call we # we use a MagicMock to register the result of the call we
@ -46,7 +50,8 @@ class TestTaskQueueManagerCallbacks(unittest.TestCase):
self._register = MagicMock() self._register = MagicMock()
def tearDown(self): def tearDown(self):
pass # Reset the stored command line args
arguments.GlobalCLIArgs._Singleton__instance = None
def test_task_queue_manager_callbacks_v2_playbook_on_start(self): def test_task_queue_manager_callbacks_v2_playbook_on_start(self):
""" """

View file

@ -11,8 +11,10 @@ import os
import pytest import pytest
from ansible import arguments
from ansible import constants as C from ansible import constants as C
from ansible.cli import CLI from ansible import context
from ansible import cli
from units.compat import unittest from units.compat import unittest
from ansible.errors import AnsibleError, AnsibleParserError from ansible.errors import AnsibleError, AnsibleParserError
from ansible.module_utils.six.moves import shlex_quote from ansible.module_utils.six.moves import shlex_quote
@ -23,7 +25,7 @@ from units.mock.loader import DictDataLoader
@pytest.fixture @pytest.fixture
def parser(): def parser():
parser = CLI.base_parser(runas_opts=True, meta_opts=True, parser = cli.base_parser(runas_opts=True, meta_opts=True,
runtask_opts=True, vault_opts=True, runtask_opts=True, vault_opts=True,
async_opts=True, connect_opts=True, async_opts=True, connect_opts=True,
subset_opts=True, check_opts=True, subset_opts=True, check_opts=True,
@ -31,9 +33,18 @@ def parser():
return parser return parser
def test_play_context(mocker, parser): @pytest.fixture
def reset_cli_args():
arguments.GlobalCLIArgs._Singleton__instance = None
yield
arguments.GlobalCLIArgs._Singleton__instance = None
def test_play_context(mocker, parser, reset_cli_args):
(options, args) = parser.parse_args(['-vv', '--check']) (options, args) = parser.parse_args(['-vv', '--check'])
play_context = PlayContext(options=options) options.args = args
context._init_global_context(options)
play_context = PlayContext()
assert play_context._attributes['connection'] == C.DEFAULT_TRANSPORT assert play_context._attributes['connection'] == C.DEFAULT_TRANSPORT
assert play_context.remote_addr is None assert play_context.remote_addr is None
@ -56,7 +67,7 @@ def test_play_context(mocker, parser):
mock_play.become_user = 'mockroot' mock_play.become_user = 'mockroot'
mock_play.no_log = True mock_play.no_log = True
play_context = PlayContext(play=mock_play, options=options) play_context = PlayContext(play=mock_play)
assert play_context.connection == 'mock' assert play_context.connection == 'mock'
assert play_context.remote_user == 'mock' assert play_context.remote_user == 'mock'
assert play_context.password == '' assert play_context.password == ''
@ -83,7 +94,7 @@ def test_play_context(mocker, parser):
mock_templar = mocker.MagicMock() mock_templar = mocker.MagicMock()
play_context = PlayContext(play=mock_play, options=options) play_context = PlayContext(play=mock_play)
play_context = play_context.set_task_and_variable_override(task=mock_task, variables=all_vars, templar=mock_templar) play_context = play_context.set_task_and_variable_override(task=mock_task, variables=all_vars, templar=mock_templar)
assert play_context.connection == 'mock_inventory' assert play_context.connection == 'mock_inventory'
@ -100,9 +111,11 @@ def test_play_context(mocker, parser):
assert play_context.no_log is False assert play_context.no_log is False
def test_play_context_make_become_cmd(parser): def test_play_context_make_become_cmd(mocker, parser, reset_cli_args):
(options, args) = parser.parse_args([]) (options, args) = parser.parse_args([])
play_context = PlayContext(options=options) options.args = args
context._init_global_context(options)
play_context = PlayContext()
default_cmd = "/bin/foo" default_cmd = "/bin/foo"
default_exe = "/bin/bash" default_exe = "/bin/bash"

View file

@ -66,7 +66,6 @@ class TestStrategyBase(unittest.TestCase):
mock_tqm = MagicMock(TaskQueueManager) mock_tqm = MagicMock(TaskQueueManager)
mock_tqm._final_q = mock_queue mock_tqm._final_q = mock_queue
mock_tqm._options = MagicMock()
strategy_base = StrategyBase(tqm=mock_tqm) strategy_base = StrategyBase(tqm=mock_tqm)
strategy_base.cleanup() strategy_base.cleanup()
@ -106,7 +105,6 @@ class TestStrategyBase(unittest.TestCase):
mock_tqm._failed_hosts = dict() mock_tqm._failed_hosts = dict()
mock_tqm._unreachable_hosts = dict() mock_tqm._unreachable_hosts = dict()
mock_tqm._options = MagicMock()
strategy_base = StrategyBase(tqm=mock_tqm) strategy_base = StrategyBase(tqm=mock_tqm)
mock_host = MagicMock() mock_host = MagicMock()
@ -187,15 +185,13 @@ class TestStrategyBase(unittest.TestCase):
mock_host.has_hostkey = True mock_host.has_hostkey = True
mock_inventory = MagicMock() mock_inventory = MagicMock()
mock_inventory.get.return_value = mock_host mock_inventory.get.return_value = mock_host
mock_options = MagicMock()
mock_options.module_path = None
tqm = TaskQueueManager( tqm = TaskQueueManager(
inventory=mock_inventory, inventory=mock_inventory,
variable_manager=mock_var_manager, variable_manager=mock_var_manager,
loader=fake_loader, loader=fake_loader,
options=mock_options,
passwords=None, passwords=None,
forks=5,
) )
tqm._initialize_processes(3) tqm._initialize_processes(3)
tqm.hostvars = dict() tqm.hostvars = dict()
@ -520,15 +516,13 @@ class TestStrategyBase(unittest.TestCase):
mock_iterator._play = mock_play mock_iterator._play = mock_play
fake_loader = DictDataLoader() fake_loader = DictDataLoader()
mock_options = MagicMock()
mock_options.module_path = None
tqm = TaskQueueManager( tqm = TaskQueueManager(
inventory=mock_inventory, inventory=mock_inventory,
variable_manager=mock_var_mgr, variable_manager=mock_var_mgr,
loader=fake_loader, loader=fake_loader,
options=mock_options,
passwords=None, passwords=None,
forks=5,
) )
tqm._initialize_processes(3) tqm._initialize_processes(3)
tqm._initialize_notified_handlers(mock_play) tqm._initialize_notified_handlers(mock_play)

View file

@ -80,15 +80,12 @@ class TestStrategyLinear(unittest.TestCase):
all_vars=dict(), all_vars=dict(),
) )
mock_options = MagicMock()
mock_options.module_path = None
tqm = TaskQueueManager( tqm = TaskQueueManager(
inventory=inventory, inventory=inventory,
variable_manager=mock_var_manager, variable_manager=mock_var_manager,
loader=fake_loader, loader=fake_loader,
options=mock_options,
passwords=None, passwords=None,
forks=5,
) )
tqm._initialize_processes(3) tqm._initialize_processes(3)
strategy = StrategyModule(tqm) strategy = StrategyModule(tqm)

View file

@ -35,6 +35,17 @@ def test_make_immutable(data, expected):
assert arguments._make_immutable(data) == expected assert arguments._make_immutable(data) == expected
def test_cliargs_from_dict():
old_dict = {'tags': [u'production', u'webservers'],
'check_mode': True,
'start_at_task': u'Start with くらとみ'}
expected = frozenset((('tags', (u'production', u'webservers')),
('check_mode', True),
('start_at_task', u'Start with くらとみ')))
assert frozenset(arguments.CLIArgs(old_dict).items()) == expected
def test_cliargs(): def test_cliargs():
class FakeOptions: class FakeOptions:
pass pass
@ -47,7 +58,7 @@ def test_cliargs():
('check_mode', True), ('check_mode', True),
('start_at_task', u'Start with くらとみ'))) ('start_at_task', u'Start with くらとみ')))
assert frozenset(arguments.CLIArgs(options).items()) == expected assert frozenset(arguments.CLIArgs.from_options(options).items()) == expected
@pytest.mark.skipIf(argparse is None) @pytest.mark.skipIf(argparse is None)

View file

@ -0,0 +1,30 @@
# -*- coding: utf-8 -*-
# Copyright: (c) 2018, Toshio Kuratomi <tkuratomi@ansible.com>
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
# Make coding more python3-ish
from __future__ import (absolute_import, division)
__metaclass__ = type
import pytest
from ansible import context
class FakeOptions:
pass
def test_set_global_context():
options = FakeOptions()
options.tags = [u'production', u'webservers']
options.check_mode = True
options.start_at_task = u'Start with くらとみ'
expected = frozenset((('tags', (u'production', u'webservers')),
('check_mode', True),
('start_at_task', u'Start with くらとみ')))
context._init_global_context(options)
assert frozenset(context.CLIARGS.items()) == expected