misc code cleanup
This commit is contained in:
parent
5a63a48e6b
commit
1d8bb17785
3 changed files with 34 additions and 68 deletions
|
@ -128,7 +128,6 @@ class Inventory(object):
|
||||||
def get_variables(self, hostname):
|
def get_variables(self, hostname):
|
||||||
|
|
||||||
if self._is_script:
|
if self._is_script:
|
||||||
# TODO: move this to inventory_script.py
|
|
||||||
host = self.get_host(hostname)
|
host = self.get_host(hostname)
|
||||||
cmd = subprocess.Popen(
|
cmd = subprocess.Popen(
|
||||||
[self.host_list,"--host",hostname],
|
[self.host_list,"--host",hostname],
|
||||||
|
|
|
@ -23,6 +23,7 @@ import ansible.constants as C
|
||||||
from ansible import utils
|
from ansible import utils
|
||||||
from ansible import errors
|
from ansible import errors
|
||||||
import os
|
import os
|
||||||
|
import collections
|
||||||
from play import Play
|
from play import Play
|
||||||
|
|
||||||
#############################################
|
#############################################
|
||||||
|
@ -81,7 +82,7 @@ class PlayBook(object):
|
||||||
sudo: if not specified per play, requests all plays use sudo mode
|
sudo: if not specified per play, requests all plays use sudo mode
|
||||||
"""
|
"""
|
||||||
|
|
||||||
self.SETUP_CACHE = {}
|
self.SETUP_CACHE = collections.defaultdict(dict)
|
||||||
|
|
||||||
if playbook is None or callbacks is None or runner_callbacks is None or stats is None:
|
if playbook is None or callbacks is None or runner_callbacks is None or stats is None:
|
||||||
raise Exception('missing required arguments')
|
raise Exception('missing required arguments')
|
||||||
|
@ -157,7 +158,7 @@ class PlayBook(object):
|
||||||
# loop through all patterns and run them
|
# loop through all patterns and run them
|
||||||
self.callbacks.on_start()
|
self.callbacks.on_start()
|
||||||
for play_ds in self.playbook:
|
for play_ds in self.playbook:
|
||||||
self.SETUP_CACHE = {}
|
self.SETUP_CACHE = collections.defaultdict(dict)
|
||||||
self._run_play(Play(self,play_ds))
|
self._run_play(Play(self,play_ds))
|
||||||
|
|
||||||
# summarize the results
|
# summarize the results
|
||||||
|
|
|
@ -30,6 +30,7 @@ import time
|
||||||
import base64
|
import base64
|
||||||
import getpass
|
import getpass
|
||||||
import codecs
|
import codecs
|
||||||
|
import collections
|
||||||
|
|
||||||
import ansible.constants as C
|
import ansible.constants as C
|
||||||
import ansible.inventory
|
import ansible.inventory
|
||||||
|
@ -51,8 +52,7 @@ def _executor_hook(job_queue, result_queue):
|
||||||
''' callback used by multiprocessing pool '''
|
''' callback used by multiprocessing pool '''
|
||||||
|
|
||||||
# attempt workaround of https://github.com/newsapps/beeswithmachineguns/issues/17
|
# attempt workaround of https://github.com/newsapps/beeswithmachineguns/issues/17
|
||||||
# does not occur for everyone, some claim still occurs on newer paramiko
|
# this function also not present in CentOS 6
|
||||||
# this function not present in CentOS 6
|
|
||||||
if HAS_ATFORK:
|
if HAS_ATFORK:
|
||||||
atfork()
|
atfork()
|
||||||
|
|
||||||
|
@ -91,14 +91,7 @@ class ReturnData(object):
|
||||||
return self.comm_ok
|
return self.comm_ok
|
||||||
|
|
||||||
def is_successful(self):
|
def is_successful(self):
|
||||||
if not self.comm_ok:
|
return self.comm_ok and ('failed' not in self.result) and (self.result.get('rc',0) == 0)
|
||||||
return False
|
|
||||||
else:
|
|
||||||
if 'failed' in self.result:
|
|
||||||
return False
|
|
||||||
if self.result.get('rc',0) != 0:
|
|
||||||
return False
|
|
||||||
return True
|
|
||||||
|
|
||||||
class Runner(object):
|
class Runner(object):
|
||||||
''' core API interface to ansible '''
|
''' core API interface to ansible '''
|
||||||
|
@ -142,7 +135,7 @@ class Runner(object):
|
||||||
"""
|
"""
|
||||||
|
|
||||||
if setup_cache is None:
|
if setup_cache is None:
|
||||||
setup_cache = {}
|
setup_cache = collections.defaultdict(dict)
|
||||||
if basedir is None:
|
if basedir is None:
|
||||||
basedir = os.getcwd()
|
basedir = os.getcwd()
|
||||||
|
|
||||||
|
@ -273,12 +266,7 @@ class Runner(object):
|
||||||
msg = 'All items succeeded'
|
msg = 'All items succeeded'
|
||||||
if all_failed:
|
if all_failed:
|
||||||
msg = "One or more items failed."
|
msg = "One or more items failed."
|
||||||
rd_result = dict(
|
rd_result = dict(failed=all_failed, changed=all_changed, results=results, msg=msg)
|
||||||
failed = all_failed,
|
|
||||||
changed = all_changed,
|
|
||||||
results = results,
|
|
||||||
msg = msg
|
|
||||||
)
|
|
||||||
if not all_failed:
|
if not all_failed:
|
||||||
del rd_result['failed']
|
del rd_result['failed']
|
||||||
return ReturnData(host=conn.host, comm_ok=all_comm_ok, result=rd_result)
|
return ReturnData(host=conn.host, comm_ok=all_comm_ok, result=rd_result)
|
||||||
|
@ -290,7 +278,7 @@ class Runner(object):
|
||||||
|
|
||||||
''' runs a module that has already been transferred '''
|
''' runs a module that has already been transferred '''
|
||||||
|
|
||||||
inject = self.setup_cache.get(conn.host,{}).copy()
|
inject = self.setup_cache[conn.host].copy()
|
||||||
host_variables = self.inventory.get_variables(conn.host)
|
host_variables = self.inventory.get_variables(conn.host)
|
||||||
inject.update(host_variables)
|
inject.update(host_variables)
|
||||||
inject.update(self.module_vars)
|
inject.update(self.module_vars)
|
||||||
|
@ -306,8 +294,6 @@ class Runner(object):
|
||||||
|
|
||||||
args = utils.template(args, inject, self.setup_cache)
|
args = utils.template(args, inject, self.setup_cache)
|
||||||
|
|
||||||
module_name_tail = remote_module_path.split("/")[-1]
|
|
||||||
|
|
||||||
argsfile = self._transfer_str(conn, tmp, 'arguments', args)
|
argsfile = self._transfer_str(conn, tmp, 'arguments', args)
|
||||||
if async_jid is None:
|
if async_jid is None:
|
||||||
cmd = "%s %s" % (remote_module_path, argsfile)
|
cmd = "%s %s" % (remote_module_path, argsfile)
|
||||||
|
@ -315,7 +301,6 @@ class Runner(object):
|
||||||
cmd = " ".join([str(x) for x in [remote_module_path, async_jid, async_limit, async_module, argsfile]])
|
cmd = " ".join([str(x) for x in [remote_module_path, async_jid, async_limit, async_module, argsfile]])
|
||||||
|
|
||||||
res = self._low_level_exec_command(conn, cmd, tmp, sudoable=True)
|
res = self._low_level_exec_command(conn, cmd, tmp, sudoable=True)
|
||||||
|
|
||||||
return ReturnData(host=conn.host, result=res)
|
return ReturnData(host=conn.host, result=res)
|
||||||
|
|
||||||
# *****************************************************
|
# *****************************************************
|
||||||
|
@ -324,16 +309,11 @@ class Runner(object):
|
||||||
''' allows discovered variables to be used in templates and action statements '''
|
''' allows discovered variables to be used in templates and action statements '''
|
||||||
|
|
||||||
host = conn.host
|
host = conn.host
|
||||||
if 'ansible_facts' in result:
|
var_result = result.get('ansible_facts',{})
|
||||||
var_result = result['ansible_facts']
|
|
||||||
else:
|
|
||||||
var_result = {}
|
|
||||||
|
|
||||||
# note: do not allow variables from playbook to be stomped on
|
# note: do not allow variables from playbook to be stomped on
|
||||||
# by variables coming up from facter/ohai/etc. They
|
# by variables coming up from facter/ohai/etc. They
|
||||||
# should be prefixed anyway
|
# should be prefixed anyway
|
||||||
if not host in self.setup_cache:
|
|
||||||
self.setup_cache[host] = {}
|
|
||||||
for (k, v) in var_result.iteritems():
|
for (k, v) in var_result.iteritems():
|
||||||
if not k in self.setup_cache[host]:
|
if not k in self.setup_cache[host]:
|
||||||
self.setup_cache[host][k] = v
|
self.setup_cache[host][k] = v
|
||||||
|
@ -342,9 +322,9 @@ class Runner(object):
|
||||||
|
|
||||||
def _execute_raw(self, conn, tmp):
|
def _execute_raw(self, conn, tmp):
|
||||||
''' execute a non-module command for bootstrapping, or if there's no python on a device '''
|
''' execute a non-module command for bootstrapping, or if there's no python on a device '''
|
||||||
|
return ReturnData(host=conn.host, result=dict(
|
||||||
stdout=self._low_level_exec_command(conn, self.module_args, tmp, sudoable = True)
|
stdout=self._low_level_exec_command(conn, self.module_args, tmp, sudoable = True)
|
||||||
data = dict(stdout=stdout)
|
))
|
||||||
return ReturnData(host=conn.host, result=data)
|
|
||||||
|
|
||||||
# ***************************************************
|
# ***************************************************
|
||||||
|
|
||||||
|
@ -396,7 +376,7 @@ class Runner(object):
|
||||||
return ReturnData(host=conn.host, result=result)
|
return ReturnData(host=conn.host, result=result)
|
||||||
|
|
||||||
# apply templating to source argument
|
# apply templating to source argument
|
||||||
inject = self.setup_cache.get(conn.host,{})
|
inject = self.setup_cache[conn.host]
|
||||||
|
|
||||||
# if we have first_available_file in our vars
|
# if we have first_available_file in our vars
|
||||||
# look up the files and use the first one we find as src
|
# look up the files and use the first one we find as src
|
||||||
|
@ -462,7 +442,7 @@ class Runner(object):
|
||||||
return ReturnData(host=conn.host, result=results)
|
return ReturnData(host=conn.host, result=results)
|
||||||
|
|
||||||
# apply templating to source argument
|
# apply templating to source argument
|
||||||
inject = self.setup_cache.get(conn.host,{})
|
inject = self.setup_cache[conn.host]
|
||||||
if self.module_vars is not None:
|
if self.module_vars is not None:
|
||||||
inject.update(self.module_vars)
|
inject.update(self.module_vars)
|
||||||
source = utils.template(source, inject, self.setup_cache)
|
source = utils.template(source, inject, self.setup_cache)
|
||||||
|
@ -538,7 +518,7 @@ class Runner(object):
|
||||||
return ReturnData(host=conn.host, comm_ok=False, result=result)
|
return ReturnData(host=conn.host, comm_ok=False, result=result)
|
||||||
|
|
||||||
# apply templating to source argument so vars can be used in the path
|
# apply templating to source argument so vars can be used in the path
|
||||||
inject = self.setup_cache.get(conn.host,{})
|
inject = self.setup_cache[conn.host]
|
||||||
|
|
||||||
# if we have first_available_file in our vars
|
# if we have first_available_file in our vars
|
||||||
# look up the files and use the first one we find as src
|
# look up the files and use the first one we find as src
|
||||||
|
@ -598,6 +578,8 @@ class Runner(object):
|
||||||
# *****************************************************
|
# *****************************************************
|
||||||
|
|
||||||
def _executor(self, host):
|
def _executor(self, host):
|
||||||
|
''' handler for multiprocessing library '''
|
||||||
|
|
||||||
try:
|
try:
|
||||||
exec_rc = self._executor_internal(host)
|
exec_rc = self._executor_internal(host)
|
||||||
if type(exec_rc) != ReturnData:
|
if type(exec_rc) != ReturnData:
|
||||||
|
@ -622,7 +604,7 @@ class Runner(object):
|
||||||
host_variables = self.inventory.get_variables(host)
|
host_variables = self.inventory.get_variables(host)
|
||||||
port = host_variables.get('ansible_ssh_port', self.remote_port)
|
port = host_variables.get('ansible_ssh_port', self.remote_port)
|
||||||
|
|
||||||
inject = self.setup_cache.get(host,{}).copy()
|
inject = self.setup_cache[host].copy()
|
||||||
inject.update(host_variables)
|
inject.update(host_variables)
|
||||||
inject.update(self.module_vars)
|
inject.update(self.module_vars)
|
||||||
|
|
||||||
|
@ -644,16 +626,9 @@ class Runner(object):
|
||||||
tmp = self._make_tmp_path(conn)
|
tmp = self._make_tmp_path(conn)
|
||||||
result = None
|
result = None
|
||||||
|
|
||||||
if self.module_name == 'copy':
|
handler = getattr(self, "_execute_%s" % self.module_name, None)
|
||||||
result = self._execute_copy(conn, tmp)
|
if handler:
|
||||||
elif self.module_name == 'fetch':
|
result = handler(conn, tmp)
|
||||||
result = self._execute_fetch(conn, tmp)
|
|
||||||
elif self.module_name == 'template':
|
|
||||||
result = self._execute_template(conn, tmp)
|
|
||||||
elif self.module_name == 'raw':
|
|
||||||
result = self._execute_raw(conn, tmp)
|
|
||||||
elif self.module_name == 'assemble':
|
|
||||||
result = self._execute_assemble(conn, tmp)
|
|
||||||
else:
|
else:
|
||||||
if self.background == 0:
|
if self.background == 0:
|
||||||
result = self._execute_normal_module(conn, tmp, module_name)
|
result = self._execute_normal_module(conn, tmp, module_name)
|
||||||
|
@ -681,25 +656,22 @@ class Runner(object):
|
||||||
|
|
||||||
def _low_level_exec_command(self, conn, cmd, tmp, sudoable=False):
|
def _low_level_exec_command(self, conn, cmd, tmp, sudoable=False):
|
||||||
''' execute a command string over SSH, return the output '''
|
''' execute a command string over SSH, return the output '''
|
||||||
|
|
||||||
sudo_user = self.sudo_user
|
sudo_user = self.sudo_user
|
||||||
stdin, stdout, stderr = conn.exec_command(cmd, tmp, sudo_user, sudoable=sudoable)
|
stdin, stdout, stderr = conn.exec_command(cmd, tmp, sudo_user, sudoable=sudoable)
|
||||||
out=None
|
|
||||||
|
|
||||||
if type(stdout) != str:
|
if type(stdout) != str:
|
||||||
out="\n".join(stdout.readlines())
|
return "\n".join(stdout.readlines())
|
||||||
else:
|
else:
|
||||||
out=stdout
|
return stdout
|
||||||
|
|
||||||
# sudo mode paramiko doesn't capture stderr, so not relaying here either...
|
|
||||||
return out
|
|
||||||
|
|
||||||
# *****************************************************
|
# *****************************************************
|
||||||
|
|
||||||
def _remote_md5(self, conn, tmp, path):
|
def _remote_md5(self, conn, tmp, path):
|
||||||
'''
|
'''
|
||||||
takes a remote md5sum without requiring python, and returns 0 if the
|
takes a remote md5sum without requiring python, and returns 0 if no file
|
||||||
file does not exist
|
|
||||||
'''
|
'''
|
||||||
|
|
||||||
test = "[[ -r %s ]]" % path
|
test = "[[ -r %s ]]" % path
|
||||||
md5s = [
|
md5s = [
|
||||||
"(%s && /usr/bin/md5sum %s 2>/dev/null)" % (test,path),
|
"(%s && /usr/bin/md5sum %s 2>/dev/null)" % (test,path),
|
||||||
|
@ -708,8 +680,7 @@ class Runner(object):
|
||||||
]
|
]
|
||||||
cmd = " || ".join(md5s)
|
cmd = " || ".join(md5s)
|
||||||
cmd = "%s || (echo \"0 %s\")" % (cmd, path)
|
cmd = "%s || (echo \"0 %s\")" % (cmd, path)
|
||||||
remote_md5 = self._low_level_exec_command(conn, cmd, tmp, True).split()[0]
|
return self._low_level_exec_command(conn, cmd, tmp, True).split()[0]
|
||||||
return remote_md5
|
|
||||||
|
|
||||||
# *****************************************************
|
# *****************************************************
|
||||||
|
|
||||||
|
@ -727,9 +698,7 @@ class Runner(object):
|
||||||
cmd += ' && echo %s' % basetmp
|
cmd += ' && echo %s' % basetmp
|
||||||
|
|
||||||
result = self._low_level_exec_command(conn, cmd, None, sudoable=False)
|
result = self._low_level_exec_command(conn, cmd, None, sudoable=False)
|
||||||
cleaned = result.split("\n")[0].strip() + '/'
|
return result.split("\n")[0].strip() + '/'
|
||||||
return cleaned
|
|
||||||
|
|
||||||
|
|
||||||
# *****************************************************
|
# *****************************************************
|
||||||
|
|
||||||
|
@ -770,7 +739,6 @@ class Runner(object):
|
||||||
|
|
||||||
job_queue = multiprocessing.Manager().Queue()
|
job_queue = multiprocessing.Manager().Queue()
|
||||||
[job_queue.put(i) for i in hosts]
|
[job_queue.put(i) for i in hosts]
|
||||||
|
|
||||||
result_queue = multiprocessing.Manager().Queue()
|
result_queue = multiprocessing.Manager().Queue()
|
||||||
|
|
||||||
workers = []
|
workers = []
|
||||||
|
@ -798,10 +766,9 @@ class Runner(object):
|
||||||
def _partition_results(self, results):
|
def _partition_results(self, results):
|
||||||
''' seperate results by ones we contacted & ones we didn't '''
|
''' seperate results by ones we contacted & ones we didn't '''
|
||||||
|
|
||||||
results2 = dict(contacted={}, dark={})
|
|
||||||
|
|
||||||
if results is None:
|
if results is None:
|
||||||
return None
|
return None
|
||||||
|
results2 = dict(contacted={}, dark={})
|
||||||
|
|
||||||
for result in results:
|
for result in results:
|
||||||
host = result.host
|
host = result.host
|
||||||
|
@ -816,7 +783,6 @@ class Runner(object):
|
||||||
for host in self.inventory.list_hosts(self.pattern):
|
for host in self.inventory.list_hosts(self.pattern):
|
||||||
if not (host in results2['dark'] or host in results2['contacted']):
|
if not (host in results2['dark'] or host in results2['contacted']):
|
||||||
results2["dark"][host] = {}
|
results2["dark"][host] = {}
|
||||||
|
|
||||||
return results2
|
return results2
|
||||||
|
|
||||||
# *****************************************************
|
# *****************************************************
|
||||||
|
@ -842,8 +808,8 @@ class Runner(object):
|
||||||
|
|
||||||
def run_async(self, time_limit):
|
def run_async(self, time_limit):
|
||||||
''' Run this module asynchronously and return a poller. '''
|
''' Run this module asynchronously and return a poller. '''
|
||||||
|
|
||||||
self.background = time_limit
|
self.background = time_limit
|
||||||
results = self.run()
|
results = self.run()
|
||||||
|
|
||||||
return results, poller.AsyncPoller(results, self)
|
return results, poller.AsyncPoller(results, self)
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue