Use unicode string for auto interpreter warnings (#53671)
* Use unicode string for auto interpreter warnings * Added some unit tests for interpreter selection * Fix python 3 syntax issues
This commit is contained in:
parent
38c742ccf3
commit
c8c326d88e
2 changed files with 116 additions and 29 deletions
|
@ -10,7 +10,7 @@ import pkgutil
|
||||||
import re
|
import re
|
||||||
|
|
||||||
from ansible import constants as C
|
from ansible import constants as C
|
||||||
from ansible.module_utils._text import to_text
|
from ansible.module_utils._text import to_native, to_text
|
||||||
from ansible.module_utils.distro import LinuxDistribution
|
from ansible.module_utils.distro import LinuxDistribution
|
||||||
from ansible.utils.display import Display
|
from ansible.utils.display import Display
|
||||||
from ansible.utils.plugin_docs import get_versioned_doclink
|
from ansible.utils.plugin_docs import get_versioned_doclink
|
||||||
|
@ -50,7 +50,7 @@ def discover_interpreter(action, interpreter_name, discovery_mode, task_vars):
|
||||||
host = task_vars.get('inventory_hostname', 'unknown')
|
host = task_vars.get('inventory_hostname', 'unknown')
|
||||||
res = None
|
res = None
|
||||||
platform_type = 'unknown'
|
platform_type = 'unknown'
|
||||||
found_interpreters = ['/usr/bin/python'] # fallback value
|
found_interpreters = [u'/usr/bin/python'] # fallback value
|
||||||
is_auto_legacy = discovery_mode.startswith('auto_legacy')
|
is_auto_legacy = discovery_mode.startswith('auto_legacy')
|
||||||
is_silent = discovery_mode.endswith('_silent')
|
is_silent = discovery_mode.endswith('_silent')
|
||||||
|
|
||||||
|
@ -58,7 +58,7 @@ def discover_interpreter(action, interpreter_name, discovery_mode, task_vars):
|
||||||
platform_python_map = C.config.get_config_value('INTERPRETER_PYTHON_DISTRO_MAP', variables=task_vars)
|
platform_python_map = C.config.get_config_value('INTERPRETER_PYTHON_DISTRO_MAP', variables=task_vars)
|
||||||
bootstrap_python_list = C.config.get_config_value('INTERPRETER_PYTHON_FALLBACK', variables=task_vars)
|
bootstrap_python_list = C.config.get_config_value('INTERPRETER_PYTHON_FALLBACK', variables=task_vars)
|
||||||
|
|
||||||
display.vvv(msg="Attempting {0} interpreter discovery".format(interpreter_name), host=host)
|
display.vvv(msg=u"Attempting {0} interpreter discovery".format(interpreter_name), host=host)
|
||||||
|
|
||||||
# not all command -v impls accept a list of commands, so we have to call it once per python
|
# not all command -v impls accept a list of commands, so we have to call it once per python
|
||||||
command_list = ["command -v '%s'" % py for py in bootstrap_python_list]
|
command_list = ["command -v '%s'" % py for py in bootstrap_python_list]
|
||||||
|
@ -67,27 +67,27 @@ def discover_interpreter(action, interpreter_name, discovery_mode, task_vars):
|
||||||
# FUTURE: in most cases we probably don't want to use become, but maybe sometimes we do?
|
# FUTURE: in most cases we probably don't want to use become, but maybe sometimes we do?
|
||||||
res = action._low_level_execute_command(shell_bootstrap, sudoable=False)
|
res = action._low_level_execute_command(shell_bootstrap, sudoable=False)
|
||||||
|
|
||||||
raw_stdout = res.get('stdout', '')
|
raw_stdout = res.get('stdout', u'')
|
||||||
|
|
||||||
match = foundre.match(raw_stdout)
|
match = foundre.match(raw_stdout)
|
||||||
|
|
||||||
if not match:
|
if not match:
|
||||||
display.debug('raw interpreter discovery output: {0}'.format(raw_stdout), host=host)
|
display.debug(u'raw interpreter discovery output: {0}'.format(raw_stdout), host=host)
|
||||||
raise ValueError('unexpected output from Python interpreter discovery')
|
raise ValueError('unexpected output from Python interpreter discovery')
|
||||||
|
|
||||||
platform_type = match.groups()[0].lower().strip()
|
platform_type = match.groups()[0].lower().strip()
|
||||||
|
|
||||||
found_interpreters = [interp.strip() for interp in match.groups()[1].splitlines() if interp.startswith('/')]
|
found_interpreters = [interp.strip() for interp in match.groups()[1].splitlines() if interp.startswith('/')]
|
||||||
|
|
||||||
display.debug("found interpreters: {0}".format(found_interpreters), host=host)
|
display.debug(u"found interpreters: {0}".format(found_interpreters), host=host)
|
||||||
|
|
||||||
if not found_interpreters:
|
if not found_interpreters:
|
||||||
action._discovery_warnings.append('No python interpreters found for host {0} (tried {1})'.format(host, bootstrap_python_list))
|
action._discovery_warnings.append(u'No python interpreters found for host {0} (tried {1})'.format(host, bootstrap_python_list))
|
||||||
# this is lame, but returning None or throwing an exception is uglier
|
# this is lame, but returning None or throwing an exception is uglier
|
||||||
return '/usr/bin/python'
|
return u'/usr/bin/python'
|
||||||
|
|
||||||
if platform_type != 'linux':
|
if platform_type != 'linux':
|
||||||
raise NotImplementedError('unsupported platform for extended discovery: {0}'.format(platform_type))
|
raise NotImplementedError('unsupported platform for extended discovery: {0}'.format(to_native(platform_type)))
|
||||||
|
|
||||||
platform_script = pkgutil.get_data('ansible.executor.discovery', 'python_target.py')
|
platform_script = pkgutil.get_data('ansible.executor.discovery', 'python_target.py')
|
||||||
|
|
||||||
|
@ -109,55 +109,55 @@ def discover_interpreter(action, interpreter_name, discovery_mode, task_vars):
|
||||||
if not version_map:
|
if not version_map:
|
||||||
raise NotImplementedError('unsupported Linux distribution: {0}'.format(distro))
|
raise NotImplementedError('unsupported Linux distribution: {0}'.format(distro))
|
||||||
|
|
||||||
platform_interpreter = _version_fuzzy_match(version, version_map)
|
platform_interpreter = to_text(_version_fuzzy_match(version, version_map), errors='surrogate_or_strict')
|
||||||
|
|
||||||
# provide a transition period for hosts that were using /usr/bin/python previously (but shouldn't have been)
|
# provide a transition period for hosts that were using /usr/bin/python previously (but shouldn't have been)
|
||||||
if is_auto_legacy:
|
if is_auto_legacy:
|
||||||
if platform_interpreter != '/usr/bin/python' and '/usr/bin/python' in found_interpreters:
|
if platform_interpreter != u'/usr/bin/python' and u'/usr/bin/python' in found_interpreters:
|
||||||
# FIXME: support comments in sivel's deprecation scanner so we can get reminded on this
|
# FIXME: support comments in sivel's deprecation scanner so we can get reminded on this
|
||||||
if not is_silent:
|
if not is_silent:
|
||||||
action._discovery_deprecation_warnings.append(dict(
|
action._discovery_deprecation_warnings.append(dict(
|
||||||
msg="Distribution {0} {1} on host {2} should use {3}, but is using "
|
msg=u"Distribution {0} {1} on host {2} should use {3}, but is using "
|
||||||
"/usr/bin/python for backward compatibility with prior Ansible releases. "
|
u"/usr/bin/python for backward compatibility with prior Ansible releases. "
|
||||||
"A future Ansible release will default to using the discovered platform "
|
u"A future Ansible release will default to using the discovered platform "
|
||||||
"python for this host. See {4} for more information"
|
u"python for this host. See {4} for more information"
|
||||||
.format(distro, version, host, platform_interpreter,
|
.format(distro, version, host, platform_interpreter,
|
||||||
get_versioned_doclink('reference_appendices/interpreter_discovery.html')),
|
get_versioned_doclink('reference_appendices/interpreter_discovery.html')),
|
||||||
version='2.12'))
|
version='2.12'))
|
||||||
return '/usr/bin/python'
|
return u'/usr/bin/python'
|
||||||
|
|
||||||
if platform_interpreter not in found_interpreters:
|
if platform_interpreter not in found_interpreters:
|
||||||
if platform_interpreter not in bootstrap_python_list:
|
if platform_interpreter not in bootstrap_python_list:
|
||||||
# sanity check to make sure we looked for it
|
# sanity check to make sure we looked for it
|
||||||
if not is_silent:
|
if not is_silent:
|
||||||
action._discovery_warnings \
|
action._discovery_warnings \
|
||||||
.append("Platform interpreter {0} on host {1} is missing from bootstrap list"
|
.append(u"Platform interpreter {0} on host {1} is missing from bootstrap list"
|
||||||
.format(platform_interpreter, host))
|
.format(platform_interpreter, host))
|
||||||
|
|
||||||
if not is_silent:
|
if not is_silent:
|
||||||
action._discovery_warnings \
|
action._discovery_warnings \
|
||||||
.append("Distribution {0} {1} on host {2} should use {3}, but is using {4}, since the "
|
.append(u"Distribution {0} {1} on host {2} should use {3}, but is using {4}, since the "
|
||||||
"discovered platform python interpreter was not present. See {5} "
|
u"discovered platform python interpreter was not present. See {5} "
|
||||||
"for more information."
|
u"for more information."
|
||||||
.format(distro, version, host, platform_interpreter, found_interpreters[0],
|
.format(distro, version, host, platform_interpreter, found_interpreters[0],
|
||||||
get_versioned_doclink('reference_appendices/interpreter_discovery.html')))
|
get_versioned_doclink('reference_appendices/interpreter_discovery.html')))
|
||||||
return found_interpreters[0]
|
return found_interpreters[0]
|
||||||
|
|
||||||
return platform_interpreter
|
return platform_interpreter
|
||||||
except NotImplementedError as ex:
|
except NotImplementedError as ex:
|
||||||
display.vvv(msg='Python interpreter discovery fallback ({0})'.format(to_text(ex)), host=host)
|
display.vvv(msg=u'Python interpreter discovery fallback ({0})'.format(to_text(ex)), host=host)
|
||||||
except Exception as ex:
|
except Exception as ex:
|
||||||
if not is_silent:
|
if not is_silent:
|
||||||
display.warning(msg='Unhandled error in Python interpreter discovery for host {0}: {1}'.format(host, to_text(ex)))
|
display.warning(msg=u'Unhandled error in Python interpreter discovery for host {0}: {1}'.format(host, to_text(ex)))
|
||||||
display.debug(msg='Interpreter discovery traceback:\n{0}'.format(to_text(format_exc())), host=host)
|
display.debug(msg=u'Interpreter discovery traceback:\n{0}'.format(to_text(format_exc())), host=host)
|
||||||
if res and res.get('stderr'):
|
if res and res.get('stderr'):
|
||||||
display.vvv(msg='Interpreter discovery remote stderr:\n{0}'.format(to_text(res.get('stderr'))), host=host)
|
display.vvv(msg=u'Interpreter discovery remote stderr:\n{0}'.format(to_text(res.get('stderr'))), host=host)
|
||||||
|
|
||||||
if not is_silent:
|
if not is_silent:
|
||||||
action._discovery_warnings \
|
action._discovery_warnings \
|
||||||
.append("Platform {0} on host {1} is using the discovered Python interpreter at {2}, but future installation of "
|
.append(u"Platform {0} on host {1} is using the discovered Python interpreter at {2}, but future installation of "
|
||||||
"another Python interpreter could change this. See {3} "
|
u"another Python interpreter could change this. See {3} "
|
||||||
"for more information."
|
u"for more information."
|
||||||
.format(platform_type, host, found_interpreters[0],
|
.format(platform_type, host, found_interpreters[0],
|
||||||
get_versioned_doclink('reference_appendices/interpreter_discovery.html')))
|
get_versioned_doclink('reference_appendices/interpreter_discovery.html')))
|
||||||
return found_interpreters[0]
|
return found_interpreters[0]
|
||||||
|
@ -172,11 +172,11 @@ def _get_linux_distro(platform_info):
|
||||||
osrelease_content = platform_info.get('osrelease_content')
|
osrelease_content = platform_info.get('osrelease_content')
|
||||||
|
|
||||||
if not osrelease_content:
|
if not osrelease_content:
|
||||||
return '', ''
|
return u'', u''
|
||||||
|
|
||||||
osr = LinuxDistribution._parse_os_release_content(osrelease_content)
|
osr = LinuxDistribution._parse_os_release_content(osrelease_content)
|
||||||
|
|
||||||
return osr.get('id', ''), osr.get('version_id', '')
|
return osr.get('id', u''), osr.get('version_id', u'')
|
||||||
|
|
||||||
|
|
||||||
def _version_fuzzy_match(version, version_map):
|
def _version_fuzzy_match(version, version_map):
|
||||||
|
|
87
test/units/executor/test_interpreter_discovery.py
Normal file
87
test/units/executor/test_interpreter_discovery.py
Normal file
|
@ -0,0 +1,87 @@
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# (c) 2019, Jordan Borean <jborean@redhat.com>
|
||||||
|
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
|
|
||||||
|
# Make coding more python3-ish
|
||||||
|
from __future__ import (absolute_import, division, print_function)
|
||||||
|
__metaclass__ = type
|
||||||
|
|
||||||
|
from units.compat.mock import MagicMock
|
||||||
|
|
||||||
|
from ansible.executor.interpreter_discovery import discover_interpreter
|
||||||
|
from ansible.module_utils._text import to_text
|
||||||
|
|
||||||
|
mock_ubuntu_platform_res = to_text(
|
||||||
|
r'{"osrelease_content": "NAME=\"Ubuntu\"\nVERSION=\"16.04.5 LTS (Xenial Xerus)\"\nID=ubuntu\nID_LIKE=debian\n'
|
||||||
|
r'PRETTY_NAME=\"Ubuntu 16.04.5 LTS\"\nVERSION_ID=\"16.04\"\nHOME_URL=\"http://www.ubuntu.com/\"\n'
|
||||||
|
r'SUPPORT_URL=\"http://help.ubuntu.com/\"\nBUG_REPORT_URL=\"http://bugs.launchpad.net/ubuntu/\"\n'
|
||||||
|
r'VERSION_CODENAME=xenial\nUBUNTU_CODENAME=xenial\n", "platform_dist_result": ["Ubuntu", "16.04", "xenial"]}'
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def test_discovery_interpreter_linux_auto_legacy():
|
||||||
|
res1 = u'PLATFORM\nLinux\nFOUND\n/usr/bin/python\n/usr/bin/python3.5\n/usr/bin/python3\nENDFOUND'
|
||||||
|
|
||||||
|
mock_action = MagicMock()
|
||||||
|
mock_action._low_level_execute_command.side_effect = [{'stdout': res1}, {'stdout': mock_ubuntu_platform_res}]
|
||||||
|
|
||||||
|
actual = discover_interpreter(mock_action, 'python', 'auto_legacy', {'inventory_hostname': u'host-fóöbär'})
|
||||||
|
|
||||||
|
assert actual == u'/usr/bin/python'
|
||||||
|
assert len(mock_action.method_calls) == 3
|
||||||
|
assert mock_action.method_calls[2][0] == '_discovery_deprecation_warnings.append'
|
||||||
|
assert u'Distribution Ubuntu 16.04 on host host-fóöbär should use /usr/bin/python3, but is using /usr/bin/python' \
|
||||||
|
u' for backward compatibility' in mock_action.method_calls[2][1][0]['msg']
|
||||||
|
assert mock_action.method_calls[2][1][0]['version'] == '2.12'
|
||||||
|
|
||||||
|
|
||||||
|
def test_discovery_interpreter_linux_auto_legacy_silent():
|
||||||
|
res1 = u'PLATFORM\nLinux\nFOUND\n/usr/bin/python\n/usr/bin/python3.5\n/usr/bin/python3\nENDFOUND'
|
||||||
|
|
||||||
|
mock_action = MagicMock()
|
||||||
|
mock_action._low_level_execute_command.side_effect = [{'stdout': res1}, {'stdout': mock_ubuntu_platform_res}]
|
||||||
|
|
||||||
|
actual = discover_interpreter(mock_action, 'python', 'auto_legacy_silent', {'inventory_hostname': u'host-fóöbär'})
|
||||||
|
|
||||||
|
assert actual == u'/usr/bin/python'
|
||||||
|
assert len(mock_action.method_calls) == 2
|
||||||
|
|
||||||
|
|
||||||
|
def test_discovery_interpreter_linux_auto():
|
||||||
|
res1 = u'PLATFORM\nLinux\nFOUND\n/usr/bin/python\n/usr/bin/python3.5\n/usr/bin/python3\nENDFOUND'
|
||||||
|
|
||||||
|
mock_action = MagicMock()
|
||||||
|
mock_action._low_level_execute_command.side_effect = [{'stdout': res1}, {'stdout': mock_ubuntu_platform_res}]
|
||||||
|
|
||||||
|
actual = discover_interpreter(mock_action, 'python', 'auto', {'inventory_hostname': u'host-fóöbär'})
|
||||||
|
|
||||||
|
assert actual == u'/usr/bin/python3'
|
||||||
|
assert len(mock_action.method_calls) == 2
|
||||||
|
|
||||||
|
|
||||||
|
def test_discovery_interpreter_non_linux():
|
||||||
|
mock_action = MagicMock()
|
||||||
|
mock_action._low_level_execute_command.return_value = \
|
||||||
|
{'stdout': u'PLATFORM\nDarwin\nFOUND\n/usr/bin/python\nENDFOUND'}
|
||||||
|
|
||||||
|
actual = discover_interpreter(mock_action, 'python', 'auto_legacy', {'inventory_hostname': u'host-fóöbär'})
|
||||||
|
|
||||||
|
assert actual == u'/usr/bin/python'
|
||||||
|
assert len(mock_action.method_calls) == 2
|
||||||
|
assert mock_action.method_calls[1][0] == '_discovery_warnings.append'
|
||||||
|
assert u'Platform darwin on host host-fóöbär is using the discovered Python interpreter at /usr/bin/python, ' \
|
||||||
|
u'but future installation of another Python interpreter could change this' \
|
||||||
|
in mock_action.method_calls[1][1][0]
|
||||||
|
|
||||||
|
|
||||||
|
def test_no_interpreters_found():
|
||||||
|
mock_action = MagicMock()
|
||||||
|
mock_action._low_level_execute_command.return_value = {'stdout': u'PLATFORM\nWindows\nFOUND\nENDFOUND'}
|
||||||
|
|
||||||
|
actual = discover_interpreter(mock_action, 'python', 'auto_legacy', {'inventory_hostname': u'host-fóöbär'})
|
||||||
|
|
||||||
|
assert actual == u'/usr/bin/python'
|
||||||
|
assert len(mock_action.method_calls) == 2
|
||||||
|
assert mock_action.method_calls[1][0] == '_discovery_warnings.append'
|
||||||
|
assert u'No python interpreters found for host host-fóöbär (tried' \
|
||||||
|
in mock_action.method_calls[1][1][0]
|
Loading…
Reference in a new issue