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
This commit is contained in:
parent
6d645c127f
commit
2732cde031
4 changed files with 120 additions and 63 deletions
2
changelogs/fragments/ansiballz-imp-deprecation.yaml
Normal file
2
changelogs/fragments/ansiballz-imp-deprecation.yaml
Normal file
|
@ -0,0 +1,2 @@
|
|||
bugfixes:
|
||||
- AnsiballZ - Use ``importlib`` to load the module instead of ``imp`` on Python3+
|
|
@ -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
|
||||
|
||||
#
|
||||
|
|
|
@ -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):
|
||||
|
|
|
@ -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
|
||||
|
|
Loading…
Reference in a new issue