From 2732cde031a80636dfda31e7f3491a680c01b0bb Mon Sep 17 00:00:00 2001 From: Matt Martz Date: Thu, 25 Apr 2019 10:28:18 -0500 Subject: [PATCH] Support using importlib on py>=3 to avoid imp deprecation (#54883) * Support using importlib on py>=3 to avoid imp deprecation * Add changelog fragment * importlib coverage for py3 * Ansiballz execute should use importlib too * recursive module_utils finder should utilize importlib too * don't be dumb * Fix up units * Clean up tests * Prefer importlib.util in plugin loader when available * insert the module into sys.modules * 3 before 2 for consistency * ci_complete * Address importlib.util.find_spec returning None --- .../fragments/ansiballz-imp-deprecation.yaml | 2 + lib/ansible/executor/module_common.py | 115 +++++++++++++----- lib/ansible/plugins/loader.py | 19 ++- .../module_common/test_recursive_finder.py | 47 +++---- 4 files changed, 120 insertions(+), 63 deletions(-) create mode 100644 changelogs/fragments/ansiballz-imp-deprecation.yaml diff --git a/changelogs/fragments/ansiballz-imp-deprecation.yaml b/changelogs/fragments/ansiballz-imp-deprecation.yaml new file mode 100644 index 0000000000..d09929a658 --- /dev/null +++ b/changelogs/fragments/ansiballz-imp-deprecation.yaml @@ -0,0 +1,2 @@ +bugfixes: +- AnsiballZ - Use ``importlib`` to load the module instead of ``imp`` on Python3+ diff --git a/lib/ansible/executor/module_common.py b/lib/ansible/executor/module_common.py index 87b7673a20..1253f76f98 100644 --- a/lib/ansible/executor/module_common.py +++ b/lib/ansible/executor/module_common.py @@ -23,7 +23,6 @@ __metaclass__ = type import ast import base64 import datetime -import imp import json import os import shlex @@ -46,6 +45,15 @@ from ansible.executor import action_write_locks from ansible.utils.display import Display + +try: + import importlib.util + import importlib.machinery + imp = None +except ImportError: + import imp + + # HACK: keep Python 2.6 controller tests happy in CI until they're properly split try: from importlib import import_module @@ -144,18 +152,20 @@ def _ansiballz_main(): sys.path = [p for p in sys.path if p != scriptdir] import base64 - import imp import shutil import tempfile import zipfile if sys.version_info < (3,): + # imp is used on Python<3 + import imp bytes = str MOD_DESC = ('.py', 'U', imp.PY_SOURCE) PY3 = False else: + # importlib is only used on Python>=3 + import importlib.util unicode = str - MOD_DESC = ('.py', 'r', imp.PY_SOURCE) PY3 = True ZIPDATA = """%(zipdata)s""" @@ -195,8 +205,13 @@ def _ansiballz_main(): basic._ANSIBLE_ARGS = json_params %(coverage)s # Run the module! By importing it as '__main__', it thinks it is executing as a script - with open(module, 'rb') as mod: - imp.load_module('__main__', mod, module, MOD_DESC) + if sys.version_info >= (3,): + spec = importlib.util.spec_from_file_location('__main__', module) + module = importlib.util.module_from_spec(spec) + spec.loader.exec_module(module) + else: + with open(module, 'rb') as mod: + imp.load_module('__main__', mod, module, MOD_DESC) # Ansible modules must exit themselves print('{"msg": "New-style module did not handle its own exit", "failed": true}') @@ -291,9 +306,15 @@ def _ansiballz_main(): basic._ANSIBLE_ARGS = json_params # Run the module! By importing it as '__main__', it thinks it is executing as a script - import imp - with open(script_path, 'r') as f: - importer = imp.load_module('__main__', f, script_path, ('.py', 'r', imp.PY_SOURCE)) + if PY3: + import importlib.util + spec = importlib.util.spec_from_file_location('__main__', script_path) + module = importlib.util.module_from_spec(spec) + spec.loader.exec_module(module) + else: + import imp + with open(script_path, 'r') as f: + imp.load_module('__main__', f, script_path, ('.py', 'r', imp.PY_SOURCE)) # Ansible modules must exit themselves print('{"msg": "New-style module did not handle its own exit", "failed": true}') @@ -372,7 +393,11 @@ ANSIBALLZ_COVERAGE_TEMPLATE = ''' ANSIBALLZ_COVERAGE_CHECK_TEMPLATE = ''' try: - imp.find_module('coverage') + if PY3: + if importlib.util.find_spec('coverage') is None: + raise ImportError + else: + imp.find_module('coverage') except ImportError: print('{"msg": "Could not find `coverage` module.", "failed": true}') sys.exit(1) @@ -488,9 +513,8 @@ class ModuleDepFinder(ast.NodeVisitor): def _slurp(path): if not os.path.exists(path): raise AnsibleError("imported module support code does not exist at %s" % os.path.abspath(path)) - fd = open(path, 'rb') - data = fd.read() - fd.close() + with open(path, 'rb') as fd: + data = fd.read() return data @@ -544,6 +568,40 @@ def _get_shebang(interpreter, task_vars, templar, args=tuple()): return shebang, interpreter_out +class ModuleInfo: + def __init__(self, name, paths): + self.py_src = False + self.pkg_dir = False + path = None + + if imp is None: + self._info = info = importlib.machinery.PathFinder.find_spec(name, paths) + if info is not None: + self.py_src = os.path.splitext(info.origin)[1] in importlib.machinery.SOURCE_SUFFIXES + self.pkg_dir = info.origin.endswith('/__init__.py') + path = info.origin + else: + raise ImportError("No module named '%s'" % name) + else: + self._info = info = imp.find_module(name, paths) + self.py_src = info[2][2] == imp.PY_SOURCE + self.pkg_dir = info[2][2] == imp.PKG_DIRECTORY + if self.pkg_dir: + path = os.path.join(info[1], '__init__.py') + else: + path = info[1] + + self.path = path + + def get_source(self): + if imp and self.py_src: + try: + return self._info[0].read() + finally: + self._info[0].close() + return _slurp(self.path) + + def recursive_finder(name, data, py_module_names, py_module_cache, zf): """ Using ModuleDepFinder, make sure we have all of the module_utils files that @@ -575,13 +633,13 @@ def recursive_finder(name, data, py_module_names, py_module_cache, zf): if py_module_name[0] == 'six': # Special case the python six library because it messes up the # import process in an incompatible way - module_info = imp.find_module('six', module_utils_paths) + module_info = ModuleInfo('six', module_utils_paths) py_module_name = ('six',) idx = 0 elif py_module_name[0] == '_six': # Special case the python six library because it messes up the # import process in an incompatible way - module_info = imp.find_module('_six', [os.path.join(p, 'six') for p in module_utils_paths]) + module_info = ModuleInfo('_six', [os.path.join(p, 'six') for p in module_utils_paths]) py_module_name = ('six', '_six') idx = 0 elif py_module_name[0] == 'ansible_collections': @@ -605,8 +663,8 @@ def recursive_finder(name, data, py_module_names, py_module_cache, zf): if len(py_module_name) < idx: break try: - module_info = imp.find_module(py_module_name[-idx], - [os.path.join(p, *py_module_name[:-idx]) for p in module_utils_paths]) + module_info = ModuleInfo(py_module_name[-idx], + [os.path.join(p, *py_module_name[:-idx]) for p in module_utils_paths]) break except ImportError: continue @@ -647,7 +705,7 @@ def recursive_finder(name, data, py_module_names, py_module_cache, zf): # imp.find_module seems to prefer to return source packages so we just # error out if imp.find_module returns byte compiled files (This is # fragile as it depends on undocumented imp.find_module behaviour) - if module_info[2][2] not in (imp.PY_SOURCE, imp.PKG_DIRECTORY): + if not module_info.pkg_dir and not module_info.py_src: msg = ['Could not find python source for imported module support code for %s. Looked for' % name] if idx == 2: msg.append('either %s.py or %s.py' % (py_module_name[-1], py_module_name[-2])) @@ -665,22 +723,19 @@ def recursive_finder(name, data, py_module_names, py_module_cache, zf): # We already have a file handle for the module open so it makes # sense to read it now if py_module_name not in py_module_cache: - if module_info[2][2] == imp.PKG_DIRECTORY: + if module_info.pkg_dir: # Read the __init__.py instead of the module file as this is # a python package normalized_name = py_module_name + ('__init__',) if normalized_name not in py_module_names: - normalized_path = os.path.join(module_info[1], '__init__.py') - normalized_data = _slurp(normalized_path) - py_module_cache[normalized_name] = (normalized_data, normalized_path) + normalized_data = module_info.get_source() + py_module_cache[normalized_name] = (normalized_data, module_info.path) normalized_modules.add(normalized_name) else: normalized_name = py_module_name if normalized_name not in py_module_names: - normalized_path = module_info[1] - normalized_data = module_info[0].read() - module_info[0].close() - py_module_cache[normalized_name] = (normalized_data, normalized_path) + normalized_data = module_info.get_source() + py_module_cache[normalized_name] = (normalized_data, module_info.path) normalized_modules.add(normalized_name) # Make sure that all the packages that this module is a part of @@ -688,10 +743,10 @@ def recursive_finder(name, data, py_module_names, py_module_cache, zf): for i in range(1, len(py_module_name)): py_pkg_name = py_module_name[:-i] + ('__init__',) if py_pkg_name not in py_module_names: - pkg_dir_info = imp.find_module(py_pkg_name[-1], - [os.path.join(p, *py_pkg_name[:-1]) for p in module_utils_paths]) + pkg_dir_info = ModuleInfo(py_pkg_name[-1], + [os.path.join(p, *py_pkg_name[:-1]) for p in module_utils_paths]) normalized_modules.add(py_pkg_name) - py_module_cache[py_pkg_name] = (_slurp(pkg_dir_info[1]), pkg_dir_info[1]) + py_module_cache[py_pkg_name] = (pkg_dir_info.get_source(), pkg_dir_info.path) # FIXME: Currently the AnsiBallZ wrapper monkeypatches module args into a global # variable in basic.py. If a module doesn't import basic.py, then the AnsiBallZ wrapper will @@ -704,9 +759,9 @@ def recursive_finder(name, data, py_module_names, py_module_cache, zf): # from the separate python module and mirror the args into its global variable for backwards # compatibility. if ('basic',) not in py_module_names: - pkg_dir_info = imp.find_module('basic', module_utils_paths) + pkg_dir_info = ModuleInfo('basic', module_utils_paths) normalized_modules.add(('basic',)) - py_module_cache[('basic',)] = (_slurp(pkg_dir_info[1]), pkg_dir_info[1]) + py_module_cache[('basic',)] = (pkg_dir_info.get_source(), pkg_dir_info.path) # End of AnsiballZ hack # diff --git a/lib/ansible/plugins/loader.py b/lib/ansible/plugins/loader.py index f245b7b30f..7a5bc075a9 100644 --- a/lib/ansible/plugins/loader.py +++ b/lib/ansible/plugins/loader.py @@ -8,7 +8,6 @@ from __future__ import (absolute_import, division, print_function) __metaclass__ = type import glob -import imp import os import os.path import pkgutil @@ -28,6 +27,12 @@ from ansible.utils.collection_loader import AnsibleCollectionLoader, AnsibleFlat from ansible.utils.display import Display from ansible.utils.plugin_docs import add_fragments +try: + import importlib.util + imp = None +except ImportError: + import imp + # HACK: keep Python 2.6 controller tests happy in CI until they're properly split try: from importlib import import_module @@ -535,9 +540,15 @@ class PluginLoader: with warnings.catch_warnings(): warnings.simplefilter("ignore", RuntimeWarning) - with open(to_bytes(path), 'rb') as module_file: - # to_native is used here because imp.load_source's path is for tracebacks and python's traceback formatting uses native strings - module = imp.load_source(to_native(full_name), to_native(path), module_file) + if imp is None: + spec = importlib.util.spec_from_file_location(to_native(full_name), to_native(path)) + module = importlib.util.module_from_spec(spec) + spec.loader.exec_module(module) + sys.modules[full_name] = module + else: + with open(to_bytes(path), 'rb') as module_file: + # to_native is used here because imp.load_source's path is for tracebacks and python's traceback formatting uses native strings + module = imp.load_source(to_native(full_name), to_native(path), module_file) return module def _update_object(self, obj, name, path): diff --git a/test/units/executor/module_common/test_recursive_finder.py b/test/units/executor/module_common/test_recursive_finder.py index f0724199ee..f2dc5e3183 100644 --- a/test/units/executor/module_common/test_recursive_finder.py +++ b/test/units/executor/module_common/test_recursive_finder.py @@ -19,23 +19,18 @@ from __future__ import (absolute_import, division, print_function) __metaclass__ = type -import imp import pytest import zipfile from collections import namedtuple -from functools import partial -from io import BytesIO, StringIO +from io import BytesIO import ansible.errors from ansible.executor.module_common import recursive_finder from ansible.module_utils.six import PY2 -from ansible.module_utils.six.moves import builtins -original_find_module = imp.find_module - # These are the modules that are brought in by module_utils/basic.py This may need to be updated # when basic.py gains new imports # We will remove these when we modify AnsiBallZ to store its args in a separate file instead of in @@ -108,18 +103,6 @@ def finder_containers(): return FinderContainers(py_module_names, py_module_cache, zf) -def find_module_foo(module_utils_data, *args, **kwargs): - if args[0] == 'foo': - return (module_utils_data, '/usr/lib/python2.7/site-packages/ansible/module_utils/foo.py', ('.py', 'r', imp.PY_SOURCE)) - return original_find_module(*args, **kwargs) - - -def find_package_foo(module_utils_data, *args, **kwargs): - if args[0] == 'foo': - return (module_utils_data, '/usr/lib/python2.7/site-packages/ansible/module_utils/foo', ('', '', imp.PKG_DIRECTORY)) - return original_find_module(*args, **kwargs) - - class TestRecursiveFinder(object): def test_no_module_utils(self, finder_containers): name = 'ping' @@ -145,11 +128,15 @@ class TestRecursiveFinder(object): def test_from_import_toplevel_package(self, finder_containers, mocker): if PY2: - module_utils_data = BytesIO(b'# License\ndef do_something():\n pass\n') + module_utils_data = b'# License\ndef do_something():\n pass\n' else: - module_utils_data = StringIO(u'# License\ndef do_something():\n pass\n') - mocker.patch('imp.find_module', side_effect=partial(find_package_foo, module_utils_data)) - mocker.patch('ansible.executor.module_common._slurp', side_effect=lambda x: b'# License\ndef do_something():\n pass\n') + module_utils_data = u'# License\ndef do_something():\n pass\n' + mi_mock = mocker.patch('ansible.executor.module_common.ModuleInfo') + mi_inst = mi_mock() + mi_inst.pkg_dir = True + mi_inst.py_src = False + mi_inst.path = '/path/to/ansible/module_utils/foo/__init__.py' + mi_inst.get_source.return_value = module_utils_data name = 'ping' data = b'#!/usr/bin/python\nfrom ansible.module_utils import foo' @@ -161,20 +148,22 @@ class TestRecursiveFinder(object): assert frozenset(finder_containers.zf.namelist()) == frozenset(('ansible/module_utils/foo/__init__.py',)).union(ONLY_BASIC_FILE) def test_from_import_toplevel_module(self, finder_containers, mocker): - if PY2: - module_utils_data = BytesIO(b'# License\ndef do_something():\n pass\n') - else: - module_utils_data = StringIO(u'# License\ndef do_something():\n pass\n') - mocker.patch('imp.find_module', side_effect=partial(find_module_foo, module_utils_data)) + module_utils_data = b'# License\ndef do_something():\n pass\n' + mi_mock = mocker.patch('ansible.executor.module_common.ModuleInfo') + mi_inst = mi_mock() + mi_inst.pkg_dir = False + mi_inst.py_src = True + mi_inst.path = '/path/to/ansible/module_utils/foo.py' + mi_inst.get_source.return_value = module_utils_data name = 'ping' data = b'#!/usr/bin/python\nfrom ansible.module_utils import foo' recursive_finder(name, data, *finder_containers) mocker.stopall() - assert finder_containers.py_module_names == set((('foo',),)).union(MODULE_UTILS_BASIC_IMPORTS) + assert finder_containers.py_module_names == set((('foo',),)).union(ONLY_BASIC_IMPORT) assert finder_containers.py_module_cache == {} - assert frozenset(finder_containers.zf.namelist()) == frozenset(('ansible/module_utils/foo.py',)).union(MODULE_UTILS_BASIC_FILES) + assert frozenset(finder_containers.zf.namelist()) == frozenset(('ansible/module_utils/foo.py',)).union(ONLY_BASIC_FILE) # # Test importing six with many permutations because it is not a normal module