2020-11-02 14:00:13 +00:00
|
|
|
"""Websocket API for blueprint."""
|
2024-03-08 13:51:32 +00:00
|
|
|
|
2021-03-17 22:34:25 +00:00
|
|
|
from __future__ import annotations
|
2020-11-02 14:00:13 +00:00
|
|
|
|
2023-08-15 13:30:20 +00:00
|
|
|
import asyncio
|
2024-06-25 18:15:11 +00:00
|
|
|
from collections.abc import Callable, Coroutine
|
|
|
|
import functools
|
2022-10-19 02:23:17 +00:00
|
|
|
from typing import Any, cast
|
|
|
|
|
2020-11-02 14:00:13 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.components import websocket_api
|
|
|
|
from homeassistant.core import HomeAssistant, callback
|
2020-11-11 22:32:46 +00:00
|
|
|
from homeassistant.exceptions import HomeAssistantError
|
2020-11-02 14:00:13 +00:00
|
|
|
from homeassistant.helpers import config_validation as cv
|
2020-11-11 22:32:46 +00:00
|
|
|
from homeassistant.util import yaml
|
2020-11-02 14:00:13 +00:00
|
|
|
|
|
|
|
from . import importer, models
|
|
|
|
from .const import DOMAIN
|
2024-06-25 18:15:11 +00:00
|
|
|
from .errors import BlueprintException, FailedToLoad, FileAlreadyExists
|
2020-11-02 14:00:13 +00:00
|
|
|
|
|
|
|
|
|
|
|
@callback
|
2024-01-02 19:48:51 +00:00
|
|
|
def async_setup(hass: HomeAssistant) -> None:
|
2020-11-02 14:00:13 +00:00
|
|
|
"""Set up the websocket API."""
|
2024-06-25 18:15:11 +00:00
|
|
|
websocket_api.async_register_command(hass, ws_delete_blueprint)
|
2020-11-02 14:00:13 +00:00
|
|
|
websocket_api.async_register_command(hass, ws_import_blueprint)
|
2024-06-25 18:15:11 +00:00
|
|
|
websocket_api.async_register_command(hass, ws_list_blueprints)
|
2020-11-11 22:32:46 +00:00
|
|
|
websocket_api.async_register_command(hass, ws_save_blueprint)
|
2024-06-25 18:15:11 +00:00
|
|
|
websocket_api.async_register_command(hass, ws_substitute_blueprint)
|
|
|
|
|
|
|
|
|
|
|
|
def _ws_with_blueprint_domain(
|
|
|
|
func: Callable[
|
|
|
|
[
|
|
|
|
HomeAssistant,
|
|
|
|
websocket_api.ActiveConnection,
|
|
|
|
dict[str, Any],
|
|
|
|
models.DomainBlueprints,
|
|
|
|
],
|
|
|
|
Coroutine[Any, Any, None],
|
|
|
|
],
|
|
|
|
) -> websocket_api.AsyncWebSocketCommandHandler:
|
|
|
|
"""Decorate a function to pass in the domain blueprints."""
|
|
|
|
|
|
|
|
@functools.wraps(func)
|
|
|
|
async def with_domain_blueprints(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
connection: websocket_api.ActiveConnection,
|
|
|
|
msg: dict[str, Any],
|
|
|
|
) -> None:
|
|
|
|
domain_blueprints: models.DomainBlueprints | None = hass.data.get(
|
|
|
|
DOMAIN, {}
|
|
|
|
).get(msg["domain"])
|
|
|
|
if domain_blueprints is None:
|
|
|
|
connection.send_error(
|
|
|
|
msg["id"], websocket_api.ERR_INVALID_FORMAT, "Unsupported domain"
|
|
|
|
)
|
|
|
|
return
|
|
|
|
|
|
|
|
await func(hass, connection, msg, domain_blueprints)
|
|
|
|
|
|
|
|
return with_domain_blueprints
|
2020-11-02 14:00:13 +00:00
|
|
|
|
|
|
|
|
|
|
|
@websocket_api.websocket_command(
|
|
|
|
{
|
|
|
|
vol.Required("type"): "blueprint/list",
|
2020-11-11 22:32:46 +00:00
|
|
|
vol.Required("domain"): cv.string,
|
2020-11-02 14:00:13 +00:00
|
|
|
}
|
|
|
|
)
|
2022-07-11 15:46:32 +00:00
|
|
|
@websocket_api.async_response
|
2022-10-19 02:23:17 +00:00
|
|
|
async def ws_list_blueprints(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
connection: websocket_api.ActiveConnection,
|
|
|
|
msg: dict[str, Any],
|
|
|
|
) -> None:
|
2020-11-02 14:00:13 +00:00
|
|
|
"""List available blueprints."""
|
2022-10-19 02:23:17 +00:00
|
|
|
domain_blueprints: dict[str, models.DomainBlueprints] = hass.data.get(DOMAIN, {})
|
|
|
|
results: dict[str, Any] = {}
|
2020-11-02 14:00:13 +00:00
|
|
|
|
2020-11-11 22:32:46 +00:00
|
|
|
if msg["domain"] not in domain_blueprints:
|
|
|
|
connection.send_result(msg["id"], results)
|
|
|
|
return
|
|
|
|
|
|
|
|
domain_results = await domain_blueprints[msg["domain"]].async_get_blueprints()
|
|
|
|
|
|
|
|
for path, value in domain_results.items():
|
|
|
|
if isinstance(value, models.Blueprint):
|
|
|
|
results[path] = {
|
|
|
|
"metadata": value.metadata,
|
|
|
|
}
|
|
|
|
else:
|
|
|
|
results[path] = {"error": str(value)}
|
2020-11-02 14:00:13 +00:00
|
|
|
|
|
|
|
connection.send_result(msg["id"], results)
|
|
|
|
|
|
|
|
|
|
|
|
@websocket_api.websocket_command(
|
|
|
|
{
|
|
|
|
vol.Required("type"): "blueprint/import",
|
|
|
|
vol.Required("url"): cv.url,
|
|
|
|
}
|
|
|
|
)
|
2022-07-11 15:46:32 +00:00
|
|
|
@websocket_api.async_response
|
2022-10-19 02:23:17 +00:00
|
|
|
async def ws_import_blueprint(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
connection: websocket_api.ActiveConnection,
|
|
|
|
msg: dict[str, Any],
|
|
|
|
) -> None:
|
2020-11-02 14:00:13 +00:00
|
|
|
"""Import a blueprint."""
|
2023-08-15 13:30:20 +00:00
|
|
|
async with asyncio.timeout(10):
|
2020-11-02 14:00:13 +00:00
|
|
|
imported_blueprint = await importer.fetch_blueprint_from_url(hass, msg["url"])
|
|
|
|
|
|
|
|
if imported_blueprint is None:
|
2024-01-02 19:48:51 +00:00
|
|
|
connection.send_error( # type: ignore[unreachable]
|
2020-11-02 14:00:13 +00:00
|
|
|
msg["id"], websocket_api.ERR_NOT_SUPPORTED, "This url is not supported"
|
|
|
|
)
|
|
|
|
return
|
|
|
|
|
2023-11-25 10:49:50 +00:00
|
|
|
# Check it exists and if so, which automations are using it
|
|
|
|
domain = imported_blueprint.blueprint.metadata["domain"]
|
|
|
|
domain_blueprints: models.DomainBlueprints | None = hass.data.get(DOMAIN, {}).get(
|
|
|
|
domain
|
|
|
|
)
|
|
|
|
if domain_blueprints is None:
|
|
|
|
connection.send_error(
|
|
|
|
msg["id"], websocket_api.ERR_INVALID_FORMAT, "Unsupported domain"
|
|
|
|
)
|
|
|
|
return
|
|
|
|
|
|
|
|
suggested_path = f"{imported_blueprint.suggested_filename}.yaml"
|
|
|
|
try:
|
|
|
|
exists = bool(await domain_blueprints.async_get_blueprint(suggested_path))
|
|
|
|
except FailedToLoad:
|
|
|
|
exists = False
|
|
|
|
|
2020-11-02 14:00:13 +00:00
|
|
|
connection.send_result(
|
|
|
|
msg["id"],
|
|
|
|
{
|
|
|
|
"suggested_filename": imported_blueprint.suggested_filename,
|
|
|
|
"raw_data": imported_blueprint.raw_data,
|
|
|
|
"blueprint": {
|
|
|
|
"metadata": imported_blueprint.blueprint.metadata,
|
|
|
|
},
|
2020-11-20 14:24:42 +00:00
|
|
|
"validation_errors": imported_blueprint.blueprint.validate(),
|
2023-11-25 10:49:50 +00:00
|
|
|
"exists": exists,
|
2020-11-02 14:00:13 +00:00
|
|
|
},
|
|
|
|
)
|
2020-11-11 22:32:46 +00:00
|
|
|
|
|
|
|
|
|
|
|
@websocket_api.websocket_command(
|
|
|
|
{
|
|
|
|
vol.Required("type"): "blueprint/save",
|
|
|
|
vol.Required("domain"): cv.string,
|
|
|
|
vol.Required("path"): cv.path,
|
|
|
|
vol.Required("yaml"): cv.string,
|
|
|
|
vol.Optional("source_url"): cv.url,
|
2023-11-25 10:49:50 +00:00
|
|
|
vol.Optional("allow_override"): bool,
|
2020-11-11 22:32:46 +00:00
|
|
|
}
|
|
|
|
)
|
2022-07-11 15:46:32 +00:00
|
|
|
@websocket_api.async_response
|
2024-06-25 18:15:11 +00:00
|
|
|
@_ws_with_blueprint_domain
|
2022-10-19 02:23:17 +00:00
|
|
|
async def ws_save_blueprint(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
connection: websocket_api.ActiveConnection,
|
|
|
|
msg: dict[str, Any],
|
2024-06-25 18:15:11 +00:00
|
|
|
domain_blueprints: models.DomainBlueprints,
|
2022-10-19 02:23:17 +00:00
|
|
|
) -> None:
|
2020-11-11 22:32:46 +00:00
|
|
|
"""Save a blueprint."""
|
|
|
|
|
|
|
|
path = msg["path"]
|
|
|
|
domain = msg["domain"]
|
|
|
|
|
|
|
|
try:
|
2022-10-19 02:23:17 +00:00
|
|
|
yaml_data = cast(dict[str, Any], yaml.parse_yaml(msg["yaml"]))
|
|
|
|
blueprint = models.Blueprint(yaml_data, expected_domain=domain)
|
2020-11-11 22:32:46 +00:00
|
|
|
if "source_url" in msg:
|
|
|
|
blueprint.update_metadata(source_url=msg["source_url"])
|
|
|
|
except HomeAssistantError as err:
|
|
|
|
connection.send_error(msg["id"], websocket_api.ERR_INVALID_FORMAT, str(err))
|
|
|
|
return
|
|
|
|
|
2023-11-25 10:49:50 +00:00
|
|
|
if not path.endswith(".yaml"):
|
|
|
|
path = f"{path}.yaml"
|
|
|
|
|
2020-11-11 22:32:46 +00:00
|
|
|
try:
|
2024-06-25 18:15:11 +00:00
|
|
|
overrides_existing = await domain_blueprints.async_add_blueprint(
|
2023-11-25 10:49:50 +00:00
|
|
|
blueprint, path, allow_override=msg.get("allow_override", False)
|
|
|
|
)
|
2020-11-11 22:32:46 +00:00
|
|
|
except FileAlreadyExists:
|
|
|
|
connection.send_error(msg["id"], "already_exists", "File already exists")
|
|
|
|
return
|
|
|
|
except OSError as err:
|
|
|
|
connection.send_error(msg["id"], websocket_api.ERR_UNKNOWN_ERROR, str(err))
|
|
|
|
return
|
|
|
|
|
|
|
|
connection.send_result(
|
|
|
|
msg["id"],
|
2023-11-25 10:49:50 +00:00
|
|
|
{
|
|
|
|
"overrides_existing": overrides_existing,
|
|
|
|
},
|
2020-11-11 22:32:46 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@websocket_api.websocket_command(
|
|
|
|
{
|
|
|
|
vol.Required("type"): "blueprint/delete",
|
|
|
|
vol.Required("domain"): cv.string,
|
|
|
|
vol.Required("path"): cv.path,
|
|
|
|
}
|
|
|
|
)
|
2022-07-11 15:46:32 +00:00
|
|
|
@websocket_api.async_response
|
2024-06-25 18:15:11 +00:00
|
|
|
@_ws_with_blueprint_domain
|
2022-10-19 02:23:17 +00:00
|
|
|
async def ws_delete_blueprint(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
connection: websocket_api.ActiveConnection,
|
|
|
|
msg: dict[str, Any],
|
2024-06-25 18:15:11 +00:00
|
|
|
domain_blueprints: models.DomainBlueprints,
|
2022-10-19 02:23:17 +00:00
|
|
|
) -> None:
|
2020-11-11 22:32:46 +00:00
|
|
|
"""Delete a blueprint."""
|
2024-06-25 18:15:11 +00:00
|
|
|
try:
|
|
|
|
await domain_blueprints.async_remove_blueprint(msg["path"])
|
|
|
|
except OSError as err:
|
|
|
|
connection.send_error(msg["id"], websocket_api.ERR_UNKNOWN_ERROR, str(err))
|
|
|
|
return
|
2020-11-11 22:32:46 +00:00
|
|
|
|
2024-06-25 18:15:11 +00:00
|
|
|
connection.send_result(
|
|
|
|
msg["id"],
|
|
|
|
)
|
2020-11-11 22:32:46 +00:00
|
|
|
|
|
|
|
|
2024-06-25 18:15:11 +00:00
|
|
|
@websocket_api.websocket_command(
|
|
|
|
{
|
|
|
|
vol.Required("type"): "blueprint/substitute",
|
|
|
|
vol.Required("domain"): cv.string,
|
|
|
|
vol.Required("path"): cv.path,
|
|
|
|
vol.Required("input"): dict,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
@websocket_api.async_response
|
|
|
|
@_ws_with_blueprint_domain
|
|
|
|
async def ws_substitute_blueprint(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
connection: websocket_api.ActiveConnection,
|
|
|
|
msg: dict[str, Any],
|
|
|
|
domain_blueprints: models.DomainBlueprints,
|
|
|
|
) -> None:
|
|
|
|
"""Process a blueprinted config to allow editing."""
|
|
|
|
|
|
|
|
blueprint_config = {"use_blueprint": {"path": msg["path"], "input": msg["input"]}}
|
|
|
|
|
|
|
|
try:
|
|
|
|
blueprint_inputs = await domain_blueprints.async_inputs_from_config(
|
|
|
|
blueprint_config
|
2020-11-11 22:32:46 +00:00
|
|
|
)
|
2024-06-25 18:15:11 +00:00
|
|
|
except BlueprintException as err:
|
|
|
|
connection.send_error(msg["id"], websocket_api.ERR_UNKNOWN_ERROR, str(err))
|
|
|
|
return
|
2020-11-11 22:32:46 +00:00
|
|
|
|
|
|
|
try:
|
2024-06-25 18:15:11 +00:00
|
|
|
config = blueprint_inputs.async_substitute()
|
|
|
|
except yaml.UndefinedSubstitution as err:
|
2020-11-11 22:32:46 +00:00
|
|
|
connection.send_error(msg["id"], websocket_api.ERR_UNKNOWN_ERROR, str(err))
|
|
|
|
return
|
|
|
|
|
2024-06-25 18:15:11 +00:00
|
|
|
connection.send_result(msg["id"], {"substituted_config": config})
|