a259b95556
When the inventory is empty, or the provided --limit subset returns an empty intersection, we don't want to loop over all plays but bail out at the very start.
181 lines
6.4 KiB
Python
Executable file
181 lines
6.4 KiB
Python
Executable file
#!/usr/bin/env python
|
|
# (C) 2012, Michael DeHaan, <michael.dehaan@gmail.com>
|
|
|
|
# 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/>.
|
|
|
|
#######################################################
|
|
|
|
import sys
|
|
import getpass
|
|
import os
|
|
|
|
import ansible.playbook
|
|
import ansible.constants as C
|
|
from ansible import errors
|
|
from ansible import callbacks
|
|
from ansible import utils
|
|
from ansible.color import ANSIBLE_COLOR, stringc
|
|
|
|
def colorize(lead, num, color):
|
|
""" Print 'lead' = 'num' in 'color' """
|
|
if num != 0 and ANSIBLE_COLOR:
|
|
return "%s%s%-15s" % (stringc(lead, color), stringc("=", color), stringc(str(num), color))
|
|
else:
|
|
return "%s=%-4s" % (lead, str(num))
|
|
|
|
def hostcolor(host, stats):
|
|
if ANSIBLE_COLOR:
|
|
if stats['failures'] != 0 or stats['unreachable'] != 0:
|
|
return "%-41s" % stringc(host, 'red')
|
|
elif stats['changed'] != 0:
|
|
return "%-41s" % stringc(host, 'yellow')
|
|
else:
|
|
return "%-41s" % stringc(host, 'green')
|
|
return "%-30s" % host
|
|
|
|
|
|
def main(args):
|
|
''' run ansible-playbook operations '''
|
|
|
|
# create parser for CLI options
|
|
usage = "%prog playbook.yml"
|
|
parser = utils.base_parser(constants=C, usage=usage, connect_opts=True, runas_opts=True, subset_opts=True)
|
|
parser.add_option('-e', '--extra-vars', dest="extra_vars", default=None,
|
|
help="set additional key=value variables from the CLI")
|
|
parser.add_option('-t', '--tags', dest='tags', default='all',
|
|
help="only run plays and tasks tagged with these values")
|
|
parser.add_option('--list-hosts', dest='listhosts', action='store_true',
|
|
help="dump out a list of hosts, each play will run against, does not run playbook!")
|
|
parser.add_option('--syntax-check', dest='syntax', action='store_true',
|
|
help="do a playbook syntax check on the playbook, do not execute the playbook")
|
|
|
|
options, args = parser.parse_args(args)
|
|
|
|
if len(args) == 0:
|
|
parser.print_help(file=sys.stderr)
|
|
return 1
|
|
|
|
inventory = ansible.inventory.Inventory(options.inventory)
|
|
inventory.subset(options.subset)
|
|
if len(inventory.list_hosts()) == 0:
|
|
raise errors.AnsibleError("provided hosts list is empty")
|
|
|
|
sshpass = None
|
|
sudopass = None
|
|
if not options.listhosts:
|
|
options.ask_pass = options.ask_pass or C.DEFAULT_ASK_PASS
|
|
if options.ask_pass:
|
|
sshpass = getpass.getpass(prompt="SSH password: ")
|
|
options.ask_sudo_pass = options.ask_sudo_pass or C.DEFAULT_ASK_SUDO_PASS
|
|
if options.ask_sudo_pass:
|
|
sudopass = getpass.getpass(prompt="sudo password: ")
|
|
options.sudo = True
|
|
if options.sudo_user:
|
|
options.sudo = True
|
|
options.sudo_user = options.sudo_user or C.DEFAULT_SUDO_USER
|
|
extra_vars = utils.parse_kv(options.extra_vars)
|
|
only_tags = options.tags.split(",")
|
|
|
|
for playbook in args:
|
|
if not os.path.exists(playbook):
|
|
raise errors.AnsibleError("the playbook: %s could not be found" % playbook)
|
|
if not os.path.isfile(playbook):
|
|
raise errors.AnsibleError("the playbook: %s does not appear to be a file" % playbook)
|
|
|
|
# run all playbooks specified on the command line
|
|
for playbook in args:
|
|
|
|
stats = callbacks.AggregateStats()
|
|
playbook_cb = callbacks.PlaybookCallbacks(verbose=utils.VERBOSITY)
|
|
runner_cb = callbacks.PlaybookRunnerCallbacks(stats, verbose=utils.VERBOSITY)
|
|
|
|
pb = ansible.playbook.PlayBook(
|
|
playbook=playbook,
|
|
module_path=options.module_path,
|
|
host_list=options.inventory,
|
|
forks=options.forks,
|
|
remote_user=options.remote_user,
|
|
remote_pass=sshpass,
|
|
callbacks=playbook_cb,
|
|
runner_callbacks=runner_cb,
|
|
stats=stats,
|
|
timeout=options.timeout,
|
|
transport=options.connection,
|
|
sudo=options.sudo,
|
|
sudo_user=options.sudo_user,
|
|
sudo_pass=sudopass,
|
|
extra_vars=extra_vars,
|
|
private_key_file=options.private_key_file,
|
|
only_tags=only_tags,
|
|
subset=options.subset,
|
|
)
|
|
|
|
if options.listhosts:
|
|
playnum = 0
|
|
for play in pb.playbook:
|
|
playnum += 1
|
|
if 'hosts' in play:
|
|
label = 'unnamed'
|
|
if 'name' in play:
|
|
label = play['name']
|
|
print 'hosts in play %s: %s' % (playnum, label)
|
|
for host in pb.inventory.list_hosts(play['hosts']):
|
|
print ' %s' % host
|
|
print '\n'
|
|
return 0
|
|
|
|
if options.syntax:
|
|
# if we've not exited by now then we are fine.
|
|
print 'Playbook Syntax is fine'
|
|
return 0
|
|
|
|
|
|
try:
|
|
|
|
pb.run()
|
|
|
|
hosts = sorted(pb.stats.processed.keys())
|
|
print callbacks.banner("PLAY RECAP")
|
|
playbook_cb.on_stats(pb.stats)
|
|
for h in hosts:
|
|
t = pb.stats.summarize(h)
|
|
print "%-30s : %s %s %s %s " % (
|
|
hostcolor(h, t),
|
|
colorize('ok', t['ok'], 'green'),
|
|
colorize('changed', t['changed'], 'yellow'),
|
|
colorize('unreachable', t['unreachable'], 'red'),
|
|
colorize('failed', t['failures'], 'red'))
|
|
|
|
print "\n"
|
|
for h in hosts:
|
|
stats = pb.stats.summarize(h)
|
|
if stats['failures'] != 0 or stats['unreachable'] != 0:
|
|
return 2
|
|
|
|
except errors.AnsibleError, e:
|
|
print >>sys.stderr, "ERROR: %s" % e
|
|
return 1
|
|
|
|
return 0
|
|
|
|
|
|
if __name__ == "__main__":
|
|
try:
|
|
sys.exit(main(sys.argv[1:]))
|
|
except errors.AnsibleError, e:
|
|
print >>sys.stderr, "ERROR: %s" % e
|
|
sys.exit(1)
|
|
|