2019-05-21 22:36:26 +00:00
|
|
|
"""Generate zeroconf file."""
|
2021-03-18 21:58:19 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2019-05-31 18:58:48 +00:00
|
|
|
from collections import OrderedDict, defaultdict
|
2019-05-21 22:36:26 +00:00
|
|
|
import json
|
|
|
|
|
2019-12-09 15:24:03 +00:00
|
|
|
from .model import Config, Integration
|
2019-05-21 22:36:26 +00:00
|
|
|
|
|
|
|
BASE = """
|
|
|
|
\"\"\"Automatically generated by hassfest.
|
|
|
|
|
2019-05-30 16:41:30 +00:00
|
|
|
To update, run python3 -m script.hassfest
|
2019-05-21 22:36:26 +00:00
|
|
|
\"\"\"
|
|
|
|
|
2019-09-09 19:01:49 +00:00
|
|
|
# fmt: off
|
2019-05-21 22:36:26 +00:00
|
|
|
|
2019-05-29 21:20:06 +00:00
|
|
|
ZEROCONF = {}
|
2019-05-31 18:58:48 +00:00
|
|
|
|
|
|
|
HOMEKIT = {}
|
2019-05-21 22:36:26 +00:00
|
|
|
""".strip()
|
|
|
|
|
|
|
|
|
2021-03-18 21:58:19 +00:00
|
|
|
def generate_and_validate(integrations: dict[str, Integration]):
|
2019-05-21 22:36:26 +00:00
|
|
|
"""Validate and generate zeroconf data."""
|
2019-05-31 18:58:48 +00:00
|
|
|
service_type_dict = defaultdict(list)
|
|
|
|
homekit_dict = {}
|
2019-05-21 22:36:26 +00:00
|
|
|
|
|
|
|
for domain in sorted(integrations):
|
|
|
|
integration = integrations[domain]
|
|
|
|
|
|
|
|
if not integration.manifest:
|
|
|
|
continue
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
service_types = integration.manifest.get("zeroconf", [])
|
|
|
|
homekit = integration.manifest.get("homekit", {})
|
|
|
|
homekit_models = homekit.get("models", [])
|
2019-05-21 22:36:26 +00:00
|
|
|
|
2020-06-15 11:38:38 +00:00
|
|
|
if not (service_types or homekit_models):
|
2019-05-27 02:48:27 +00:00
|
|
|
continue
|
|
|
|
|
2020-09-11 10:19:21 +00:00
|
|
|
for entry in service_types:
|
|
|
|
data = {"domain": domain}
|
|
|
|
if isinstance(entry, dict):
|
|
|
|
typ = entry["type"]
|
|
|
|
entry_without_type = entry.copy()
|
|
|
|
del entry_without_type["type"]
|
|
|
|
data.update(entry_without_type)
|
|
|
|
else:
|
|
|
|
typ = entry
|
|
|
|
|
|
|
|
service_type_dict[typ].append(data)
|
2019-05-21 22:36:26 +00:00
|
|
|
|
2019-05-31 18:58:48 +00:00
|
|
|
for model in homekit_models:
|
|
|
|
if model in homekit_dict:
|
|
|
|
integration.add_error(
|
2019-07-31 19:25:30 +00:00
|
|
|
"zeroconf",
|
2020-04-05 15:48:55 +00:00
|
|
|
f"Integrations {domain} and {homekit_dict[model]} "
|
|
|
|
"have overlapping HomeKit models",
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-05-31 18:58:48 +00:00
|
|
|
break
|
2019-05-21 22:36:26 +00:00
|
|
|
|
2019-05-31 18:58:48 +00:00
|
|
|
homekit_dict[model] = domain
|
|
|
|
|
|
|
|
# HomeKit models are matched on starting string, make sure none overlap.
|
|
|
|
warned = set()
|
|
|
|
for key in homekit_dict:
|
|
|
|
if key in warned:
|
|
|
|
continue
|
2019-05-23 21:41:57 +00:00
|
|
|
|
2019-05-31 18:58:48 +00:00
|
|
|
# n^2 yoooo
|
|
|
|
for key_2 in homekit_dict:
|
|
|
|
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",
|
2020-04-05 15:48:55 +00:00
|
|
|
f"Integrations {homekit_dict[key]} and {homekit_dict[key_2]} "
|
|
|
|
"have overlapping HomeKit models",
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-05-31 18:58:48 +00:00
|
|
|
warned.add(key)
|
|
|
|
warned.add(key_2)
|
|
|
|
break
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
zeroconf = OrderedDict(
|
|
|
|
(key, service_type_dict[key]) for key in sorted(service_type_dict)
|
2019-05-31 18:58:48 +00:00
|
|
|
)
|
2019-07-31 19:25:30 +00:00
|
|
|
homekit = OrderedDict((key, homekit_dict[key]) for key in sorted(homekit_dict))
|
|
|
|
|
|
|
|
return BASE.format(json.dumps(zeroconf, indent=4), json.dumps(homekit, indent=4))
|
2019-05-21 22:36:26 +00:00
|
|
|
|
|
|
|
|
2021-03-18 21:58:19 +00:00
|
|
|
def validate(integrations: dict[str, Integration], config: Config):
|
2019-05-21 22:36:26 +00:00
|
|
|
"""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)
|
2019-05-21 22:36:26 +00:00
|
|
|
|
2020-04-16 16:00:04 +00:00
|
|
|
if config.specific_integrations:
|
|
|
|
return
|
|
|
|
|
2020-04-05 10:49:57 +00:00
|
|
|
with open(str(zeroconf_path)) as fp:
|
2019-05-29 18:19:50 +00:00
|
|
|
current = fp.read().strip()
|
|
|
|
if current != content:
|
2019-05-21 22:36:26 +00:00
|
|
|
config.add_error(
|
|
|
|
"zeroconf",
|
2020-01-02 19:17:10 +00:00
|
|
|
"File zeroconf.py is not up to date. Run python3 -m script.hassfest",
|
2019-07-31 19:25:30 +00:00
|
|
|
fixable=True,
|
2019-05-21 22:36:26 +00:00
|
|
|
)
|
|
|
|
return
|
|
|
|
|
|
|
|
|
2021-03-18 21:58:19 +00:00
|
|
|
def generate(integrations: dict[str, Integration], config: Config):
|
2019-05-21 22:36:26 +00:00
|
|
|
"""Generate zeroconf file."""
|
2019-07-31 19:25:30 +00:00
|
|
|
zeroconf_path = config.root / "homeassistant/generated/zeroconf.py"
|
|
|
|
with open(str(zeroconf_path), "w") as fp:
|
2020-04-04 18:17:11 +00:00
|
|
|
fp.write(f"{config.cache['zeroconf']}\n")
|