core/script/hassfest/zeroconf.py

105 lines
3.3 KiB
Python
Raw Normal View History

"""Generate zeroconf file."""
2021-03-18 21:58:19 +00:00
from __future__ import annotations
from collections import defaultdict
from homeassistant.loader import (
async_process_zeroconf_match_dict,
homekit_always_discover,
)
from .model import Config, Integration
2022-11-09 15:58:20 +00:00
from .serializer import format_python_namespace
2022-11-23 18:05:31 +00:00
def generate_and_validate(integrations: dict[str, Integration]) -> str:
"""Validate and generate zeroconf data."""
service_type_dict = defaultdict(list)
homekit_dict: dict[str, dict[str, str]] = {}
for domain in sorted(integrations):
integration = integrations[domain]
2019-07-31 19:25:30 +00:00
service_types = integration.manifest.get("zeroconf", [])
homekit = integration.manifest.get("homekit", {})
homekit_models = homekit.get("models", [])
if not (service_types or homekit_models):
continue
for entry in service_types:
data = {"domain": domain}
if isinstance(entry, dict):
typ = entry["type"]
data.update(async_process_zeroconf_match_dict(entry))
else:
typ = entry
service_type_dict[typ].append(data)
for model in homekit_models:
if model in homekit_dict:
integration.add_error(
2019-07-31 19:25:30 +00:00
"zeroconf",
f"Integrations {domain} and {homekit_dict[model]} "
"have overlapping HomeKit models",
2019-07-31 19:25:30 +00:00
)
break
homekit_dict[model] = {
"domain": domain,
"always_discover": homekit_always_discover(
integration.manifest["iot_class"]
),
}
# HomeKit models are matched on starting string, make sure none overlap.
warned = set()
2024-11-22 15:53:26 +00:00
for key, value in homekit_dict.items():
if key in warned:
continue
2019-05-23 21:41:57 +00:00
# n^2 yoooo
2024-11-22 15:53:26 +00:00
for key_2, value_2 in homekit_dict.items():
if key == key_2 or key_2 in warned:
continue
if key.startswith(key_2) or key_2.startswith(key):
integration.add_error(
2019-07-31 19:25:30 +00:00
"zeroconf",
2024-11-22 15:53:26 +00:00
f"Integrations {value} and {value_2} "
"have overlapping HomeKit models",
2019-07-31 19:25:30 +00:00
)
warned.add(key)
warned.add(key_2)
break
2022-11-09 15:58:20 +00:00
return format_python_namespace(
{
"HOMEKIT": {key: homekit_dict[key] for key in homekit_dict},
"ZEROCONF": {key: service_type_dict[key] for key in service_type_dict},
}
)
2022-11-23 18:05:31 +00:00
def validate(integrations: dict[str, Integration], config: Config) -> None:
"""Validate zeroconf file."""
2019-07-31 19:25:30 +00:00
zeroconf_path = config.root / "homeassistant/generated/zeroconf.py"
config.cache["zeroconf"] = content = generate_and_validate(integrations)
if config.specific_integrations:
return
if zeroconf_path.read_text() != content:
config.add_error(
"zeroconf",
"File zeroconf.py is not up to date. Run python3 -m script.hassfest",
fixable=True,
)
2022-11-23 18:05:31 +00:00
def generate(integrations: dict[str, Integration], config: Config) -> None:
"""Generate zeroconf file."""
2019-07-31 19:25:30 +00:00
zeroconf_path = config.root / "homeassistant/generated/zeroconf.py"
zeroconf_path.write_text(f"{config.cache['zeroconf']}")