2021-04-26 12:23:21 +00:00
|
|
|
"""Generate mypy config."""
|
|
|
|
from __future__ import annotations
|
|
|
|
|
2022-11-14 05:37:47 +00:00
|
|
|
from collections.abc import Iterable
|
2021-04-26 12:23:21 +00:00
|
|
|
import configparser
|
|
|
|
import io
|
2021-04-29 04:29:53 +00:00
|
|
|
import os
|
|
|
|
from pathlib import Path
|
2021-04-26 12:23:21 +00:00
|
|
|
from typing import Final
|
|
|
|
|
2021-12-22 20:21:05 +00:00
|
|
|
from homeassistant.const import REQUIRED_PYTHON_VER
|
|
|
|
|
2021-04-26 12:23:21 +00:00
|
|
|
from .model import Config, Integration
|
|
|
|
|
|
|
|
# Modules which have type hints which known to be broken.
|
|
|
|
# If you are an author of component listed here, please fix these errors and
|
|
|
|
# remove your component from this list to enable type checks.
|
|
|
|
# Do your best to not add anything new here.
|
2022-09-05 18:12:37 +00:00
|
|
|
IGNORED_MODULES: Final[list[str]] = []
|
2021-04-26 12:23:21 +00:00
|
|
|
|
2022-01-26 09:55:06 +00:00
|
|
|
# Component modules which should set no_implicit_reexport = true.
|
|
|
|
NO_IMPLICIT_REEXPORT_MODULES: set[str] = {
|
|
|
|
"homeassistant.components",
|
2022-05-17 13:29:22 +00:00
|
|
|
"homeassistant.components.application_credentials.*",
|
2022-01-26 09:55:06 +00:00
|
|
|
"homeassistant.components.diagnostics.*",
|
2022-05-17 13:29:22 +00:00
|
|
|
"homeassistant.components.spotify.*",
|
2022-05-15 15:58:57 +00:00
|
|
|
"homeassistant.components.stream.*",
|
2022-05-17 13:29:22 +00:00
|
|
|
"homeassistant.components.update.*",
|
2022-01-26 09:55:06 +00:00
|
|
|
}
|
|
|
|
|
2021-04-26 12:23:21 +00:00
|
|
|
HEADER: Final = """
|
|
|
|
# Automatically generated by hassfest.
|
|
|
|
#
|
2022-01-27 04:52:09 +00:00
|
|
|
# To update, run python3 -m script.hassfest -p mypy_config
|
2021-04-26 12:23:21 +00:00
|
|
|
|
|
|
|
""".lstrip()
|
|
|
|
|
|
|
|
GENERAL_SETTINGS: Final[dict[str, str]] = {
|
2021-12-22 20:21:05 +00:00
|
|
|
"python_version": ".".join(str(x) for x in REQUIRED_PYTHON_VER[:2]),
|
2021-04-26 12:23:21 +00:00
|
|
|
"show_error_codes": "true",
|
|
|
|
"follow_imports": "silent",
|
2021-05-10 12:20:25 +00:00
|
|
|
# Enable some checks globally.
|
2021-04-26 12:23:21 +00:00
|
|
|
"ignore_missing_imports": "true",
|
2021-05-10 12:20:25 +00:00
|
|
|
"strict_equality": "true",
|
2022-08-13 17:33:57 +00:00
|
|
|
"no_implicit_optional": "true",
|
2021-04-26 12:23:21 +00:00
|
|
|
"warn_incomplete_stub": "true",
|
|
|
|
"warn_redundant_casts": "true",
|
|
|
|
"warn_unused_configs": "true",
|
2021-05-10 12:20:25 +00:00
|
|
|
"warn_unused_ignores": "true",
|
2022-11-08 13:41:39 +00:00
|
|
|
"enable_error_code": ", ".join(["ignore-without-code"]),
|
|
|
|
"disable_error_code": ", ".join(["annotation-unchecked"]),
|
2022-04-28 01:49:54 +00:00
|
|
|
# Strict_concatenate breaks passthrough ParamSpec typing
|
|
|
|
"strict_concatenate": "false",
|
2021-04-26 12:23:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
# This is basically the list of checks which is enabled for "strict=true".
|
2021-08-16 20:47:37 +00:00
|
|
|
# "strict=false" in config files does not turn strict settings off if they've been
|
|
|
|
# set in a more general section (it instead means as if strict was not specified at
|
|
|
|
# all), so we need to list all checks manually to be able to flip them wholesale.
|
2021-04-26 12:23:21 +00:00
|
|
|
STRICT_SETTINGS: Final[list[str]] = [
|
|
|
|
"check_untyped_defs",
|
|
|
|
"disallow_incomplete_defs",
|
|
|
|
"disallow_subclassing_any",
|
|
|
|
"disallow_untyped_calls",
|
|
|
|
"disallow_untyped_decorators",
|
|
|
|
"disallow_untyped_defs",
|
|
|
|
"warn_return_any",
|
|
|
|
"warn_unreachable",
|
|
|
|
# TODO: turn these on, address issues
|
|
|
|
# "disallow_any_generics",
|
|
|
|
# "no_implicit_reexport",
|
|
|
|
]
|
|
|
|
|
2022-01-10 11:07:22 +00:00
|
|
|
# Strict settings are already applied for core files.
|
|
|
|
# To enable granular typing, add additional settings if core files are given.
|
|
|
|
STRICT_SETTINGS_CORE: Final[list[str]] = [
|
|
|
|
"disallow_any_generics",
|
|
|
|
]
|
|
|
|
|
2021-04-26 12:23:21 +00:00
|
|
|
|
2022-02-03 13:21:06 +00:00
|
|
|
def _strict_module_in_ignore_list(
|
|
|
|
module: str, ignored_modules_set: set[str]
|
|
|
|
) -> str | None:
|
|
|
|
if module in ignored_modules_set:
|
|
|
|
return module
|
|
|
|
if module.endswith("*"):
|
|
|
|
module = module[:-1]
|
|
|
|
for ignored_module in ignored_modules_set:
|
|
|
|
if ignored_module.startswith(module):
|
|
|
|
return ignored_module
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
2022-11-14 05:37:47 +00:00
|
|
|
def _sort_within_sections(line_iter: Iterable[str]) -> Iterable[str]:
|
|
|
|
"""
|
|
|
|
Sort lines within sections.
|
|
|
|
|
|
|
|
Sections are defined as anything not delimited by a blank line
|
|
|
|
or an octothorpe-prefixed comment line.
|
|
|
|
"""
|
|
|
|
section: list[str] = []
|
|
|
|
for line in line_iter:
|
|
|
|
if line.startswith("#") or not line.strip():
|
|
|
|
yield from sorted(section)
|
|
|
|
section.clear()
|
|
|
|
yield line
|
|
|
|
continue
|
|
|
|
section.append(line)
|
|
|
|
yield from sorted(section)
|
|
|
|
|
|
|
|
|
|
|
|
def _get_strict_typing_path(config: Config) -> Path:
|
|
|
|
return config.root / ".strict-typing"
|
|
|
|
|
|
|
|
|
|
|
|
def _get_mypy_ini_path(config: Config) -> Path:
|
|
|
|
return config.root / "mypy.ini"
|
|
|
|
|
|
|
|
|
|
|
|
def _generate_and_validate_strict_typing(config: Config) -> str:
|
|
|
|
"""Validate and generate strict_typing."""
|
|
|
|
lines = [
|
|
|
|
line.strip()
|
|
|
|
for line in _get_strict_typing_path(config).read_text().splitlines()
|
|
|
|
]
|
|
|
|
return "\n".join(_sort_within_sections(lines)) + "\n"
|
2021-04-26 12:23:21 +00:00
|
|
|
|
|
|
|
|
2022-11-14 05:37:47 +00:00
|
|
|
def _generate_and_validate_mypy_config(config: Config) -> str:
|
|
|
|
"""Validate and generate mypy config."""
|
2021-04-26 12:23:21 +00:00
|
|
|
|
|
|
|
# Filter empty and commented lines.
|
2022-01-10 11:07:22 +00:00
|
|
|
parsed_modules: list[str] = [
|
2021-04-26 12:23:21 +00:00
|
|
|
line.strip()
|
2022-11-14 05:37:47 +00:00
|
|
|
for line in config.cache["strict_typing"].splitlines()
|
2021-04-26 12:23:21 +00:00
|
|
|
if line.strip() != "" and not line.startswith("#")
|
|
|
|
]
|
2021-04-27 16:13:11 +00:00
|
|
|
|
2022-01-10 11:07:22 +00:00
|
|
|
strict_modules: list[str] = []
|
|
|
|
strict_core_modules: list[str] = []
|
|
|
|
for module in parsed_modules:
|
|
|
|
if module.startswith("homeassistant.components"):
|
|
|
|
strict_modules.append(module)
|
|
|
|
else:
|
|
|
|
strict_core_modules.append(module)
|
|
|
|
|
2021-04-27 16:13:11 +00:00
|
|
|
ignored_modules_set: set[str] = set(IGNORED_MODULES)
|
|
|
|
for module in strict_modules:
|
|
|
|
if (
|
|
|
|
not module.startswith("homeassistant.components.")
|
|
|
|
and module != "homeassistant.components"
|
|
|
|
):
|
2021-04-26 12:23:21 +00:00
|
|
|
config.add_error(
|
|
|
|
"mypy_config", f"Only components should be added: {module}"
|
|
|
|
)
|
2022-02-03 13:21:06 +00:00
|
|
|
if ignored_module := _strict_module_in_ignore_list(module, ignored_modules_set):
|
2021-05-11 12:29:14 +00:00
|
|
|
config.add_error(
|
2022-02-03 13:21:06 +00:00
|
|
|
"mypy_config",
|
|
|
|
f"Module '{ignored_module}' is in ignored list in mypy_config.py",
|
2021-05-11 12:29:14 +00:00
|
|
|
)
|
2021-04-26 12:23:21 +00:00
|
|
|
|
2021-04-29 04:29:53 +00:00
|
|
|
# Validate that all modules exist.
|
2022-01-26 09:55:06 +00:00
|
|
|
all_modules = (
|
|
|
|
strict_modules
|
|
|
|
+ strict_core_modules
|
|
|
|
+ IGNORED_MODULES
|
|
|
|
+ list(NO_IMPLICIT_REEXPORT_MODULES)
|
|
|
|
)
|
2021-04-29 04:29:53 +00:00
|
|
|
for module in all_modules:
|
|
|
|
if module.endswith(".*"):
|
|
|
|
module_path = Path(module[:-2].replace(".", os.path.sep))
|
|
|
|
if not module_path.is_dir():
|
|
|
|
config.add_error("mypy_config", f"Module '{module} is not a folder")
|
|
|
|
else:
|
|
|
|
module = module.replace(".", os.path.sep)
|
|
|
|
module_path = Path(f"{module}.py")
|
|
|
|
if module_path.is_file():
|
|
|
|
continue
|
|
|
|
module_path = Path(module) / "__init__.py"
|
|
|
|
if not module_path.is_file():
|
|
|
|
config.add_error("mypy_config", f"Module '{module} doesn't exist")
|
|
|
|
|
2021-05-11 12:29:14 +00:00
|
|
|
# Don't generate mypy.ini if there're errors found because it will likely crash.
|
2021-05-16 14:04:09 +00:00
|
|
|
if any(err.plugin == "mypy_config" for err in config.errors):
|
2021-05-11 12:29:14 +00:00
|
|
|
return ""
|
|
|
|
|
2021-04-26 12:23:21 +00:00
|
|
|
mypy_config = configparser.ConfigParser()
|
|
|
|
|
|
|
|
general_section = "mypy"
|
|
|
|
mypy_config.add_section(general_section)
|
|
|
|
for key, value in GENERAL_SETTINGS.items():
|
|
|
|
mypy_config.set(general_section, key, value)
|
|
|
|
for key in STRICT_SETTINGS:
|
|
|
|
mypy_config.set(general_section, key, "true")
|
|
|
|
|
2022-01-26 09:55:06 +00:00
|
|
|
# By default enable no_implicit_reexport only for homeassistant.*
|
|
|
|
# Disable it afterwards for all components
|
|
|
|
components_section = "mypy-homeassistant.*"
|
|
|
|
mypy_config.add_section(components_section)
|
|
|
|
mypy_config.set(components_section, "no_implicit_reexport", "true")
|
|
|
|
|
2022-01-10 11:07:22 +00:00
|
|
|
for core_module in strict_core_modules:
|
|
|
|
core_section = f"mypy-{core_module}"
|
|
|
|
mypy_config.add_section(core_section)
|
|
|
|
for key in STRICT_SETTINGS_CORE:
|
|
|
|
mypy_config.set(core_section, key, "true")
|
|
|
|
|
2021-04-27 16:13:11 +00:00
|
|
|
# By default strict checks are disabled for components.
|
|
|
|
components_section = "mypy-homeassistant.components.*"
|
|
|
|
mypy_config.add_section(components_section)
|
|
|
|
for key in STRICT_SETTINGS:
|
|
|
|
mypy_config.set(components_section, key, "false")
|
2022-01-26 09:55:06 +00:00
|
|
|
mypy_config.set(components_section, "no_implicit_reexport", "false")
|
2021-04-27 16:13:11 +00:00
|
|
|
|
2021-05-03 16:45:38 +00:00
|
|
|
for strict_module in strict_modules:
|
|
|
|
strict_section = f"mypy-{strict_module}"
|
|
|
|
mypy_config.add_section(strict_section)
|
|
|
|
for key in STRICT_SETTINGS:
|
|
|
|
mypy_config.set(strict_section, key, "true")
|
2022-01-26 09:55:06 +00:00
|
|
|
if strict_module in NO_IMPLICIT_REEXPORT_MODULES:
|
|
|
|
mypy_config.set(strict_section, "no_implicit_reexport", "true")
|
|
|
|
|
2022-05-17 15:35:03 +00:00
|
|
|
for reexport_module in sorted(
|
|
|
|
NO_IMPLICIT_REEXPORT_MODULES.difference(strict_modules)
|
|
|
|
):
|
2022-01-26 09:55:06 +00:00
|
|
|
reexport_section = f"mypy-{reexport_module}"
|
|
|
|
mypy_config.add_section(reexport_section)
|
|
|
|
mypy_config.set(reexport_section, "no_implicit_reexport", "true")
|
2021-04-26 12:23:21 +00:00
|
|
|
|
2021-04-29 13:57:02 +00:00
|
|
|
# Disable strict checks for tests
|
|
|
|
tests_section = "mypy-tests.*"
|
|
|
|
mypy_config.add_section(tests_section)
|
|
|
|
for key in STRICT_SETTINGS:
|
|
|
|
mypy_config.set(tests_section, key, "false")
|
|
|
|
|
2021-05-03 16:45:38 +00:00
|
|
|
for ignored_module in IGNORED_MODULES:
|
|
|
|
ignored_section = f"mypy-{ignored_module}"
|
|
|
|
mypy_config.add_section(ignored_section)
|
|
|
|
mypy_config.set(ignored_section, "ignore_errors", "true")
|
2021-04-26 12:23:21 +00:00
|
|
|
|
|
|
|
with io.StringIO() as fp:
|
|
|
|
mypy_config.write(fp)
|
|
|
|
fp.seek(0)
|
2022-11-14 05:37:47 +00:00
|
|
|
return f"{HEADER}{fp.read().strip()}\n"
|
2021-04-26 12:23:21 +00:00
|
|
|
|
|
|
|
|
|
|
|
def validate(integrations: dict[str, Integration], config: Config) -> None:
|
2022-11-14 05:37:47 +00:00
|
|
|
"""Validate strict_typing and mypy config."""
|
|
|
|
strict_typing_content = _generate_and_validate_strict_typing(config)
|
|
|
|
config.cache["strict_typing"] = strict_typing_content
|
|
|
|
|
|
|
|
mypy_content = _generate_and_validate_mypy_config(config)
|
|
|
|
config.cache["mypy_config"] = mypy_content
|
2021-04-26 12:23:21 +00:00
|
|
|
|
2021-05-16 14:04:09 +00:00
|
|
|
if any(err.plugin == "mypy_config" for err in config.errors):
|
2021-05-11 12:29:14 +00:00
|
|
|
return
|
|
|
|
|
2022-11-14 05:37:47 +00:00
|
|
|
if _get_strict_typing_path(config).read_text() != strict_typing_content:
|
|
|
|
config.add_error(
|
|
|
|
"mypy_config",
|
|
|
|
"File .strict_typing is not up to date. Run python3 -m script.hassfest",
|
|
|
|
fixable=True,
|
|
|
|
)
|
|
|
|
|
|
|
|
if _get_mypy_ini_path(config).read_text() != mypy_content:
|
|
|
|
config.add_error(
|
|
|
|
"mypy_config",
|
|
|
|
"File mypy.ini is not up to date. Run python3 -m script.hassfest",
|
|
|
|
fixable=True,
|
|
|
|
)
|
2021-04-26 12:23:21 +00:00
|
|
|
|
|
|
|
|
|
|
|
def generate(integrations: dict[str, Integration], config: Config) -> None:
|
2022-11-14 05:37:47 +00:00
|
|
|
"""Generate strict_typing and mypy config."""
|
|
|
|
_get_mypy_ini_path(config).write_text(config.cache["mypy_config"])
|
|
|
|
_get_strict_typing_path(config).write_text(config.cache["strict_typing"])
|