2018-03-09 03:34:24 +00:00
|
|
|
"""Script to check the configuration file."""
|
2021-03-17 20:46:07 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2016-08-23 04:42:05 +00:00
|
|
|
import argparse
|
2020-07-06 22:58:53 +00:00
|
|
|
import asyncio
|
2019-07-10 18:56:50 +00:00
|
|
|
from collections import OrderedDict
|
2021-09-29 14:32:11 +00:00
|
|
|
from collections.abc import Callable, Mapping, Sequence
|
2016-08-23 04:42:05 +00:00
|
|
|
from glob import glob
|
2019-12-09 15:42:10 +00:00
|
|
|
import logging
|
|
|
|
import os
|
2021-09-29 14:32:11 +00:00
|
|
|
from typing import Any
|
2016-09-21 04:26:40 +00:00
|
|
|
from unittest.mock import patch
|
|
|
|
|
2021-03-02 20:58:53 +00:00
|
|
|
from homeassistant import core
|
2019-07-10 18:56:50 +00:00
|
|
|
from homeassistant.config import get_default_config_dir
|
2019-12-09 15:42:10 +00:00
|
|
|
from homeassistant.exceptions import HomeAssistantError
|
2021-09-03 17:15:57 +00:00
|
|
|
from homeassistant.helpers import area_registry, device_registry, entity_registry
|
2019-07-10 18:56:50 +00:00
|
|
|
from homeassistant.helpers.check_config import async_check_ha_config_file
|
2021-03-02 20:58:53 +00:00
|
|
|
from homeassistant.util.yaml import Secrets
|
2019-05-09 16:07:56 +00:00
|
|
|
import homeassistant.util.yaml.loader as yaml_loader
|
2019-07-24 20:18:40 +00:00
|
|
|
|
|
|
|
# mypy: allow-untyped-calls, allow-untyped-defs
|
|
|
|
|
2021-09-23 14:12:13 +00:00
|
|
|
REQUIREMENTS = ("colorlog==6.4.1",)
|
2016-08-23 04:42:05 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
# pylint: disable=protected-access
|
2021-03-17 20:46:07 +00:00
|
|
|
MOCKS: dict[str, tuple[str, Callable]] = {
|
2019-07-31 19:25:30 +00:00
|
|
|
"load": ("homeassistant.util.yaml.loader.load_yaml", yaml_loader.load_yaml),
|
|
|
|
"load*": ("homeassistant.config.load_yaml", yaml_loader.load_yaml),
|
|
|
|
"secrets": ("homeassistant.util.yaml.loader.secret_yaml", yaml_loader.secret_yaml),
|
2019-09-04 03:36:04 +00:00
|
|
|
}
|
2018-03-09 03:34:24 +00:00
|
|
|
|
2021-03-17 20:46:07 +00:00
|
|
|
PATCHES: dict[str, Any] = {}
|
2016-08-23 04:42:05 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
C_HEAD = "bold"
|
|
|
|
ERROR_STR = "General Errors"
|
2016-08-23 04:42:05 +00:00
|
|
|
|
|
|
|
|
|
|
|
def color(the_color, *args, reset=None):
|
|
|
|
"""Color helper."""
|
2020-04-04 15:07:36 +00:00
|
|
|
# pylint: disable=import-outside-toplevel
|
2019-03-08 00:48:14 +00:00
|
|
|
from colorlog.escape_codes import escape_codes, parse_colors
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2016-08-23 04:42:05 +00:00
|
|
|
try:
|
2019-03-08 00:48:14 +00:00
|
|
|
if not args:
|
|
|
|
assert reset is None, "You cannot reset if nothing being printed"
|
|
|
|
return parse_colors(the_color)
|
2019-07-31 19:25:30 +00:00
|
|
|
return parse_colors(the_color) + " ".join(args) + escape_codes[reset or "reset"]
|
2019-03-08 00:48:14 +00:00
|
|
|
except KeyError as k:
|
2020-08-28 11:50:32 +00:00
|
|
|
raise ValueError(f"Invalid color {k!s} in {the_color}") from k
|
2016-08-23 04:42:05 +00:00
|
|
|
|
|
|
|
|
2021-03-17 20:46:07 +00:00
|
|
|
def run(script_args: list) -> int:
|
2019-07-24 20:18:40 +00:00
|
|
|
"""Handle check config commandline script."""
|
2019-07-31 19:25:30 +00:00
|
|
|
parser = argparse.ArgumentParser(description="Check Home Assistant configuration.")
|
|
|
|
parser.add_argument("--script", choices=["check_config"])
|
2016-08-23 04:42:05 +00:00
|
|
|
parser.add_argument(
|
2019-07-31 19:25:30 +00:00
|
|
|
"-c",
|
|
|
|
"--config",
|
2018-03-09 03:34:24 +00:00
|
|
|
default=get_default_config_dir(),
|
2019-07-31 19:25:30 +00:00
|
|
|
help="Directory that contains the Home Assistant configuration",
|
|
|
|
)
|
2016-08-23 04:42:05 +00:00
|
|
|
parser.add_argument(
|
2019-07-31 19:25:30 +00:00
|
|
|
"-i",
|
|
|
|
"--info",
|
|
|
|
nargs="?",
|
|
|
|
default=None,
|
|
|
|
const="all",
|
|
|
|
help="Show a portion of the config",
|
|
|
|
)
|
2016-08-23 04:42:05 +00:00
|
|
|
parser.add_argument(
|
2019-07-31 19:25:30 +00:00
|
|
|
"-f", "--files", action="store_true", help="Show used configuration files"
|
|
|
|
)
|
2016-08-23 04:42:05 +00:00
|
|
|
parser.add_argument(
|
2019-07-31 19:25:30 +00:00
|
|
|
"-s", "--secrets", action="store_true", help="Show secret information"
|
|
|
|
)
|
2016-08-23 04:42:05 +00:00
|
|
|
|
2018-03-09 03:34:24 +00:00
|
|
|
args, unknown = parser.parse_known_args()
|
|
|
|
if unknown:
|
2019-07-31 19:25:30 +00:00
|
|
|
print(color("red", "Unknown arguments:", ", ".join(unknown)))
|
2016-08-23 04:42:05 +00:00
|
|
|
|
|
|
|
config_dir = os.path.join(os.getcwd(), args.config)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
print(color("bold", "Testing configuration at", config_dir))
|
2016-08-23 04:42:05 +00:00
|
|
|
|
2018-03-09 03:34:24 +00:00
|
|
|
res = check(config_dir, args.secrets)
|
|
|
|
|
2021-03-17 20:46:07 +00:00
|
|
|
domain_info: list[str] = []
|
2016-08-23 04:42:05 +00:00
|
|
|
if args.info:
|
2019-07-31 19:25:30 +00:00
|
|
|
domain_info = args.info.split(",")
|
2016-08-23 04:42:05 +00:00
|
|
|
|
|
|
|
if args.files:
|
2019-07-31 19:25:30 +00:00
|
|
|
print(color(C_HEAD, "yaml files"), "(used /", color("red", "not used") + ")")
|
|
|
|
deps = os.path.join(config_dir, "deps")
|
|
|
|
yaml_files = [
|
|
|
|
f
|
|
|
|
for f in glob(os.path.join(config_dir, "**/*.yaml"), recursive=True)
|
|
|
|
if not f.startswith(deps)
|
|
|
|
]
|
2018-03-15 11:10:54 +00:00
|
|
|
|
|
|
|
for yfn in sorted(yaml_files):
|
2019-07-31 19:25:30 +00:00
|
|
|
the_color = "" if yfn in res["yaml_files"] else "red"
|
|
|
|
print(color(the_color, "-", yfn))
|
2016-08-23 04:42:05 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
if res["except"]:
|
|
|
|
print(color("bold_white", "Failed config"))
|
|
|
|
for domain, config in res["except"].items():
|
2016-08-23 04:42:05 +00:00
|
|
|
domain_info.append(domain)
|
2019-07-31 19:25:30 +00:00
|
|
|
print(" ", color("bold_red", domain + ":"), color("red", "", reset="red"))
|
|
|
|
dump_dict(config, reset="red")
|
|
|
|
print(color("reset"))
|
2016-08-23 04:42:05 +00:00
|
|
|
|
|
|
|
if domain_info:
|
2019-07-31 19:25:30 +00:00
|
|
|
if "all" in domain_info:
|
|
|
|
print(color("bold_white", "Successful config (all)"))
|
|
|
|
for domain, config in res["components"].items():
|
|
|
|
print(" ", color(C_HEAD, domain + ":"))
|
2016-09-23 07:10:19 +00:00
|
|
|
dump_dict(config)
|
2016-08-23 04:42:05 +00:00
|
|
|
else:
|
2019-07-31 19:25:30 +00:00
|
|
|
print(color("bold_white", "Successful config (partial)"))
|
2016-08-23 04:42:05 +00:00
|
|
|
for domain in domain_info:
|
|
|
|
if domain == ERROR_STR:
|
|
|
|
continue
|
2019-07-31 19:25:30 +00:00
|
|
|
print(" ", color(C_HEAD, domain + ":"))
|
2020-04-07 19:06:05 +00:00
|
|
|
dump_dict(res["components"].get(domain))
|
2016-08-23 04:42:05 +00:00
|
|
|
|
|
|
|
if args.secrets:
|
2021-03-17 20:46:07 +00:00
|
|
|
flatsecret: dict[str, str] = {}
|
2016-08-23 04:42:05 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
for sfn, sdict in res["secret_cache"].items():
|
2016-08-23 04:42:05 +00:00
|
|
|
sss = []
|
2018-01-30 22:44:05 +00:00
|
|
|
for skey in sdict:
|
2016-08-23 04:42:05 +00:00
|
|
|
if skey in flatsecret:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.error(
|
|
|
|
"Duplicated secrets in files %s and %s", flatsecret[skey], sfn
|
|
|
|
)
|
2016-08-23 04:42:05 +00:00
|
|
|
flatsecret[skey] = sfn
|
2019-07-31 19:25:30 +00:00
|
|
|
sss.append(color("green", skey) if skey in res["secrets"] else skey)
|
|
|
|
print(color(C_HEAD, "Secrets from", sfn + ":"), ", ".join(sss))
|
2016-08-23 04:42:05 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
print(color(C_HEAD, "Used Secrets:"))
|
|
|
|
for skey, sval in res["secrets"].items():
|
2018-03-11 10:51:03 +00:00
|
|
|
if sval is None:
|
2019-07-31 19:25:30 +00:00
|
|
|
print(" -", skey + ":", color("red", "not found"))
|
2018-03-11 10:51:03 +00:00
|
|
|
continue
|
2021-02-25 08:48:19 +00:00
|
|
|
print(" -", skey + ":", sval)
|
2016-08-23 04:42:05 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
return len(res["except"])
|
2016-08-23 04:42:05 +00:00
|
|
|
|
|
|
|
|
2018-03-09 03:34:24 +00:00
|
|
|
def check(config_dir, secrets=False):
|
2016-08-23 04:42:05 +00:00
|
|
|
"""Perform a check by mocking hass load functions."""
|
2019-07-31 19:25:30 +00:00
|
|
|
logging.getLogger("homeassistant.loader").setLevel(logging.CRITICAL)
|
2021-03-17 20:46:07 +00:00
|
|
|
res: dict[str, Any] = {
|
2019-07-31 19:25:30 +00:00
|
|
|
"yaml_files": OrderedDict(), # yaml_files loaded
|
|
|
|
"secrets": OrderedDict(), # secret cache and secrets loaded
|
|
|
|
"except": OrderedDict(), # exceptions raised (with config)
|
2019-07-24 20:18:40 +00:00
|
|
|
#'components' is a HomeAssistantConfig # noqa: E265
|
2021-03-02 20:58:53 +00:00
|
|
|
"secret_cache": {},
|
2019-09-04 03:36:04 +00:00
|
|
|
}
|
2016-08-23 04:42:05 +00:00
|
|
|
|
2018-07-26 06:55:42 +00:00
|
|
|
# pylint: disable=possibly-unused-variable
|
2021-03-02 20:58:53 +00:00
|
|
|
def mock_load(filename, secrets=None):
|
2018-03-09 03:34:24 +00:00
|
|
|
"""Mock hass.util.load_yaml to save config file names."""
|
2019-07-31 19:25:30 +00:00
|
|
|
res["yaml_files"][filename] = True
|
2021-03-02 20:58:53 +00:00
|
|
|
return MOCKS["load"][1](filename, secrets)
|
2016-08-23 04:42:05 +00:00
|
|
|
|
2018-07-26 06:55:42 +00:00
|
|
|
# pylint: disable=possibly-unused-variable
|
2016-10-30 21:18:53 +00:00
|
|
|
def mock_secrets(ldr, node):
|
2016-08-23 04:42:05 +00:00
|
|
|
"""Mock _get_secrets."""
|
|
|
|
try:
|
2019-07-31 19:25:30 +00:00
|
|
|
val = MOCKS["secrets"][1](ldr, node)
|
2016-08-23 04:42:05 +00:00
|
|
|
except HomeAssistantError:
|
|
|
|
val = None
|
2019-07-31 19:25:30 +00:00
|
|
|
res["secrets"][node.value] = val
|
2016-08-23 04:42:05 +00:00
|
|
|
return val
|
|
|
|
|
|
|
|
# Patches with local mock functions
|
|
|
|
for key, val in MOCKS.items():
|
2019-07-31 19:25:30 +00:00
|
|
|
if not secrets and key == "secrets":
|
2018-03-09 03:34:24 +00:00
|
|
|
continue
|
2016-08-25 05:18:32 +00:00
|
|
|
# The * in the key is removed to find the mock_function (side_effect)
|
|
|
|
# This allows us to use one side_effect to patch multiple locations
|
2020-04-04 21:09:34 +00:00
|
|
|
mock_function = locals()[f"mock_{key.replace('*', '')}"]
|
2016-08-23 04:42:05 +00:00
|
|
|
PATCHES[key] = patch(val[0], side_effect=mock_function)
|
|
|
|
|
|
|
|
# Start all patches
|
|
|
|
for pat in PATCHES.values():
|
|
|
|
pat.start()
|
2018-03-09 03:34:24 +00:00
|
|
|
|
|
|
|
if secrets:
|
|
|
|
# Ensure !secrets point to the patched function
|
2021-03-02 20:58:53 +00:00
|
|
|
yaml_loader.SafeLineLoader.add_constructor("!secret", yaml_loader.secret_yaml)
|
|
|
|
|
|
|
|
def secrets_proxy(*args):
|
|
|
|
secrets = Secrets(*args)
|
|
|
|
res["secret_cache"] = secrets._cache
|
|
|
|
return secrets
|
2016-08-23 04:42:05 +00:00
|
|
|
|
|
|
|
try:
|
2021-03-02 20:58:53 +00:00
|
|
|
with patch.object(yaml_loader, "Secrets", secrets_proxy):
|
|
|
|
res["components"] = asyncio.run(async_check_config(config_dir))
|
|
|
|
res["secret_cache"] = {
|
|
|
|
str(key): val for key, val in res["secret_cache"].items()
|
|
|
|
}
|
2019-07-31 19:25:30 +00:00
|
|
|
for err in res["components"].errors:
|
2018-03-09 03:34:24 +00:00
|
|
|
domain = err.domain or ERROR_STR
|
2019-07-31 19:25:30 +00:00
|
|
|
res["except"].setdefault(domain, []).append(err.message)
|
2018-03-09 03:34:24 +00:00
|
|
|
if err.config:
|
2019-07-31 19:25:30 +00:00
|
|
|
res["except"].setdefault(domain, []).append(err.config)
|
2018-03-09 03:34:24 +00:00
|
|
|
|
2016-09-08 20:20:38 +00:00
|
|
|
except Exception as err: # pylint: disable=broad-except
|
2019-07-31 19:25:30 +00:00
|
|
|
print(color("red", "Fatal error while loading config:"), str(err))
|
|
|
|
res["except"].setdefault(ERROR_STR, []).append(str(err))
|
2016-08-23 04:42:05 +00:00
|
|
|
finally:
|
|
|
|
# Stop all patches
|
|
|
|
for pat in PATCHES.values():
|
|
|
|
pat.stop()
|
2018-03-09 03:34:24 +00:00
|
|
|
if secrets:
|
|
|
|
# Ensure !secrets point to the original function
|
2021-03-02 20:58:53 +00:00
|
|
|
yaml_loader.SafeLineLoader.add_constructor(
|
2019-07-31 19:25:30 +00:00
|
|
|
"!secret", yaml_loader.secret_yaml
|
|
|
|
)
|
2016-09-08 20:20:38 +00:00
|
|
|
|
|
|
|
return res
|
2016-08-23 04:42:05 +00:00
|
|
|
|
|
|
|
|
2020-07-06 22:58:53 +00:00
|
|
|
async def async_check_config(config_dir):
|
|
|
|
"""Check the HA config."""
|
|
|
|
hass = core.HomeAssistant()
|
|
|
|
hass.config.config_dir = config_dir
|
2021-09-03 17:15:57 +00:00
|
|
|
await area_registry.async_load(hass)
|
|
|
|
await device_registry.async_load(hass)
|
|
|
|
await entity_registry.async_load(hass)
|
2020-07-06 22:58:53 +00:00
|
|
|
components = await async_check_ha_config_file(hass)
|
|
|
|
await hass.async_stop(force=True)
|
|
|
|
return components
|
|
|
|
|
|
|
|
|
2017-01-27 05:42:14 +00:00
|
|
|
def line_info(obj, **kwargs):
|
|
|
|
"""Display line config source."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if hasattr(obj, "__config_file__"):
|
|
|
|
return color(
|
2020-04-05 15:48:55 +00:00
|
|
|
"cyan", f"[source {obj.__config_file__}:{obj.__line__ or '?'}]", **kwargs
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
return "?"
|
2017-01-27 05:42:14 +00:00
|
|
|
|
|
|
|
|
2016-09-23 07:10:19 +00:00
|
|
|
def dump_dict(layer, indent_count=3, listi=False, **kwargs):
|
2016-08-23 04:42:05 +00:00
|
|
|
"""Display a dict.
|
|
|
|
|
2019-05-09 16:07:56 +00:00
|
|
|
A friendly version of print yaml_loader.yaml.dump(config).
|
2016-08-23 04:42:05 +00:00
|
|
|
"""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2016-09-23 07:10:19 +00:00
|
|
|
def sort_dict_key(val):
|
|
|
|
"""Return the dict key for sorting."""
|
2018-03-30 20:50:08 +00:00
|
|
|
key = str(val[0]).lower()
|
2019-07-31 19:25:30 +00:00
|
|
|
return "0" if key == "platform" else key
|
2016-09-23 07:10:19 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
indent_str = indent_count * " "
|
2016-08-23 04:42:05 +00:00
|
|
|
if listi or isinstance(layer, list):
|
2019-07-31 19:25:30 +00:00
|
|
|
indent_str = indent_str[:-1] + "-"
|
2020-05-02 21:57:48 +00:00
|
|
|
if isinstance(layer, Mapping):
|
2016-09-23 07:10:19 +00:00
|
|
|
for key, value in sorted(layer.items(), key=sort_dict_key):
|
2017-07-06 03:02:16 +00:00
|
|
|
if isinstance(value, (dict, list)):
|
2019-07-31 19:25:30 +00:00
|
|
|
print(indent_str, str(key) + ":", line_info(value, **kwargs))
|
2016-09-21 04:26:40 +00:00
|
|
|
dump_dict(value, indent_count + 2)
|
2016-08-23 04:42:05 +00:00
|
|
|
else:
|
2019-07-31 19:25:30 +00:00
|
|
|
print(indent_str, str(key) + ":", value)
|
|
|
|
indent_str = indent_count * " "
|
2016-08-23 04:42:05 +00:00
|
|
|
if isinstance(layer, Sequence):
|
|
|
|
for i in layer:
|
|
|
|
if isinstance(i, dict):
|
2016-09-24 21:45:01 +00:00
|
|
|
dump_dict(i, indent_count + 2, True)
|
2016-08-23 04:42:05 +00:00
|
|
|
else:
|
2019-07-31 19:25:30 +00:00
|
|
|
print(" ", indent_str, i)
|