2021-09-13 11:44:22 +00:00
|
|
|
"""The Energy websocket API."""
|
2021-09-27 22:43:29 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2021-12-07 12:16:24 +00:00
|
|
|
import logging
|
2021-11-04 15:46:45 +00:00
|
|
|
from typing import TYPE_CHECKING
|
|
|
|
|
2021-09-13 11:44:22 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.components import websocket_api
|
|
|
|
from homeassistant.core import HomeAssistant, callback
|
|
|
|
|
2021-11-04 15:46:45 +00:00
|
|
|
from .const import DATA_INSTANCE, MAX_QUEUE_BACKLOG
|
2022-03-22 04:14:47 +00:00
|
|
|
from .statistics import list_statistic_ids, validate_statistics
|
2021-11-05 09:40:56 +00:00
|
|
|
from .util import async_migration_in_progress
|
2021-09-13 11:44:22 +00:00
|
|
|
|
2021-11-04 15:46:45 +00:00
|
|
|
if TYPE_CHECKING:
|
|
|
|
from . import Recorder
|
|
|
|
|
2021-12-07 12:16:24 +00:00
|
|
|
_LOGGER: logging.Logger = logging.getLogger(__package__)
|
|
|
|
|
2021-09-13 11:44:22 +00:00
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_setup(hass: HomeAssistant) -> None:
|
|
|
|
"""Set up the recorder websocket API."""
|
|
|
|
websocket_api.async_register_command(hass, ws_validate_statistics)
|
2021-09-27 21:30:13 +00:00
|
|
|
websocket_api.async_register_command(hass, ws_clear_statistics)
|
2022-03-22 04:14:47 +00:00
|
|
|
websocket_api.async_register_command(hass, ws_get_statistics_metadata)
|
2021-09-27 22:43:29 +00:00
|
|
|
websocket_api.async_register_command(hass, ws_update_statistics_metadata)
|
2021-11-04 15:46:45 +00:00
|
|
|
websocket_api.async_register_command(hass, ws_info)
|
2021-12-07 12:16:24 +00:00
|
|
|
websocket_api.async_register_command(hass, ws_backup_start)
|
|
|
|
websocket_api.async_register_command(hass, ws_backup_end)
|
2021-09-13 11:44:22 +00:00
|
|
|
|
|
|
|
|
|
|
|
@websocket_api.websocket_command(
|
|
|
|
{
|
|
|
|
vol.Required("type"): "recorder/validate_statistics",
|
|
|
|
}
|
|
|
|
)
|
|
|
|
@websocket_api.async_response
|
|
|
|
async def ws_validate_statistics(
|
|
|
|
hass: HomeAssistant, connection: websocket_api.ActiveConnection, msg: dict
|
|
|
|
) -> None:
|
|
|
|
"""Fetch a list of available statistic_id."""
|
2022-03-18 09:09:01 +00:00
|
|
|
instance: Recorder = hass.data[DATA_INSTANCE]
|
|
|
|
statistic_ids = await instance.async_add_executor_job(
|
2021-09-13 11:44:22 +00:00
|
|
|
validate_statistics,
|
|
|
|
hass,
|
|
|
|
)
|
|
|
|
connection.send_result(msg["id"], statistic_ids)
|
2021-09-27 21:30:13 +00:00
|
|
|
|
|
|
|
|
|
|
|
@websocket_api.require_admin
|
|
|
|
@websocket_api.websocket_command(
|
|
|
|
{
|
|
|
|
vol.Required("type"): "recorder/clear_statistics",
|
|
|
|
vol.Required("statistic_ids"): [str],
|
|
|
|
}
|
|
|
|
)
|
|
|
|
@callback
|
|
|
|
def ws_clear_statistics(
|
|
|
|
hass: HomeAssistant, connection: websocket_api.ActiveConnection, msg: dict
|
|
|
|
) -> None:
|
|
|
|
"""Clear statistics for a list of statistic_ids.
|
|
|
|
|
|
|
|
Note: The WS call posts a job to the recorder's queue and then returns, it doesn't
|
|
|
|
wait until the job is completed.
|
|
|
|
"""
|
|
|
|
hass.data[DATA_INSTANCE].async_clear_statistics(msg["statistic_ids"])
|
|
|
|
connection.send_result(msg["id"])
|
2021-09-27 22:43:29 +00:00
|
|
|
|
|
|
|
|
2022-03-22 04:14:47 +00:00
|
|
|
@websocket_api.websocket_command(
|
|
|
|
{
|
|
|
|
vol.Required("type"): "recorder/get_statistics_metadata",
|
|
|
|
vol.Optional("statistic_ids"): [str],
|
|
|
|
}
|
|
|
|
)
|
|
|
|
@websocket_api.async_response
|
|
|
|
async def ws_get_statistics_metadata(
|
|
|
|
hass: HomeAssistant, connection: websocket_api.ActiveConnection, msg: dict
|
|
|
|
) -> None:
|
|
|
|
"""Get metadata for a list of statistic_ids."""
|
|
|
|
statistic_ids = await hass.async_add_executor_job(
|
|
|
|
list_statistic_ids, hass, msg.get("statistic_ids")
|
|
|
|
)
|
|
|
|
connection.send_result(msg["id"], statistic_ids)
|
|
|
|
|
|
|
|
|
2021-09-27 22:43:29 +00:00
|
|
|
@websocket_api.require_admin
|
|
|
|
@websocket_api.websocket_command(
|
|
|
|
{
|
|
|
|
vol.Required("type"): "recorder/update_statistics_metadata",
|
|
|
|
vol.Required("statistic_id"): str,
|
|
|
|
vol.Required("unit_of_measurement"): vol.Any(str, None),
|
|
|
|
}
|
|
|
|
)
|
|
|
|
@callback
|
|
|
|
def ws_update_statistics_metadata(
|
|
|
|
hass: HomeAssistant, connection: websocket_api.ActiveConnection, msg: dict
|
|
|
|
) -> None:
|
|
|
|
"""Update statistics metadata for a statistic_id."""
|
|
|
|
hass.data[DATA_INSTANCE].async_update_statistics_metadata(
|
|
|
|
msg["statistic_id"], msg["unit_of_measurement"]
|
|
|
|
)
|
|
|
|
connection.send_result(msg["id"])
|
2021-11-04 15:46:45 +00:00
|
|
|
|
|
|
|
|
|
|
|
@websocket_api.websocket_command(
|
|
|
|
{
|
|
|
|
vol.Required("type"): "recorder/info",
|
|
|
|
}
|
|
|
|
)
|
|
|
|
@callback
|
|
|
|
def ws_info(
|
|
|
|
hass: HomeAssistant, connection: websocket_api.ActiveConnection, msg: dict
|
|
|
|
) -> None:
|
|
|
|
"""Return status of the recorder."""
|
|
|
|
instance: Recorder = hass.data[DATA_INSTANCE]
|
|
|
|
|
|
|
|
backlog = instance.queue.qsize() if instance and instance.queue else None
|
2021-11-05 09:40:56 +00:00
|
|
|
migration_in_progress = async_migration_in_progress(hass)
|
2021-11-04 15:46:45 +00:00
|
|
|
recording = instance.recording if instance else False
|
|
|
|
thread_alive = instance.is_alive() if instance else False
|
|
|
|
|
|
|
|
recorder_info = {
|
|
|
|
"backlog": backlog,
|
|
|
|
"max_backlog": MAX_QUEUE_BACKLOG,
|
|
|
|
"migration_in_progress": migration_in_progress,
|
|
|
|
"recording": recording,
|
|
|
|
"thread_running": thread_alive,
|
|
|
|
}
|
|
|
|
connection.send_result(msg["id"], recorder_info)
|
2021-12-07 12:16:24 +00:00
|
|
|
|
|
|
|
|
2021-12-09 00:49:35 +00:00
|
|
|
@websocket_api.ws_require_user(only_supervisor=True)
|
2021-12-07 12:16:24 +00:00
|
|
|
@websocket_api.websocket_command({vol.Required("type"): "backup/start"})
|
|
|
|
@websocket_api.async_response
|
|
|
|
async def ws_backup_start(
|
|
|
|
hass: HomeAssistant, connection: websocket_api.ActiveConnection, msg: dict
|
|
|
|
) -> None:
|
|
|
|
"""Backup start notification."""
|
|
|
|
|
|
|
|
_LOGGER.info("Backup start notification, locking database for writes")
|
|
|
|
instance: Recorder = hass.data[DATA_INSTANCE]
|
|
|
|
try:
|
|
|
|
await instance.lock_database()
|
|
|
|
except TimeoutError as err:
|
|
|
|
connection.send_error(msg["id"], "timeout_error", str(err))
|
|
|
|
return
|
|
|
|
connection.send_result(msg["id"])
|
|
|
|
|
|
|
|
|
2021-12-09 00:49:35 +00:00
|
|
|
@websocket_api.ws_require_user(only_supervisor=True)
|
2021-12-07 12:16:24 +00:00
|
|
|
@websocket_api.websocket_command({vol.Required("type"): "backup/end"})
|
|
|
|
@websocket_api.async_response
|
|
|
|
async def ws_backup_end(
|
|
|
|
hass: HomeAssistant, connection: websocket_api.ActiveConnection, msg: dict
|
|
|
|
) -> None:
|
|
|
|
"""Backup end notification."""
|
|
|
|
|
|
|
|
instance: Recorder = hass.data[DATA_INSTANCE]
|
|
|
|
_LOGGER.info("Backup end notification, releasing write lock")
|
|
|
|
if not instance.unlock_database():
|
|
|
|
connection.send_error(
|
|
|
|
msg["id"], "database_unlock_failed", "Failed to unlock database."
|
|
|
|
)
|
|
|
|
connection.send_result(msg["id"])
|