mirror of https://github.com/ansible/ansible.git
Initial mypy sanity test support for core.
parent
27923aad7e
commit
3d5637beec
@ -0,0 +1,14 @@
|
||||
mypy
|
||||
====
|
||||
|
||||
The ``mypy`` static type checker is used to check the following code against each Python version supported by the controller:
|
||||
|
||||
* ``lib/ansible/``
|
||||
* ``test/lib/ansible_test/_internal/``
|
||||
|
||||
Additionally, the following code is checked against Python versions supported only on managed nodes:
|
||||
|
||||
* ``lib/ansible/modules/``
|
||||
* ``lib/ansible/module_utils/``
|
||||
|
||||
See https://mypy.readthedocs.io/en/stable/ for additional details.
|
@ -0,0 +1,9 @@
|
||||
mypy[python2]
|
||||
packaging # type stubs not published separately
|
||||
types-backports
|
||||
types-jinja2
|
||||
types-paramiko
|
||||
types-pyyaml < 6 # PyYAML 6+ stubs do not support Python 2.7
|
||||
types-requests
|
||||
types-setuptools
|
||||
types-toml
|
@ -0,0 +1,20 @@
|
||||
# edit "sanity.mypy.in" and generate with: hacking/update-sanity-requirements.py --test mypy
|
||||
mypy==0.931
|
||||
mypy-extensions==0.4.3
|
||||
packaging==21.2
|
||||
pyparsing==2.4.7
|
||||
tomli==2.0.1
|
||||
typed-ast==1.5.2
|
||||
types-backports==0.1.3
|
||||
types-cryptography==3.3.15
|
||||
types-enum34==1.1.8
|
||||
types-ipaddress==1.0.8
|
||||
types-Jinja2==2.11.9
|
||||
types-MarkupSafe==1.1.10
|
||||
types-paramiko==2.8.13
|
||||
types-PyYAML==5.4.12
|
||||
types-requests==2.27.10
|
||||
types-setuptools==57.4.9
|
||||
types-toml==0.10.4
|
||||
types-urllib3==1.26.9
|
||||
typing-extensions==3.10.0.2
|
@ -0,0 +1,247 @@
|
||||
"""Sanity test which executes mypy."""
|
||||
from __future__ import annotations
|
||||
|
||||
import dataclasses
|
||||
import os
|
||||
import re
|
||||
import typing as t
|
||||
|
||||
from . import (
|
||||
SanityMultipleVersion,
|
||||
SanityMessage,
|
||||
SanityFailure,
|
||||
SanitySuccess,
|
||||
SanitySkipped,
|
||||
SanityTargets,
|
||||
create_sanity_virtualenv,
|
||||
)
|
||||
|
||||
from ...constants import (
|
||||
CONTROLLER_PYTHON_VERSIONS,
|
||||
REMOTE_ONLY_PYTHON_VERSIONS,
|
||||
)
|
||||
|
||||
from ...test import (
|
||||
TestResult,
|
||||
)
|
||||
|
||||
from ...target import (
|
||||
TestTarget,
|
||||
)
|
||||
|
||||
from ...util import (
|
||||
SubprocessError,
|
||||
display,
|
||||
parse_to_list_of_dict,
|
||||
ANSIBLE_TEST_CONTROLLER_ROOT,
|
||||
ApplicationError,
|
||||
is_subdir,
|
||||
)
|
||||
|
||||
from ...util_common import (
|
||||
intercept_python,
|
||||
)
|
||||
|
||||
from ...ansible_util import (
|
||||
ansible_environment,
|
||||
)
|
||||
|
||||
from ...config import (
|
||||
SanityConfig,
|
||||
)
|
||||
|
||||
from ...host_configs import (
|
||||
PythonConfig,
|
||||
VirtualPythonConfig,
|
||||
)
|
||||
|
||||
|
||||
class MypyTest(SanityMultipleVersion):
|
||||
"""Sanity test which executes mypy."""
|
||||
ansible_only = True
|
||||
|
||||
def filter_targets(self, targets): # type: (t.List[TestTarget]) -> t.List[TestTarget]
|
||||
"""Return the given list of test targets, filtered to include only those relevant for the test."""
|
||||
return [target for target in targets if os.path.splitext(target.path)[1] == '.py' and (
|
||||
target.path.startswith('lib/ansible/') or target.path.startswith('test/lib/ansible_test/_internal/'))]
|
||||
|
||||
@property
|
||||
def error_code(self): # type: () -> t.Optional[str]
|
||||
"""Error code for ansible-test matching the format used by the underlying test program, or None if the program does not use error codes."""
|
||||
return 'ansible-test'
|
||||
|
||||
@property
|
||||
def needs_pypi(self): # type: () -> bool
|
||||
"""True if the test requires PyPI, otherwise False."""
|
||||
return True
|
||||
|
||||
def test(self, args, targets, python): # type: (SanityConfig, SanityTargets, PythonConfig) -> TestResult
|
||||
settings = self.load_processor(args, python.version)
|
||||
|
||||
paths = [target.path for target in targets.include]
|
||||
|
||||
virtualenv_python = create_sanity_virtualenv(args, args.controller_python, self.name)
|
||||
|
||||
if args.prime_venvs:
|
||||
return SanitySkipped(self.name, python_version=python.version)
|
||||
|
||||
if not virtualenv_python:
|
||||
display.warning(f'Skipping sanity test "{self.name}" due to missing virtual environment support on Python {args.controller_python.version}.')
|
||||
return SanitySkipped(self.name, python.version)
|
||||
|
||||
contexts = (
|
||||
MyPyContext('ansible-test', ['test/lib/ansible_test/_internal/'], CONTROLLER_PYTHON_VERSIONS),
|
||||
MyPyContext('ansible-core', ['lib/ansible/'], CONTROLLER_PYTHON_VERSIONS),
|
||||
MyPyContext('modules', ['lib/ansible/modules/', 'lib/ansible/module_utils/'], REMOTE_ONLY_PYTHON_VERSIONS),
|
||||
)
|
||||
|
||||
unfiltered_messages = [] # type: t.List[SanityMessage]
|
||||
|
||||
for context in contexts:
|
||||
if python.version not in context.python_versions:
|
||||
continue
|
||||
|
||||
unfiltered_messages.extend(self.test_context(args, virtualenv_python, python, context, paths))
|
||||
|
||||
notices = []
|
||||
messages = []
|
||||
|
||||
for message in unfiltered_messages:
|
||||
if message.level != 'error':
|
||||
notices.append(message)
|
||||
continue
|
||||
|
||||
match = re.search(r'^(?P<message>.*) {2}\[(?P<code>.*)]$', message.message)
|
||||
|
||||
messages.append(SanityMessage(
|
||||
message=match.group('message'),
|
||||
path=message.path,
|
||||
line=message.line,
|
||||
column=message.column,
|
||||
level=message.level,
|
||||
code=match.group('code'),
|
||||
))
|
||||
|
||||
for notice in notices:
|
||||
display.info(notice.format(), verbosity=3)
|
||||
|
||||
# The following error codes from mypy indicate that results are incomplete.
|
||||
# That prevents the test from completing successfully, just as if mypy were to traceback or generate unexpected output.
|
||||
fatal_error_codes = {
|
||||
'import',
|
||||
'syntax',
|
||||
}
|
||||
|
||||
fatal_errors = [message for message in messages if message.code in fatal_error_codes]
|
||||
|
||||
if fatal_errors:
|
||||
error_message = '\n'.join(error.format() for error in fatal_errors)
|
||||
raise ApplicationError(f'Encountered {len(fatal_errors)} fatal errors reported by mypy:\n{error_message}')
|
||||
|
||||
paths_set = set(paths)
|
||||
|
||||
# Only report messages for paths that were specified as targets.
|
||||
# Imports in our code are followed by mypy in order to perform its analysis, which is important for accurate results.
|
||||
# However, it will also report issues on those files, which is not the desired behavior.
|
||||
messages = [message for message in messages if message.path in paths_set]
|
||||
|
||||
results = settings.process_errors(messages, paths)
|
||||
|
||||
if results:
|
||||
return SanityFailure(self.name, messages=results, python_version=python.version)
|
||||
|
||||
return SanitySuccess(self.name, python_version=python.version)
|
||||
|
||||
@staticmethod
|
||||
def test_context(
|
||||
args, # type: SanityConfig
|
||||
virtualenv_python, # type: VirtualPythonConfig
|
||||
python, # type: PythonConfig
|
||||
context, # type: MyPyContext
|
||||
paths, # type: t.List[str]
|
||||
): # type: (...) -> t.List[SanityMessage]
|
||||
"""Run mypy tests for the specified context."""
|
||||
context_paths = [path for path in paths if any(is_subdir(path, match_path) for match_path in context.paths)]
|
||||
|
||||
if not context_paths:
|
||||
return []
|
||||
|
||||
config_path = os.path.join(ANSIBLE_TEST_CONTROLLER_ROOT, 'sanity', 'mypy', f'{context.name}.ini')
|
||||
|
||||
display.info(f'Checking context "{context.name}"', verbosity=1)
|
||||
|
||||
env = ansible_environment(args, color=False)
|
||||
|
||||
# The --no-site-packages option should not be used, as it will prevent loading of type stubs from the sanity test virtual environment.
|
||||
|
||||
# Enabling the --warn-unused-configs option would help keep the config files clean.
|
||||
# However, the option can only be used when all files in tested contexts are evaluated.
|
||||
# Unfortunately sanity tests have no way of making that determination currently.
|
||||
# The option is also incompatible with incremental mode and caching.
|
||||
|
||||
cmd = [
|
||||
# Below are arguments common to all contexts.
|
||||
# They are kept here to avoid repetition in each config file.
|
||||
virtualenv_python.path,
|
||||
'-m', 'mypy',
|
||||
'--show-column-numbers',
|
||||
'--show-error-codes',
|
||||
'--no-error-summary',
|
||||
# This is a fairly common pattern in our code, so we'll allow it.
|
||||
'--allow-redefinition',
|
||||
# Since we specify the path(s) to test, it's important that mypy is configured to use the default behavior of following imports.
|
||||
'--follow-imports', 'normal',
|
||||
# Incremental results and caching do not provide significant performance benefits.
|
||||
# It also prevents the use of the --warn-unused-configs option.
|
||||
'--no-incremental',
|
||||
'--cache-dir', '/dev/null',
|
||||
# The platform is specified here so that results are consistent regardless of what platform the tests are run from.
|
||||
# In the future, if testing of other platforms is desired, the platform should become part of the test specification, just like the Python version.
|
||||
'--platform', 'linux',
|
||||
# Despite what the documentation [1] states, the --python-version option does not cause mypy to search for a corresponding Python executable.
|
||||
# It will instead use the Python executable that is used to run mypy itself.
|
||||
# The --python-executable option can be used to specify the Python executable, with the default being the executable used to run mypy.
|
||||
# As a precaution, that option is used in case the behavior of mypy is updated in the future to match the documentation.
|
||||
# That should help guarantee that the Python executable providing type hints is the one used to run mypy.
|
||||
# [1] https://mypy.readthedocs.io/en/stable/command_line.html#cmdoption-mypy-python-version
|
||||
'--python-executable', virtualenv_python.path,
|
||||
'--python-version', python.version,
|
||||
# Below are context specific arguments.
|
||||
# They are primarily useful for listing individual 'ignore_missing_imports' entries instead of using a global ignore.
|
||||
'--config-file', config_path,
|
||||
]
|
||||
|
||||
cmd.extend(context_paths)
|
||||
|
||||
try:
|
||||
stdout, stderr = intercept_python(args, virtualenv_python, cmd, env, capture=True)
|
||||
|
||||
if stdout or stderr:
|
||||
raise SubprocessError(cmd, stdout=stdout, stderr=stderr)
|
||||
except SubprocessError as ex:
|
||||
if ex.status != 1 or ex.stderr or not ex.stdout:
|
||||
raise
|
||||
|
||||
stdout = ex.stdout
|
||||
|
||||
pattern = r'^(?P<path>[^:]*):(?P<line>[0-9]+):((?P<column>[0-9]+):)? (?P<level>[^:]+): (?P<message>.*)$'
|
||||
|
||||
parsed = parse_to_list_of_dict(pattern, stdout)
|
||||
|
||||
messages = [SanityMessage(
|
||||
level=r['level'],
|
||||
message=r['message'],
|
||||
path=r['path'],
|
||||
line=int(r['line']),
|
||||
column=int(r.get('column') or '0'),
|
||||
) for r in parsed]
|
||||
|
||||
return messages
|
||||
|
||||
|
||||
@dataclasses.dataclass(frozen=True)
|
||||
class MyPyContext:
|
||||
"""Context details for a single run of mypy."""
|
||||
name: str
|
||||
paths: t.List[str]
|
||||
python_versions: t.Tuple[str, ...]
|
@ -0,0 +1,119 @@
|
||||
# IMPORTANT
|
||||
# Set "ignore_missing_imports" per package below, rather than globally.
|
||||
# That will help identify missing type stubs that should be added to the sanity test environment.
|
||||
|
||||
[mypy]
|
||||
# There are ~20 errors reported in ansible-core when strict optional checking is enabled.
|
||||
# Until the number of occurrences are reduced, it's better to disable strict checking.
|
||||
strict_optional = False
|
||||
# There are ~70 errors reported in ansible-core when checking attributes.
|
||||
# Until the number of occurrences are reduced, it's better to disable the check.
|
||||
disable_error_code = attr-defined
|
||||
|
||||
[mypy-ansible.module_utils.six.moves.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-passlib.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-pexpect.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-pypsrp.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-winrm.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-kerberos.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-xmltodict.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-md5.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-scp.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-ncclient.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-lxml.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-yum.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-rpmUtils.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-rpm.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-psutil.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-dnf.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-apt.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-apt_pkg.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-gssapi.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-_ssl.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-urllib_gssapi.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-systemd.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-sha.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-distro.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-selectors2.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-resolvelib.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-urlparse.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-argcomplete.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-selinux.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-urllib2.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-httplib.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-compiler.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-aptsources.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-urllib3.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-requests.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-jinja2.nativetypes]
|
||||
ignore_missing_imports = True
|
@ -0,0 +1,21 @@
|
||||
# IMPORTANT
|
||||
# Set "ignore_missing_imports" per package below, rather than globally.
|
||||
# That will help identify missing type stubs that should be added to the sanity test environment.
|
||||
|
||||
[mypy]
|
||||
# There are ~350 errors reported in ansible-test when strict optional checking is enabled.
|
||||
# Until the number of occurrences are greatly reduced, it's better to disable strict checking.
|
||||
strict_optional = False
|
||||
# There are ~25 errors reported in ansible-test under the 'misc' code.
|
||||
# The majority of those errors are "Only concrete class can be given", which is due to a limitation of mypy.
|
||||
# See: https://github.com/python/mypy/issues/5374
|
||||
disable_error_code = misc
|
||||
|
||||
[mypy-argcomplete]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-coverage]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-ansible_release]
|
||||
ignore_missing_imports = True
|
@ -0,0 +1,98 @@
|
||||
# IMPORTANT
|
||||
# Set "ignore_missing_imports" per package below, rather than globally.
|
||||
# That will help identify missing type stubs that should be added to the sanity test environment.
|
||||
|
||||
[mypy]
|
||||
|
||||
[mypy-ansible.module_utils.six.moves.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-pexpect.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-md5.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-yum.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-rpmUtils.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-rpm.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-psutil.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-dnf.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-apt.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-apt_pkg.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-gssapi.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-_ssl.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-urllib_gssapi.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-systemd.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-sha.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-distro.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-selectors2.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-selinux.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-urllib2.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-httplib.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-compiler.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-aptsources.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-urllib3.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-requests.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-pkg_resources.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-urllib.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-email.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-selectors.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-importlib.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-collections.*]
|
||||
ignore_missing_imports = True
|
||||
|
||||
[mypy-http.*]
|
||||
ignore_missing_imports = True
|
Loading…
Reference in New Issue