553 lines
20 KiB
Python
553 lines
20 KiB
Python
"""Validate integration translation files."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from functools import partial
|
|
import json
|
|
import re
|
|
from typing import Any
|
|
|
|
import voluptuous as vol
|
|
from voluptuous.humanize import humanize_error
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
from script.translations import upload
|
|
|
|
from .model import Config, Integration
|
|
|
|
UNDEFINED = 0
|
|
REQUIRED = 1
|
|
REMOVED = 2
|
|
|
|
RE_REFERENCE = r"\[\%key:(.+)\%\]"
|
|
RE_TRANSLATION_KEY = re.compile(r"^(?!.+[_-]{2})(?![_-])[a-z0-9-_]+(?<![_-])$")
|
|
RE_COMBINED_REFERENCE = re.compile(r"(.+\[%)|(%\].+)")
|
|
RE_PLACEHOLDER_IN_SINGLE_QUOTES = re.compile(r"'{\w+}'")
|
|
|
|
# Only allow translation of integration names if they contain non-brand names
|
|
ALLOW_NAME_TRANSLATION = {
|
|
"cert_expiry",
|
|
"cpuspeed",
|
|
"emulated_roku",
|
|
"faa_delays",
|
|
"garages_amsterdam",
|
|
"generic",
|
|
"google_travel_time",
|
|
"holiday",
|
|
"homekit_controller",
|
|
"islamic_prayer_times",
|
|
"local_calendar",
|
|
"local_ip",
|
|
"local_todo",
|
|
"nmap_tracker",
|
|
"rpi_power",
|
|
"swiss_public_transport",
|
|
"waze_travel_time",
|
|
"zodiac",
|
|
}
|
|
|
|
REMOVED_TITLE_MSG = (
|
|
"config.title key has been moved out of config and into the root of strings.json. "
|
|
"Starting Home Assistant 0.109 you only need to define this key in the root "
|
|
"if the title needs to be different than the name of your integration in the "
|
|
"manifest."
|
|
)
|
|
|
|
MOVED_TRANSLATIONS_DIRECTORY_MSG = (
|
|
"The '.translations' directory has been moved, the new name is 'translations', "
|
|
"starting with Home Assistant 0.112 your translations will no longer "
|
|
"load if you do not move/rename this "
|
|
)
|
|
|
|
|
|
def allow_name_translation(integration: Integration) -> bool:
|
|
"""Validate that the translation name is not the same as the integration name."""
|
|
# Only enforce for core because custom integrations can't be
|
|
# added to allow list.
|
|
return (
|
|
not integration.core
|
|
or integration.domain in ALLOW_NAME_TRANSLATION
|
|
or integration.quality_scale == "internal"
|
|
)
|
|
|
|
|
|
def check_translations_directory_name(integration: Integration) -> None:
|
|
"""Check that the correct name is used for the translations directory."""
|
|
legacy_translations = integration.path / ".translations"
|
|
translations = integration.path / "translations"
|
|
|
|
if translations.is_dir():
|
|
# No action required
|
|
return
|
|
|
|
if legacy_translations.is_dir():
|
|
integration.add_error("translations", MOVED_TRANSLATIONS_DIRECTORY_MSG)
|
|
|
|
|
|
def find_references(
|
|
strings: dict[str, Any],
|
|
prefix: str,
|
|
found: list[dict[str, str]],
|
|
) -> None:
|
|
"""Find references."""
|
|
for key, value in strings.items():
|
|
if isinstance(value, dict):
|
|
find_references(value, f"{prefix}::{key}", found)
|
|
continue
|
|
|
|
if match := re.match(RE_REFERENCE, value):
|
|
found.append({"source": f"{prefix}::{key}", "ref": match.groups()[0]})
|
|
|
|
|
|
def removed_title_validator(
|
|
config: Config,
|
|
integration: Integration,
|
|
value: Any,
|
|
) -> Any:
|
|
"""Mark removed title."""
|
|
if not config.specific_integrations:
|
|
raise vol.Invalid(REMOVED_TITLE_MSG)
|
|
|
|
# Don't mark it as an error yet for custom components to allow backwards compat.
|
|
integration.add_warning("translations", REMOVED_TITLE_MSG)
|
|
return value
|
|
|
|
|
|
def translation_key_validator(value: str) -> str:
|
|
"""Validate value is valid translation key."""
|
|
if RE_TRANSLATION_KEY.match(value) is None:
|
|
raise vol.Invalid(
|
|
f"Invalid translation key '{value}', need to be [a-z0-9-_]+ and"
|
|
" cannot start or end with a hyphen or underscore."
|
|
)
|
|
|
|
return value
|
|
|
|
|
|
def translation_value_validator(value: Any) -> str:
|
|
"""Validate that the value is a valid translation.
|
|
|
|
- prevents string with HTML
|
|
- prevents strings with single quoted placeholders
|
|
- prevents combined translations
|
|
"""
|
|
string_value = cv.string_with_no_html(value)
|
|
string_value = string_no_single_quoted_placeholders(string_value)
|
|
if RE_COMBINED_REFERENCE.search(string_value):
|
|
raise vol.Invalid("the string should not contain combined translations")
|
|
if string_value != string_value.strip():
|
|
raise vol.Invalid("the string should not contain leading or trailing spaces")
|
|
return string_value
|
|
|
|
|
|
def string_no_single_quoted_placeholders(value: str) -> str:
|
|
"""Validate that the value does not contain placeholders inside single quotes."""
|
|
if RE_PLACEHOLDER_IN_SINGLE_QUOTES.search(value):
|
|
raise vol.Invalid(
|
|
"the string should not contain placeholders inside single quotes"
|
|
)
|
|
return value
|
|
|
|
|
|
def gen_data_entry_schema(
|
|
*,
|
|
config: Config,
|
|
integration: Integration,
|
|
flow_title: int,
|
|
require_step_title: bool,
|
|
mandatory_description: str | None = None,
|
|
) -> vol.All:
|
|
"""Generate a data entry schema."""
|
|
step_title_class = vol.Required if require_step_title else vol.Optional
|
|
schema = {
|
|
vol.Optional("flow_title"): translation_value_validator,
|
|
vol.Required("step"): {
|
|
str: {
|
|
step_title_class("title"): translation_value_validator,
|
|
vol.Optional("description"): translation_value_validator,
|
|
vol.Optional("data"): {str: translation_value_validator},
|
|
vol.Optional("data_description"): {str: translation_value_validator},
|
|
vol.Optional("menu_options"): {str: translation_value_validator},
|
|
vol.Optional("submit"): translation_value_validator,
|
|
vol.Optional("sections"): {
|
|
str: {
|
|
vol.Optional("data"): {str: translation_value_validator},
|
|
vol.Optional("data_description"): {
|
|
str: translation_value_validator
|
|
},
|
|
vol.Optional("description"): translation_value_validator,
|
|
vol.Optional("name"): translation_value_validator,
|
|
},
|
|
},
|
|
}
|
|
},
|
|
vol.Optional("error"): {str: translation_value_validator},
|
|
vol.Optional("abort"): {str: translation_value_validator},
|
|
vol.Optional("progress"): {str: translation_value_validator},
|
|
vol.Optional("create_entry"): {str: translation_value_validator},
|
|
}
|
|
if flow_title == REQUIRED:
|
|
schema[vol.Required("title")] = translation_value_validator
|
|
elif flow_title == REMOVED:
|
|
schema[vol.Optional("title", msg=REMOVED_TITLE_MSG)] = partial(
|
|
removed_title_validator, config, integration
|
|
)
|
|
|
|
def data_description_validator(value: dict[str, Any]) -> dict[str, Any]:
|
|
"""Validate data description."""
|
|
for step_info in value["step"].values():
|
|
if "data_description" not in step_info:
|
|
continue
|
|
|
|
for key in step_info["data_description"]:
|
|
if key not in step_info["data"]:
|
|
raise vol.Invalid(f"data_description key {key} is not in data")
|
|
|
|
return value
|
|
|
|
validators = [vol.Schema(schema), data_description_validator]
|
|
|
|
if mandatory_description is not None:
|
|
|
|
def validate_description_set(value: dict[str, Any]) -> dict[str, Any]:
|
|
"""Validate description is set."""
|
|
steps = value["step"]
|
|
if mandatory_description not in steps:
|
|
raise vol.Invalid(f"{mandatory_description} needs to be defined")
|
|
|
|
if "description" not in steps[mandatory_description]:
|
|
raise vol.Invalid(f"Step {mandatory_description} needs a description")
|
|
|
|
return value
|
|
|
|
validators.append(validate_description_set)
|
|
|
|
if not allow_name_translation(integration):
|
|
|
|
def name_validator(value: dict[str, Any]) -> dict[str, Any]:
|
|
"""Validate name."""
|
|
for step_id, info in value["step"].items():
|
|
if info.get("title") == integration.name:
|
|
raise vol.Invalid(
|
|
f"Do not set title of step {step_id} if it's a brand name "
|
|
"or add exception to ALLOW_NAME_TRANSLATION"
|
|
)
|
|
|
|
return value
|
|
|
|
validators.append(name_validator)
|
|
|
|
return vol.All(*validators)
|
|
|
|
|
|
def gen_issues_schema(config: Config, integration: Integration) -> dict[str, Any]:
|
|
"""Generate the issues schema."""
|
|
return {
|
|
str: vol.All(
|
|
cv.has_at_least_one_key("description", "fix_flow"),
|
|
vol.Schema(
|
|
{
|
|
vol.Required("title"): translation_value_validator,
|
|
vol.Exclusive(
|
|
"description", "fixable"
|
|
): translation_value_validator,
|
|
vol.Exclusive("fix_flow", "fixable"): gen_data_entry_schema(
|
|
config=config,
|
|
integration=integration,
|
|
flow_title=UNDEFINED,
|
|
require_step_title=False,
|
|
),
|
|
},
|
|
),
|
|
)
|
|
}
|
|
|
|
|
|
_EXCEPTIONS_SCHEMA = {
|
|
vol.Optional("exceptions"): cv.schema_with_slug_keys(
|
|
{vol.Optional("message"): translation_value_validator},
|
|
slug_validator=cv.slug,
|
|
),
|
|
}
|
|
|
|
|
|
def gen_strings_schema(config: Config, integration: Integration) -> vol.Schema:
|
|
"""Generate a strings schema."""
|
|
return vol.Schema(
|
|
{
|
|
vol.Optional("title"): translation_value_validator,
|
|
vol.Optional("config"): gen_data_entry_schema(
|
|
config=config,
|
|
integration=integration,
|
|
flow_title=REMOVED,
|
|
require_step_title=False,
|
|
mandatory_description=(
|
|
"user" if integration.integration_type == "helper" else None
|
|
),
|
|
),
|
|
vol.Optional("options"): gen_data_entry_schema(
|
|
config=config,
|
|
integration=integration,
|
|
flow_title=UNDEFINED,
|
|
require_step_title=False,
|
|
),
|
|
vol.Optional("selector"): cv.schema_with_slug_keys(
|
|
{
|
|
"options": cv.schema_with_slug_keys(
|
|
translation_value_validator,
|
|
slug_validator=translation_key_validator,
|
|
)
|
|
},
|
|
slug_validator=vol.Any("_", cv.slug),
|
|
),
|
|
vol.Optional("device_automation"): {
|
|
vol.Optional("action_type"): {str: translation_value_validator},
|
|
vol.Optional("condition_type"): {str: translation_value_validator},
|
|
vol.Optional("trigger_type"): {str: translation_value_validator},
|
|
vol.Optional("trigger_subtype"): {str: translation_value_validator},
|
|
vol.Optional("extra_fields"): {str: translation_value_validator},
|
|
vol.Optional("extra_fields_descriptions"): {
|
|
str: translation_value_validator
|
|
},
|
|
},
|
|
vol.Optional("system_health"): {
|
|
vol.Optional("info"): cv.schema_with_slug_keys(
|
|
translation_value_validator,
|
|
slug_validator=translation_key_validator,
|
|
),
|
|
},
|
|
vol.Optional("config_panel"): cv.schema_with_slug_keys(
|
|
cv.schema_with_slug_keys(
|
|
translation_value_validator,
|
|
slug_validator=translation_key_validator,
|
|
),
|
|
slug_validator=vol.Any("_", cv.slug),
|
|
),
|
|
vol.Optional("application_credentials"): {
|
|
vol.Optional("description"): translation_value_validator,
|
|
},
|
|
vol.Optional("issues"): gen_issues_schema(config, integration),
|
|
vol.Optional("entity_component"): cv.schema_with_slug_keys(
|
|
{
|
|
vol.Optional("name"): str,
|
|
vol.Optional("state"): cv.schema_with_slug_keys(
|
|
translation_value_validator,
|
|
slug_validator=translation_key_validator,
|
|
),
|
|
vol.Optional("state_attributes"): cv.schema_with_slug_keys(
|
|
{
|
|
vol.Optional("name"): str,
|
|
vol.Optional("state"): cv.schema_with_slug_keys(
|
|
translation_value_validator,
|
|
slug_validator=translation_key_validator,
|
|
),
|
|
},
|
|
slug_validator=translation_key_validator,
|
|
),
|
|
},
|
|
slug_validator=vol.Any("_", cv.slug),
|
|
),
|
|
vol.Optional("device"): cv.schema_with_slug_keys(
|
|
{
|
|
vol.Optional("name"): translation_value_validator,
|
|
},
|
|
slug_validator=translation_key_validator,
|
|
),
|
|
vol.Optional("entity"): cv.schema_with_slug_keys(
|
|
cv.schema_with_slug_keys(
|
|
{
|
|
vol.Optional("name"): translation_value_validator,
|
|
vol.Optional("state"): cv.schema_with_slug_keys(
|
|
translation_value_validator,
|
|
slug_validator=translation_key_validator,
|
|
),
|
|
vol.Optional("state_attributes"): cv.schema_with_slug_keys(
|
|
{
|
|
vol.Optional("name"): translation_value_validator,
|
|
vol.Optional("state"): cv.schema_with_slug_keys(
|
|
translation_value_validator,
|
|
slug_validator=translation_key_validator,
|
|
),
|
|
},
|
|
slug_validator=translation_key_validator,
|
|
),
|
|
vol.Optional(
|
|
"unit_of_measurement"
|
|
): translation_value_validator,
|
|
},
|
|
slug_validator=translation_key_validator,
|
|
),
|
|
slug_validator=cv.slug,
|
|
),
|
|
**_EXCEPTIONS_SCHEMA,
|
|
vol.Optional("services"): cv.schema_with_slug_keys(
|
|
{
|
|
vol.Required("name"): translation_value_validator,
|
|
vol.Required("description"): translation_value_validator,
|
|
vol.Optional("fields"): cv.schema_with_slug_keys(
|
|
{
|
|
vol.Required("name"): str,
|
|
vol.Required("description"): translation_value_validator,
|
|
vol.Optional("example"): translation_value_validator,
|
|
},
|
|
slug_validator=translation_key_validator,
|
|
),
|
|
vol.Optional("sections"): cv.schema_with_slug_keys(
|
|
{
|
|
vol.Required("name"): str,
|
|
vol.Optional("description"): translation_value_validator,
|
|
},
|
|
slug_validator=translation_key_validator,
|
|
),
|
|
},
|
|
slug_validator=translation_key_validator,
|
|
),
|
|
vol.Optional("conversation"): {
|
|
vol.Required("agent"): {
|
|
vol.Required("done"): translation_value_validator,
|
|
},
|
|
},
|
|
vol.Optional("common"): vol.Schema({cv.slug: translation_value_validator}),
|
|
}
|
|
)
|
|
|
|
|
|
def gen_auth_schema(config: Config, integration: Integration) -> vol.Schema:
|
|
"""Generate auth schema."""
|
|
return vol.Schema(
|
|
{
|
|
vol.Optional("mfa_setup"): {
|
|
str: gen_data_entry_schema(
|
|
config=config,
|
|
integration=integration,
|
|
flow_title=REQUIRED,
|
|
require_step_title=True,
|
|
)
|
|
},
|
|
vol.Optional("issues"): gen_issues_schema(config, integration),
|
|
**_EXCEPTIONS_SCHEMA,
|
|
}
|
|
)
|
|
|
|
|
|
def gen_ha_hardware_schema(config: Config, integration: Integration):
|
|
"""Generate auth schema."""
|
|
return vol.Schema(
|
|
{
|
|
str: {
|
|
vol.Optional("options"): gen_data_entry_schema(
|
|
config=config,
|
|
integration=integration,
|
|
flow_title=UNDEFINED,
|
|
require_step_title=False,
|
|
)
|
|
}
|
|
}
|
|
)
|
|
|
|
|
|
ONBOARDING_SCHEMA = vol.Schema(
|
|
{
|
|
vol.Required("area"): {str: translation_value_validator},
|
|
vol.Required("dashboard"): {str: {"title": translation_value_validator}},
|
|
}
|
|
)
|
|
|
|
|
|
def validate_translation_file( # noqa: C901
|
|
config: Config,
|
|
integration: Integration,
|
|
all_strings: dict[str, Any] | None,
|
|
) -> None:
|
|
"""Validate translation files for integration."""
|
|
if config.specific_integrations:
|
|
check_translations_directory_name(integration)
|
|
|
|
strings_files = [integration.path / "strings.json"]
|
|
|
|
# Also validate translations for custom integrations
|
|
if config.specific_integrations:
|
|
# Only English needs to be always complete
|
|
strings_files.append(integration.path / "translations/en.json")
|
|
|
|
references: list[dict[str, str]] = []
|
|
|
|
if integration.domain == "auth":
|
|
strings_schema = gen_auth_schema(config, integration)
|
|
elif integration.domain == "onboarding":
|
|
strings_schema = ONBOARDING_SCHEMA
|
|
elif integration.domain == "homeassistant_hardware":
|
|
strings_schema = gen_ha_hardware_schema(config, integration)
|
|
else:
|
|
strings_schema = gen_strings_schema(config, integration)
|
|
|
|
for strings_file in strings_files:
|
|
if not strings_file.is_file():
|
|
continue
|
|
|
|
name = str(strings_file.relative_to(integration.path))
|
|
|
|
try:
|
|
strings = json.loads(strings_file.read_text())
|
|
except ValueError as err:
|
|
integration.add_error("translations", f"Invalid JSON in {name}: {err}")
|
|
continue
|
|
|
|
try:
|
|
strings_schema(strings)
|
|
except vol.Invalid as err:
|
|
integration.add_error(
|
|
"translations", f"Invalid {name}: {humanize_error(strings, err)}"
|
|
)
|
|
else:
|
|
if strings_file.name == "strings.json":
|
|
find_references(strings, name, references)
|
|
|
|
if (title := strings.get("title")) is not None:
|
|
integration.translated_name = True
|
|
if title == integration.name and not allow_name_translation(
|
|
integration
|
|
):
|
|
integration.add_error(
|
|
"translations",
|
|
"Don't specify title in translation strings if it's a brand "
|
|
"name or add exception to ALLOW_NAME_TRANSLATION",
|
|
)
|
|
|
|
if config.specific_integrations:
|
|
return
|
|
|
|
if not all_strings: # Nothing to validate against
|
|
return
|
|
|
|
# Validate references
|
|
for reference in references:
|
|
parts = reference["ref"].split("::")
|
|
search = all_strings
|
|
key = parts.pop(0)
|
|
while parts and key in search:
|
|
search = search[key]
|
|
key = parts.pop(0)
|
|
|
|
if parts or key not in search:
|
|
integration.add_error(
|
|
"translations",
|
|
f"{reference['source']} contains invalid reference {reference['ref']}: Could not find {key}",
|
|
)
|
|
elif match := re.match(RE_REFERENCE, search[key]):
|
|
integration.add_error(
|
|
"translations",
|
|
f"Lokalise supports only one level of references: \"{reference['source']}\" should point to directly to \"{match.groups()[0]}\"",
|
|
)
|
|
|
|
|
|
def validate(integrations: dict[str, Integration], config: Config) -> None:
|
|
"""Handle JSON files inside integrations."""
|
|
if config.specific_integrations:
|
|
all_strings = None
|
|
else:
|
|
all_strings = upload.generate_upload_data() # type: ignore[no-untyped-call]
|
|
|
|
for integration in integrations.values():
|
|
validate_translation_file(config, integration, all_strings)
|