mirror of
https://github.com/ansible-collections/community.general.git
synced 2024-09-14 20:13:21 +02:00
Enable imports to work on a snippet based system, allowing for instance a library of common EC2 functions
to be reused between modules. See library/system/service and library/system/ping for initial examples. Can work the old way to just import 'basic', or can import the new way to import multiple pieces of code from module_utils/.
This commit is contained in:
parent
43f48a2e02
commit
9858b1f2f3
6 changed files with 51 additions and 98 deletions
|
@ -77,39 +77,32 @@ def write_argsfile(argstring, json=False):
|
||||||
def boilerplate_module(modfile, args):
|
def boilerplate_module(modfile, args):
|
||||||
""" simulate what ansible does with new style modules """
|
""" simulate what ansible does with new style modules """
|
||||||
|
|
||||||
module_fh = open(modfile)
|
#module_fh = open(modfile)
|
||||||
module_data = module_fh.read()
|
#module_data = module_fh.read()
|
||||||
included_boilerplate = module_data.find(module_common.REPLACER) != -1
|
#module_fh.close()
|
||||||
module_fh.close()
|
|
||||||
|
|
||||||
if included_boilerplate:
|
replacer = module_common.ModuleReplacer()
|
||||||
|
|
||||||
module_data = module_data.replace(module_common.REPLACER, module_common.MODULE_COMMON)
|
#included_boilerplate = module_data.find(module_common.REPLACER) != -1 or module_data.find("import ansible.module_utils") != -1
|
||||||
encoded_args = repr(str(args))
|
|
||||||
module_data = module_data.replace(module_common.REPLACER_ARGS, encoded_args)
|
|
||||||
encoded_lang = repr(C.DEFAULT_MODULE_LANG)
|
|
||||||
empty_complex = repr("{}")
|
|
||||||
module_data = module_data.replace(module_common.REPLACER_LANG, encoded_lang)
|
|
||||||
module_data = module_data.replace('syslog.LOG_USER', "syslog.%s" % C.DEFAULT_SYSLOG_FACILITY)
|
|
||||||
module_data = module_data.replace(module_common.REPLACER_COMPLEX, empty_complex)
|
|
||||||
|
|
||||||
modfile2_path = os.path.expanduser("~/.ansible_module_generated")
|
complex_args = {}
|
||||||
print "* including generated source, if any, saving to: %s" % modfile2_path
|
inject = {}
|
||||||
print "* this will offset any line numbers in tracebacks/debuggers!"
|
(module_data, module_style, shebang) = replacer.modify_module(
|
||||||
modfile2 = open(modfile2_path, 'w')
|
modfile,
|
||||||
modfile2.write(module_data)
|
complex_args,
|
||||||
modfile2.close()
|
args,
|
||||||
modfile = modfile2_path
|
inject
|
||||||
|
)
|
||||||
|
|
||||||
return (modfile2_path, included_boilerplate, False)
|
modfile2_path = os.path.expanduser("~/.ansible_module_generated")
|
||||||
else:
|
print "* including generated source, if any, saving to: %s" % modfile2_path
|
||||||
|
print "* this may offset any line numbers in tracebacks/debuggers!"
|
||||||
|
modfile2 = open(modfile2_path, 'w')
|
||||||
|
modfile2.write(module_data)
|
||||||
|
modfile2.close()
|
||||||
|
modfile = modfile2_path
|
||||||
|
|
||||||
old_style_but_json = False
|
return (modfile2_path, module_style)
|
||||||
if 'WANT_JSON' in module_data:
|
|
||||||
old_style_but_json = True
|
|
||||||
|
|
||||||
print "* module boilerplate substitution not requested in module, line numbers will be unaltered"
|
|
||||||
return (modfile, included_boilerplate, old_style_but_json)
|
|
||||||
|
|
||||||
def runtest( modfile, argspath):
|
def runtest( modfile, argspath):
|
||||||
"""Test run a module, piping it's output for reporting."""
|
"""Test run a module, piping it's output for reporting."""
|
||||||
|
@ -151,14 +144,16 @@ def rundebug(debugger, modfile, argspath):
|
||||||
def main():
|
def main():
|
||||||
|
|
||||||
options, args = parse()
|
options, args = parse()
|
||||||
(modfile, is_new_style, old_style_but_json) = boilerplate_module(options.module_path, options.module_args)
|
(modfile, module_style) = boilerplate_module(options.module_path, options.module_args)
|
||||||
|
|
||||||
argspath=None
|
argspath=None
|
||||||
if not is_new_style:
|
if module_style != 'new':
|
||||||
if old_style_but_json:
|
if module_style == 'non_native_want_json':
|
||||||
argspath = write_argsfile(options.module_args, json=True)
|
argspath = write_argsfile(options.module_args, json=True)
|
||||||
else:
|
elif module_style == 'old':
|
||||||
argspath = write_argsfile(options.module_args, json=False)
|
argspath = write_argsfile(options.module_args, json=False)
|
||||||
|
else:
|
||||||
|
raise Exception("internal error, unexpected module style: %s" % module_style)
|
||||||
if options.debugger:
|
if options.debugger:
|
||||||
rundebug(options.debugger, modfile, argspath)
|
rundebug(options.debugger, modfile, argspath)
|
||||||
else:
|
else:
|
||||||
|
|
|
@ -27,18 +27,11 @@
|
||||||
# USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
# USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
#
|
#
|
||||||
|
|
||||||
REPLACER = "#<<INCLUDE_ANSIBLE_MODULE_COMMON>>"
|
|
||||||
REPLACER_ARGS = "<<INCLUDE_ANSIBLE_MODULE_ARGS>>"
|
|
||||||
REPLACER_LANG = "<<INCLUDE_ANSIBLE_MODULE_LANG>>"
|
|
||||||
REPLACER_COMPLEX = "<<INCLUDE_ANSIBLE_MODULE_COMPLEX_ARGS>>"
|
|
||||||
|
|
||||||
MODULE_COMMON = """
|
|
||||||
|
|
||||||
# == BEGIN DYNAMICALLY INSERTED CODE ==
|
# == BEGIN DYNAMICALLY INSERTED CODE ==
|
||||||
|
|
||||||
MODULE_ARGS = <<INCLUDE_ANSIBLE_MODULE_ARGS>>
|
MODULE_ARGS = "<<INCLUDE_ANSIBLE_MODULE_ARGS>>"
|
||||||
MODULE_LANG = <<INCLUDE_ANSIBLE_MODULE_LANG>>
|
MODULE_LANG = "<<INCLUDE_ANSIBLE_MODULE_LANG>>"
|
||||||
MODULE_COMPLEX_ARGS = <<INCLUDE_ANSIBLE_MODULE_COMPLEX_ARGS>>
|
MODULE_COMPLEX_ARGS = "<<INCLUDE_ANSIBLE_MODULE_COMPLEX_ARGS>>"
|
||||||
|
|
||||||
BOOLEANS_TRUE = ['yes', 'on', '1', 'true', 1]
|
BOOLEANS_TRUE = ['yes', 'on', '1', 'true', 1]
|
||||||
BOOLEANS_FALSE = ['no', 'off', '0', 'false', 0]
|
BOOLEANS_FALSE = ['no', 'off', '0', 'false', 0]
|
||||||
|
@ -963,7 +956,3 @@ class AnsibleModule(object):
|
||||||
if size >= limit:
|
if size >= limit:
|
||||||
break
|
break
|
||||||
return '%.2f %s' % (float(size)/ limit, suffix)
|
return '%.2f %s' % (float(size)/ limit, suffix)
|
||||||
|
|
||||||
# == END DYNAMICALLY INSERTED CODE ===
|
|
||||||
|
|
||||||
"""
|
|
|
@ -40,6 +40,7 @@ from ansible.utils import template
|
||||||
from ansible.utils import check_conditional
|
from ansible.utils import check_conditional
|
||||||
from ansible import errors
|
from ansible import errors
|
||||||
from ansible import module_common
|
from ansible import module_common
|
||||||
|
from ansible.module_common import ModuleReplacer
|
||||||
import poller
|
import poller
|
||||||
import connection
|
import connection
|
||||||
from return_data import ReturnData
|
from return_data import ReturnData
|
||||||
|
@ -51,6 +52,7 @@ try:
|
||||||
except ImportError:
|
except ImportError:
|
||||||
HAS_ATFORK=False
|
HAS_ATFORK=False
|
||||||
|
|
||||||
|
module_replacer = ModuleReplacer(strip_comments=False)
|
||||||
multiprocessing_runner = None
|
multiprocessing_runner = None
|
||||||
|
|
||||||
OUTPUT_LOCKFILE = tempfile.TemporaryFile()
|
OUTPUT_LOCKFILE = tempfile.TemporaryFile()
|
||||||
|
@ -825,60 +827,19 @@ class Runner(object):
|
||||||
def _copy_module(self, conn, tmp, module_name, module_args, inject, complex_args=None):
|
def _copy_module(self, conn, tmp, module_name, module_args, inject, complex_args=None):
|
||||||
''' transfer a module over SFTP, does not run it '''
|
''' transfer a module over SFTP, does not run it '''
|
||||||
|
|
||||||
# FIXME if complex args is none, set to {}
|
|
||||||
|
|
||||||
if module_name.startswith("/"):
|
|
||||||
raise errors.AnsibleFileNotFound("%s is not a module" % module_name)
|
|
||||||
|
|
||||||
# Search module path(s) for named module.
|
# Search module path(s) for named module.
|
||||||
in_path = utils.plugins.module_finder.find_plugin(module_name)
|
in_path = utils.plugins.module_finder.find_plugin(module_name)
|
||||||
if in_path is None:
|
if in_path is None:
|
||||||
raise errors.AnsibleFileNotFound("module %s not found in %s" % (module_name, utils.plugins.module_finder.print_paths()))
|
raise errors.AnsibleFileNotFound("module %s not found in %s" % (module_name, utils.plugins.module_finder.print_paths()))
|
||||||
|
|
||||||
out_path = os.path.join(tmp, module_name)
|
out_path = os.path.join(tmp, module_name)
|
||||||
|
|
||||||
module_data = ""
|
# insert shared code and arguments into the module
|
||||||
module_style = 'old'
|
(module_data, module_style, shebang) = module_replacer.modify_module(
|
||||||
|
in_path, complex_args, module_args, inject
|
||||||
with open(in_path) as f:
|
)
|
||||||
module_data = f.read()
|
|
||||||
if module_common.REPLACER in module_data:
|
|
||||||
module_style = 'new'
|
|
||||||
if 'WANT_JSON' in module_data:
|
|
||||||
module_style = 'non_native_want_json'
|
|
||||||
|
|
||||||
complex_args_json = utils.jsonify(complex_args)
|
|
||||||
# We force conversion of module_args to str because module_common calls shlex.split,
|
|
||||||
# a standard library function that incorrectly handles Unicode input before Python 2.7.3.
|
|
||||||
encoded_args = repr(module_args.encode('utf-8'))
|
|
||||||
encoded_lang = repr(C.DEFAULT_MODULE_LANG)
|
|
||||||
encoded_complex = repr(complex_args_json)
|
|
||||||
|
|
||||||
module_data = module_data.replace(module_common.REPLACER, module_common.MODULE_COMMON)
|
|
||||||
module_data = module_data.replace(module_common.REPLACER_ARGS, encoded_args)
|
|
||||||
module_data = module_data.replace(module_common.REPLACER_LANG, encoded_lang)
|
|
||||||
module_data = module_data.replace(module_common.REPLACER_COMPLEX, encoded_complex)
|
|
||||||
|
|
||||||
if module_style == 'new':
|
|
||||||
facility = C.DEFAULT_SYSLOG_FACILITY
|
|
||||||
if 'ansible_syslog_facility' in inject:
|
|
||||||
facility = inject['ansible_syslog_facility']
|
|
||||||
module_data = module_data.replace('syslog.LOG_USER', "syslog.%s" % facility)
|
|
||||||
|
|
||||||
lines = module_data.split("\n")
|
|
||||||
shebang = None
|
|
||||||
if lines[0].startswith("#!"):
|
|
||||||
shebang = lines[0].strip()
|
|
||||||
args = shlex.split(str(shebang[2:]))
|
|
||||||
interpreter = args[0]
|
|
||||||
interpreter_config = 'ansible_%s_interpreter' % os.path.basename(interpreter)
|
|
||||||
|
|
||||||
if interpreter_config in inject:
|
|
||||||
lines[0] = shebang = "#!%s %s" % (inject[interpreter_config], " ".join(args[1:]))
|
|
||||||
module_data = "\n".join(lines)
|
|
||||||
|
|
||||||
|
# ship the module
|
||||||
self._transfer_str(conn, tmp, module_name, module_data)
|
self._transfer_str(conn, tmp, module_name, module_data)
|
||||||
|
|
||||||
return (out_path, module_style, shebang)
|
return (out_path, module_style, shebang)
|
||||||
|
|
||||||
# *****************************************************
|
# *****************************************************
|
||||||
|
|
|
@ -462,7 +462,7 @@ def template_from_file(basedir, path, vars):
|
||||||
res = res + '\n'
|
res = res + '\n'
|
||||||
return template(basedir, res, vars)
|
return template(basedir, res, vars)
|
||||||
|
|
||||||
def template_from_string(basedir, data, vars, fail_on_undefined=False, lookups=True):
|
def template_from_string(basedir, data, vars, fail_on_undefined=False, lookups=True, filters=True):
|
||||||
''' run a string through the (Jinja2) templating engine '''
|
''' run a string through the (Jinja2) templating engine '''
|
||||||
|
|
||||||
def my_lookup(*args, **kwargs):
|
def my_lookup(*args, **kwargs):
|
||||||
|
@ -472,7 +472,9 @@ def template_from_string(basedir, data, vars, fail_on_undefined=False, lookups=T
|
||||||
if type(data) == str:
|
if type(data) == str:
|
||||||
data = unicode(data, 'utf-8')
|
data = unicode(data, 'utf-8')
|
||||||
environment = jinja2.Environment(trim_blocks=True, undefined=StrictUndefined, extensions=_get_extensions())
|
environment = jinja2.Environment(trim_blocks=True, undefined=StrictUndefined, extensions=_get_extensions())
|
||||||
environment.filters.update(_get_filters())
|
|
||||||
|
if filters:
|
||||||
|
environment.filters.update(_get_filters())
|
||||||
|
|
||||||
if '_original_file' in vars:
|
if '_original_file' in vars:
|
||||||
basedir = os.path.dirname(vars['_original_file'])
|
basedir = os.path.dirname(vars['_original_file'])
|
||||||
|
|
|
@ -48,7 +48,9 @@ def main():
|
||||||
result['ping'] = module.params['data']
|
result['ping'] = module.params['data']
|
||||||
module.exit_json(**result)
|
module.exit_json(**result)
|
||||||
|
|
||||||
# this is magic, see lib/ansible/module_common.py
|
### boilerplate: import common module snippets here
|
||||||
#<<INCLUDE_ANSIBLE_MODULE_COMMON>>
|
from ansible.module_utils.basic import *
|
||||||
|
|
||||||
|
### invoke the module
|
||||||
main()
|
main()
|
||||||
|
|
||||||
|
|
|
@ -1205,7 +1205,11 @@ def main():
|
||||||
|
|
||||||
module.exit_json(**result)
|
module.exit_json(**result)
|
||||||
|
|
||||||
# this is magic, see lib/ansible/module_common.py
|
### boilerplate: import common module snippets here
|
||||||
#<<INCLUDE_ANSIBLE_MODULE_COMMON>>
|
from ansible.module_utils.basic import *
|
||||||
|
|
||||||
|
### invoke the module
|
||||||
main()
|
main()
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue