2019-04-13 20:17:01 +00:00
|
|
|
"""Models for manifest validator."""
|
2024-03-08 15:36:11 +00:00
|
|
|
|
2021-03-18 21:58:19 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2022-12-23 14:46:00 +00:00
|
|
|
from dataclasses import dataclass, field
|
2024-11-22 18:27:40 +00:00
|
|
|
from enum import IntEnum
|
2019-04-13 20:17:01 +00:00
|
|
|
import json
|
|
|
|
import pathlib
|
2023-09-19 15:30:38 +00:00
|
|
|
from typing import Any, Literal
|
2019-04-13 20:17:01 +00:00
|
|
|
|
|
|
|
|
2022-12-23 14:46:00 +00:00
|
|
|
@dataclass
|
2019-04-13 20:17:01 +00:00
|
|
|
class Error:
|
|
|
|
"""Error validating an integration."""
|
|
|
|
|
2022-12-23 14:46:00 +00:00
|
|
|
plugin: str
|
|
|
|
error: str
|
|
|
|
fixable: bool = False
|
2019-04-13 20:17:01 +00:00
|
|
|
|
|
|
|
def __str__(self) -> str:
|
|
|
|
"""Represent error as string."""
|
2020-04-05 10:49:57 +00:00
|
|
|
return f"[{self.plugin.upper()}] {self.error}"
|
2019-04-13 20:17:01 +00:00
|
|
|
|
|
|
|
|
2022-12-23 14:46:00 +00:00
|
|
|
@dataclass
|
2019-04-13 20:17:01 +00:00
|
|
|
class Config:
|
|
|
|
"""Config for the run."""
|
|
|
|
|
2022-12-23 14:46:00 +00:00
|
|
|
specific_integrations: list[pathlib.Path] | None
|
|
|
|
root: pathlib.Path
|
2023-09-19 15:30:38 +00:00
|
|
|
action: Literal["validate", "generate"]
|
2022-12-23 14:46:00 +00:00
|
|
|
requirements: bool
|
2024-12-11 07:55:00 +00:00
|
|
|
core_integrations_path: pathlib.Path = field(init=False)
|
2022-12-23 14:46:00 +00:00
|
|
|
errors: list[Error] = field(default_factory=list)
|
|
|
|
cache: dict[str, Any] = field(default_factory=dict)
|
|
|
|
plugins: set[str] = field(default_factory=set)
|
2019-04-13 20:17:01 +00:00
|
|
|
|
2024-12-11 07:55:00 +00:00
|
|
|
def __post_init__(self) -> None:
|
|
|
|
"""Post init."""
|
|
|
|
self.core_integrations_path = self.root / "homeassistant/components"
|
|
|
|
|
2021-04-26 12:23:21 +00:00
|
|
|
def add_error(self, *args: Any, **kwargs: Any) -> None:
|
2019-04-13 20:17:01 +00:00
|
|
|
"""Add an error."""
|
|
|
|
self.errors.append(Error(*args, **kwargs))
|
|
|
|
|
|
|
|
|
2022-12-23 14:46:00 +00:00
|
|
|
@dataclass
|
2022-09-28 15:31:48 +00:00
|
|
|
class Brand:
|
|
|
|
"""Represent a brand in our validator."""
|
|
|
|
|
|
|
|
@classmethod
|
2022-11-14 11:41:49 +00:00
|
|
|
def load_dir(cls, path: pathlib.Path, config: Config) -> dict[str, Brand]:
|
2022-09-28 15:31:48 +00:00
|
|
|
"""Load all brands in a directory."""
|
|
|
|
assert path.is_dir()
|
2022-11-14 11:41:49 +00:00
|
|
|
brands: dict[str, Brand] = {}
|
2022-09-28 15:31:48 +00:00
|
|
|
for fil in path.iterdir():
|
|
|
|
brand = cls(fil)
|
|
|
|
brand.load_brand(config)
|
|
|
|
brands[brand.domain] = brand
|
|
|
|
|
|
|
|
return brands
|
|
|
|
|
2022-12-23 14:46:00 +00:00
|
|
|
path: pathlib.Path
|
|
|
|
_brand: dict[str, Any] | None = None
|
2022-11-17 08:10:37 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def brand(self) -> dict[str, Any]:
|
|
|
|
"""Guarded access to brand."""
|
|
|
|
assert self._brand is not None, "brand has not been loaded"
|
|
|
|
return self._brand
|
2022-09-28 15:31:48 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def domain(self) -> str:
|
|
|
|
"""Integration domain."""
|
|
|
|
return self.path.stem
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self) -> str | None:
|
|
|
|
"""Return name of the integration."""
|
|
|
|
return self.brand.get("name")
|
|
|
|
|
|
|
|
@property
|
|
|
|
def integrations(self) -> list[str]:
|
|
|
|
"""Return the sub integrations of this brand."""
|
2022-11-14 11:41:49 +00:00
|
|
|
return self.brand.get("integrations", [])
|
2022-09-28 15:31:48 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def iot_standards(self) -> list[str]:
|
|
|
|
"""Return list of supported IoT standards."""
|
|
|
|
return self.brand.get("iot_standards", [])
|
|
|
|
|
|
|
|
def load_brand(self, config: Config) -> None:
|
|
|
|
"""Load brand file."""
|
|
|
|
if not self.path.is_file():
|
|
|
|
config.add_error("model", f"Brand file {self.path} not found")
|
|
|
|
return
|
|
|
|
|
|
|
|
try:
|
2022-11-14 11:41:49 +00:00
|
|
|
brand: dict[str, Any] = json.loads(self.path.read_text())
|
2022-09-28 15:31:48 +00:00
|
|
|
except ValueError as err:
|
|
|
|
config.add_error(
|
|
|
|
"model", f"Brand file {self.path.name} contains invalid JSON: {err}"
|
|
|
|
)
|
|
|
|
return
|
|
|
|
|
2022-11-17 08:10:37 +00:00
|
|
|
self._brand = brand
|
2022-09-28 15:31:48 +00:00
|
|
|
|
|
|
|
|
2022-12-23 14:46:00 +00:00
|
|
|
@dataclass
|
2019-04-13 20:17:01 +00:00
|
|
|
class Integration:
|
|
|
|
"""Represent an integration in our validator."""
|
|
|
|
|
|
|
|
@classmethod
|
2024-08-28 14:38:12 +00:00
|
|
|
def load_dir(cls, path: pathlib.Path, config: Config) -> dict[str, Integration]:
|
2019-04-13 20:17:01 +00:00
|
|
|
"""Load all integrations in a directory."""
|
|
|
|
assert path.is_dir()
|
2022-11-14 11:41:49 +00:00
|
|
|
integrations: dict[str, Integration] = {}
|
2019-04-13 20:17:01 +00:00
|
|
|
for fil in path.iterdir():
|
2019-07-31 19:25:30 +00:00
|
|
|
if fil.is_file() or fil.name == "__pycache__":
|
2019-04-13 20:17:01 +00:00
|
|
|
continue
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
init = fil / "__init__.py"
|
2022-10-21 03:09:06 +00:00
|
|
|
manifest = fil / "manifest.json"
|
|
|
|
if not init.exists() and not manifest.exists():
|
2019-07-31 19:25:30 +00:00
|
|
|
print(
|
2022-10-21 03:09:06 +00:00
|
|
|
f"Warning: {init} and manifest.json missing, "
|
|
|
|
"skipping directory. If this is your development "
|
|
|
|
"environment, you can safely delete this folder."
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-04-29 08:53:27 +00:00
|
|
|
continue
|
|
|
|
|
2024-08-28 14:38:12 +00:00
|
|
|
integration = cls(fil, config)
|
2019-04-13 20:17:01 +00:00
|
|
|
integration.load_manifest()
|
|
|
|
integrations[integration.domain] = integration
|
|
|
|
|
|
|
|
return integrations
|
|
|
|
|
2022-12-23 14:46:00 +00:00
|
|
|
path: pathlib.Path
|
2024-08-28 14:38:12 +00:00
|
|
|
_config: Config
|
2022-12-23 14:46:00 +00:00
|
|
|
_manifest: dict[str, Any] | None = None
|
2023-02-08 20:48:58 +00:00
|
|
|
manifest_path: pathlib.Path | None = None
|
2022-12-23 14:46:00 +00:00
|
|
|
errors: list[Error] = field(default_factory=list)
|
|
|
|
warnings: list[Error] = field(default_factory=list)
|
|
|
|
translated_name: bool = False
|
2019-04-13 20:17:01 +00:00
|
|
|
|
2022-11-17 08:10:37 +00:00
|
|
|
@property
|
|
|
|
def manifest(self) -> dict[str, Any]:
|
|
|
|
"""Guarded access to manifest."""
|
|
|
|
assert self._manifest is not None, "manifest has not been loaded"
|
|
|
|
return self._manifest
|
|
|
|
|
2019-04-13 20:17:01 +00:00
|
|
|
@property
|
|
|
|
def domain(self) -> str:
|
|
|
|
"""Integration domain."""
|
|
|
|
return self.path.name
|
|
|
|
|
2021-01-25 12:31:14 +00:00
|
|
|
@property
|
|
|
|
def core(self) -> bool:
|
|
|
|
"""Core integration."""
|
2024-08-28 14:38:12 +00:00
|
|
|
return self.path.as_posix().startswith(
|
|
|
|
self._config.core_integrations_path.as_posix()
|
|
|
|
)
|
2021-01-25 12:31:14 +00:00
|
|
|
|
2020-08-26 08:20:14 +00:00
|
|
|
@property
|
2021-03-18 21:58:19 +00:00
|
|
|
def disabled(self) -> str | None:
|
2021-05-17 06:12:23 +00:00
|
|
|
"""Return if integration is disabled."""
|
2020-08-26 08:20:14 +00:00
|
|
|
return self.manifest.get("disabled")
|
|
|
|
|
2021-07-22 06:37:33 +00:00
|
|
|
@property
|
|
|
|
def name(self) -> str:
|
|
|
|
"""Return name of the integration."""
|
2022-11-14 11:41:49 +00:00
|
|
|
name: str = self.manifest["name"]
|
|
|
|
return name
|
2021-07-22 06:37:33 +00:00
|
|
|
|
|
|
|
@property
|
2022-11-14 11:41:49 +00:00
|
|
|
def quality_scale(self) -> str | None:
|
2021-07-22 06:37:33 +00:00
|
|
|
"""Return quality scale of the integration."""
|
|
|
|
return self.manifest.get("quality_scale")
|
|
|
|
|
2021-08-16 14:28:26 +00:00
|
|
|
@property
|
2022-05-01 23:26:22 +00:00
|
|
|
def config_flow(self) -> bool:
|
2021-08-16 14:28:26 +00:00
|
|
|
"""Return if the integration has a config flow."""
|
2022-05-01 23:26:22 +00:00
|
|
|
return self.manifest.get("config_flow", False)
|
2021-08-16 14:28:26 +00:00
|
|
|
|
2020-04-03 19:58:19 +00:00
|
|
|
@property
|
2021-03-18 21:58:19 +00:00
|
|
|
def requirements(self) -> list[str]:
|
2020-04-03 19:58:19 +00:00
|
|
|
"""List of requirements."""
|
|
|
|
return self.manifest.get("requirements", [])
|
|
|
|
|
|
|
|
@property
|
2021-03-18 21:58:19 +00:00
|
|
|
def dependencies(self) -> list[str]:
|
2020-04-03 19:58:19 +00:00
|
|
|
"""List of dependencies."""
|
|
|
|
return self.manifest.get("dependencies", [])
|
|
|
|
|
2022-07-12 20:49:54 +00:00
|
|
|
@property
|
2022-10-21 03:09:06 +00:00
|
|
|
def supported_by(self) -> str:
|
|
|
|
"""Return the integration supported by this virtual integration."""
|
|
|
|
return self.manifest.get("supported_by", {})
|
2022-07-12 20:49:54 +00:00
|
|
|
|
2022-03-21 03:38:13 +00:00
|
|
|
@property
|
|
|
|
def integration_type(self) -> str:
|
|
|
|
"""Get integration_type."""
|
2022-10-19 10:41:43 +00:00
|
|
|
return self.manifest.get("integration_type", "hub")
|
2022-03-21 03:38:13 +00:00
|
|
|
|
2022-09-28 15:31:48 +00:00
|
|
|
@property
|
|
|
|
def iot_class(self) -> str | None:
|
|
|
|
"""Return the integration IoT Class."""
|
|
|
|
return self.manifest.get("iot_class")
|
|
|
|
|
2022-10-21 03:09:06 +00:00
|
|
|
@property
|
2022-10-25 11:43:40 +00:00
|
|
|
def iot_standards(self) -> list[str]:
|
2022-10-21 03:09:06 +00:00
|
|
|
"""Return the IoT standard supported by this virtual integration."""
|
2022-10-25 11:43:40 +00:00
|
|
|
return self.manifest.get("iot_standards", [])
|
2022-10-21 03:09:06 +00:00
|
|
|
|
2021-04-26 12:23:21 +00:00
|
|
|
def add_error(self, *args: Any, **kwargs: Any) -> None:
|
2019-04-13 20:17:01 +00:00
|
|
|
"""Add an error."""
|
|
|
|
self.errors.append(Error(*args, **kwargs))
|
|
|
|
|
2021-04-29 09:43:23 +00:00
|
|
|
def add_warning(self, *args: Any, **kwargs: Any) -> None:
|
2022-09-23 01:54:22 +00:00
|
|
|
"""Add a warning."""
|
2020-04-17 01:00:30 +00:00
|
|
|
self.warnings.append(Error(*args, **kwargs))
|
|
|
|
|
2019-04-13 20:17:01 +00:00
|
|
|
def load_manifest(self) -> None:
|
|
|
|
"""Load manifest."""
|
2019-07-31 19:25:30 +00:00
|
|
|
manifest_path = self.path / "manifest.json"
|
2019-04-18 20:40:46 +00:00
|
|
|
if not manifest_path.is_file():
|
2019-08-23 16:53:33 +00:00
|
|
|
self.add_error("model", f"Manifest file {manifest_path} not found")
|
2019-04-13 20:17:01 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
try:
|
2022-11-14 11:41:49 +00:00
|
|
|
manifest: dict[str, Any] = json.loads(manifest_path.read_text())
|
2019-04-13 20:17:01 +00:00
|
|
|
except ValueError as err:
|
2019-08-23 16:53:33 +00:00
|
|
|
self.add_error("model", f"Manifest contains invalid JSON: {err}")
|
2019-04-13 20:17:01 +00:00
|
|
|
return
|
|
|
|
|
2022-11-17 08:10:37 +00:00
|
|
|
self._manifest = manifest
|
2023-02-08 20:48:58 +00:00
|
|
|
self.manifest_path = manifest_path
|
2024-11-22 18:27:40 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ScaledQualityScaleTiers(IntEnum):
|
|
|
|
"""Supported manifest quality scales."""
|
|
|
|
|
|
|
|
BRONZE = 1
|
|
|
|
SILVER = 2
|
|
|
|
GOLD = 3
|
|
|
|
PLATINUM = 4
|