2018-10-01 09:21:00 +00:00
|
|
|
"""Message templates for websocket commands."""
|
2021-03-18 14:08:35 +00:00
|
|
|
from __future__ import annotations
|
2018-10-01 09:21:00 +00:00
|
|
|
|
2020-09-22 13:47:04 +00:00
|
|
|
from functools import lru_cache
|
|
|
|
import logging
|
2021-05-21 16:39:18 +00:00
|
|
|
from typing import Any, Final
|
2020-09-22 13:47:04 +00:00
|
|
|
|
2018-10-01 09:21:00 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2020-09-22 13:47:04 +00:00
|
|
|
from homeassistant.core import Event
|
2018-10-01 09:21:00 +00:00
|
|
|
from homeassistant.helpers import config_validation as cv
|
2020-09-22 13:47:04 +00:00
|
|
|
from homeassistant.util.json import (
|
|
|
|
find_paths_unserializable_data,
|
|
|
|
format_unserializable_data,
|
|
|
|
)
|
2020-10-22 23:28:22 +00:00
|
|
|
from homeassistant.util.yaml.loader import JSON_TYPE
|
2018-10-01 09:21:00 +00:00
|
|
|
|
|
|
|
from . import const
|
|
|
|
|
2021-05-21 16:39:18 +00:00
|
|
|
_LOGGER: Final = logging.getLogger(__name__)
|
2019-09-29 17:07:49 +00:00
|
|
|
|
2018-10-01 09:21:00 +00:00
|
|
|
# Minimal requirements of a message
|
2021-05-21 16:39:18 +00:00
|
|
|
MINIMAL_MESSAGE_SCHEMA: Final = vol.Schema(
|
2019-07-31 19:25:30 +00:00
|
|
|
{vol.Required("id"): cv.positive_int, vol.Required("type"): cv.string},
|
|
|
|
extra=vol.ALLOW_EXTRA,
|
|
|
|
)
|
2018-10-01 09:21:00 +00:00
|
|
|
|
|
|
|
# Base schema to extend by message handlers
|
2021-05-21 16:39:18 +00:00
|
|
|
BASE_COMMAND_MESSAGE_SCHEMA: Final = vol.Schema({vol.Required("id"): cv.positive_int})
|
2018-10-01 09:21:00 +00:00
|
|
|
|
2021-05-21 16:39:18 +00:00
|
|
|
IDEN_TEMPLATE: Final = "__IDEN__"
|
|
|
|
IDEN_JSON_TEMPLATE: Final = '"__IDEN__"'
|
2020-10-22 23:28:22 +00:00
|
|
|
|
2018-10-01 09:21:00 +00:00
|
|
|
|
2021-05-21 16:39:18 +00:00
|
|
|
def result_message(iden: int, result: Any = None) -> dict[str, Any]:
|
2018-10-01 09:21:00 +00:00
|
|
|
"""Return a success result message."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return {"id": iden, "type": const.TYPE_RESULT, "success": True, "result": result}
|
2018-10-01 09:21:00 +00:00
|
|
|
|
|
|
|
|
2021-05-21 16:39:18 +00:00
|
|
|
def error_message(iden: int | None, code: str, message: str) -> dict[str, Any]:
|
2018-10-01 09:21:00 +00:00
|
|
|
"""Return an error result message."""
|
|
|
|
return {
|
2019-07-31 19:25:30 +00:00
|
|
|
"id": iden,
|
|
|
|
"type": const.TYPE_RESULT,
|
|
|
|
"success": False,
|
|
|
|
"error": {"code": code, "message": message},
|
2018-10-01 09:21:00 +00:00
|
|
|
}
|
2019-03-11 03:07:09 +00:00
|
|
|
|
|
|
|
|
2021-05-21 16:39:18 +00:00
|
|
|
def event_message(iden: JSON_TYPE, event: Any) -> dict[str, Any]:
|
2019-03-11 03:07:09 +00:00
|
|
|
"""Return an event message."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return {"id": iden, "type": "event", "event": event}
|
2020-09-22 13:47:04 +00:00
|
|
|
|
|
|
|
|
|
|
|
def cached_event_message(iden: int, event: Event) -> str:
|
|
|
|
"""Return an event message.
|
|
|
|
|
|
|
|
Serialize to json once per message.
|
|
|
|
|
|
|
|
Since we can have many clients connected that are
|
|
|
|
all getting many of the same events (mostly state changed)
|
|
|
|
we can avoid serializing the same data for each connection.
|
|
|
|
"""
|
2020-10-22 23:28:22 +00:00
|
|
|
return _cached_event_message(event).replace(IDEN_JSON_TEMPLATE, str(iden), 1)
|
|
|
|
|
|
|
|
|
|
|
|
@lru_cache(maxsize=128)
|
|
|
|
def _cached_event_message(event: Event) -> str:
|
|
|
|
"""Cache and serialize the event to json.
|
|
|
|
|
|
|
|
The IDEN_TEMPLATE is used which will be replaced
|
|
|
|
with the actual iden in cached_event_message
|
|
|
|
"""
|
|
|
|
return message_to_json(event_message(IDEN_TEMPLATE, event))
|
2020-09-22 13:47:04 +00:00
|
|
|
|
|
|
|
|
2021-05-21 16:39:18 +00:00
|
|
|
def message_to_json(message: dict[str, Any]) -> str:
|
2020-09-22 13:47:04 +00:00
|
|
|
"""Serialize a websocket message to json."""
|
|
|
|
try:
|
|
|
|
return const.JSON_DUMP(message)
|
|
|
|
except (ValueError, TypeError):
|
|
|
|
_LOGGER.error(
|
|
|
|
"Unable to serialize to JSON. Bad data found at %s",
|
|
|
|
format_unserializable_data(
|
|
|
|
find_paths_unserializable_data(message, dump=const.JSON_DUMP)
|
|
|
|
),
|
|
|
|
)
|
|
|
|
return const.JSON_DUMP(
|
|
|
|
error_message(
|
|
|
|
message["id"], const.ERR_UNKNOWN_ERROR, "Invalid JSON in response"
|
|
|
|
)
|
|
|
|
)
|