2024-01-19 15:56:56 +00:00
|
|
|
"""Validate integration icon translation files."""
|
2024-03-08 15:36:11 +00:00
|
|
|
|
2024-01-19 15:56:56 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
from typing import Any
|
|
|
|
|
|
|
|
import orjson
|
|
|
|
import voluptuous as vol
|
|
|
|
from voluptuous.humanize import humanize_error
|
|
|
|
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
2024-08-28 09:15:26 +00:00
|
|
|
from homeassistant.helpers.icon import convert_shorthand_service_icon
|
2024-01-19 15:56:56 +00:00
|
|
|
|
|
|
|
from .model import Config, Integration
|
|
|
|
from .translations import translation_key_validator
|
|
|
|
|
|
|
|
|
|
|
|
def icon_value_validator(value: Any) -> str:
|
|
|
|
"""Validate that the icon is a valid icon."""
|
|
|
|
value = cv.string_with_no_html(value)
|
|
|
|
if not value.startswith("mdi:"):
|
|
|
|
raise vol.Invalid(
|
|
|
|
"The icon needs to be a valid icon from Material Design Icons and start with `mdi:`"
|
|
|
|
)
|
|
|
|
return str(value)
|
|
|
|
|
|
|
|
|
|
|
|
def require_default_icon_validator(value: dict) -> dict:
|
|
|
|
"""Validate that a default icon is set."""
|
|
|
|
if "_" not in value:
|
|
|
|
raise vol.Invalid(
|
|
|
|
"An entity component needs to have a default icon defined with `_`"
|
|
|
|
)
|
|
|
|
return value
|
|
|
|
|
|
|
|
|
2024-01-21 11:02:15 +00:00
|
|
|
def ensure_not_same_as_default(value: dict) -> dict:
|
|
|
|
"""Validate an icon isn't the same as its default icon."""
|
|
|
|
for translation_key, section in value.items():
|
|
|
|
if (default := section.get("default")) and (states := section.get("state")):
|
|
|
|
for state, icon in states.items():
|
|
|
|
if icon == default:
|
|
|
|
raise vol.Invalid(
|
|
|
|
f"The icon for state `{translation_key}.{state}` is the"
|
|
|
|
" same as the default icon and thus can be removed"
|
|
|
|
)
|
|
|
|
|
|
|
|
return value
|
|
|
|
|
|
|
|
|
2024-06-25 09:02:00 +00:00
|
|
|
DATA_ENTRY_ICONS_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
"step": {
|
|
|
|
str: {
|
2024-08-26 16:45:28 +00:00
|
|
|
"sections": {
|
2024-06-25 09:02:00 +00:00
|
|
|
str: icon_value_validator,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2024-08-28 12:14:45 +00:00
|
|
|
CORE_SERVICE_ICONS_SCHEMA = cv.schema_with_slug_keys(
|
|
|
|
vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Optional("service"): icon_value_validator,
|
|
|
|
vol.Optional("sections"): cv.schema_with_slug_keys(
|
|
|
|
icon_value_validator, slug_validator=translation_key_validator
|
|
|
|
),
|
|
|
|
}
|
|
|
|
),
|
|
|
|
slug_validator=translation_key_validator,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
CUSTOM_INTEGRATION_SERVICE_ICONS_SCHEMA = cv.schema_with_slug_keys(
|
2024-08-28 09:15:26 +00:00
|
|
|
vol.All(
|
|
|
|
convert_shorthand_service_icon,
|
|
|
|
vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Optional("service"): icon_value_validator,
|
|
|
|
vol.Optional("sections"): cv.schema_with_slug_keys(
|
|
|
|
icon_value_validator, slug_validator=translation_key_validator
|
|
|
|
),
|
|
|
|
}
|
|
|
|
),
|
|
|
|
),
|
|
|
|
slug_validator=translation_key_validator,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2024-08-28 12:14:45 +00:00
|
|
|
def icon_schema(
|
|
|
|
core_integration: bool, integration_type: str, no_entity_platform: bool
|
|
|
|
) -> vol.Schema:
|
2024-05-30 13:46:08 +00:00
|
|
|
"""Create an icon schema."""
|
2024-01-19 15:56:56 +00:00
|
|
|
|
|
|
|
state_validator = cv.schema_with_slug_keys(
|
|
|
|
icon_value_validator,
|
|
|
|
slug_validator=translation_key_validator,
|
|
|
|
)
|
|
|
|
|
|
|
|
def icon_schema_slug(marker: type[vol.Marker]) -> dict[vol.Marker, Any]:
|
|
|
|
return {
|
|
|
|
marker("default"): icon_value_validator,
|
|
|
|
vol.Optional("state"): state_validator,
|
2024-01-21 11:02:15 +00:00
|
|
|
vol.Optional("state_attributes"): vol.All(
|
|
|
|
cv.schema_with_slug_keys(
|
|
|
|
{
|
|
|
|
marker("default"): icon_value_validator,
|
|
|
|
marker("state"): state_validator,
|
|
|
|
},
|
|
|
|
slug_validator=translation_key_validator,
|
|
|
|
),
|
|
|
|
ensure_not_same_as_default,
|
2024-01-19 15:56:56 +00:00
|
|
|
),
|
|
|
|
}
|
|
|
|
|
2024-01-29 18:26:55 +00:00
|
|
|
schema = vol.Schema(
|
2024-01-19 15:56:56 +00:00
|
|
|
{
|
2024-06-25 09:02:00 +00:00
|
|
|
vol.Optional("config"): DATA_ENTRY_ICONS_SCHEMA,
|
|
|
|
vol.Optional("issues"): vol.Schema(
|
|
|
|
{str: {"fix_flow": DATA_ENTRY_ICONS_SCHEMA}}
|
|
|
|
),
|
|
|
|
vol.Optional("options"): DATA_ENTRY_ICONS_SCHEMA,
|
2024-08-28 12:14:45 +00:00
|
|
|
vol.Optional("services"): CORE_SERVICE_ICONS_SCHEMA
|
|
|
|
if core_integration
|
|
|
|
else CUSTOM_INTEGRATION_SERVICE_ICONS_SCHEMA,
|
2024-01-19 15:56:56 +00:00
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2024-01-29 18:26:55 +00:00
|
|
|
if integration_type in ("entity", "helper", "system"):
|
2024-03-14 21:51:18 +00:00
|
|
|
if integration_type != "entity" or no_entity_platform:
|
2024-02-04 21:57:11 +00:00
|
|
|
field = vol.Optional("entity_component")
|
|
|
|
else:
|
|
|
|
field = vol.Required("entity_component")
|
2024-01-29 18:26:55 +00:00
|
|
|
schema = schema.extend(
|
2024-01-19 15:56:56 +00:00
|
|
|
{
|
2024-02-04 21:57:11 +00:00
|
|
|
field: vol.All(
|
2024-01-19 15:56:56 +00:00
|
|
|
cv.schema_with_slug_keys(
|
|
|
|
icon_schema_slug(vol.Required),
|
|
|
|
slug_validator=vol.Any("_", cv.slug),
|
|
|
|
),
|
|
|
|
require_default_icon_validator,
|
2024-01-21 11:02:15 +00:00
|
|
|
ensure_not_same_as_default,
|
2024-01-19 15:56:56 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
)
|
2024-01-29 18:26:55 +00:00
|
|
|
if integration_type not in ("entity", "system"):
|
|
|
|
schema = schema.extend(
|
|
|
|
{
|
|
|
|
vol.Optional("entity"): vol.All(
|
2024-01-21 11:02:15 +00:00
|
|
|
cv.schema_with_slug_keys(
|
2024-01-29 18:26:55 +00:00
|
|
|
cv.schema_with_slug_keys(
|
|
|
|
icon_schema_slug(vol.Optional),
|
|
|
|
slug_validator=translation_key_validator,
|
|
|
|
),
|
|
|
|
slug_validator=cv.slug,
|
2024-01-21 11:02:15 +00:00
|
|
|
),
|
2024-01-29 18:26:55 +00:00
|
|
|
ensure_not_same_as_default,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
return schema
|
2024-01-19 15:56:56 +00:00
|
|
|
|
|
|
|
|
2024-03-16 19:48:37 +00:00
|
|
|
def validate_icon_file(config: Config, integration: Integration) -> None:
|
2024-01-19 15:56:56 +00:00
|
|
|
"""Validate icon file for integration."""
|
|
|
|
icons_file = integration.path / "icons.json"
|
|
|
|
if not icons_file.is_file():
|
|
|
|
return
|
|
|
|
|
|
|
|
name = str(icons_file.relative_to(integration.path))
|
|
|
|
|
|
|
|
try:
|
|
|
|
icons = orjson.loads(icons_file.read_text())
|
|
|
|
except ValueError as err:
|
|
|
|
integration.add_error("icons", f"Invalid JSON in {name}: {err}")
|
|
|
|
return
|
|
|
|
|
2024-03-14 21:51:18 +00:00
|
|
|
no_entity_platform = integration.domain in ("notify", "image_processing")
|
2024-03-14 18:10:52 +00:00
|
|
|
|
2024-08-28 12:14:45 +00:00
|
|
|
schema = icon_schema(
|
|
|
|
integration.core, integration.integration_type, no_entity_platform
|
|
|
|
)
|
2024-01-19 15:56:56 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
schema(icons)
|
|
|
|
except vol.Invalid as err:
|
|
|
|
integration.add_error("icons", f"Invalid {name}: {humanize_error(icons, err)}")
|
|
|
|
|
|
|
|
|
|
|
|
def validate(integrations: dict[str, Integration], config: Config) -> None:
|
|
|
|
"""Handle JSON files inside integrations."""
|
|
|
|
for integration in integrations.values():
|
|
|
|
validate_icon_file(config, integration)
|