198 lines
6.1 KiB
Python

import logging
import voluptuous as vol
from datetime import datetime, timezone
from homeassistant.components.websocket_api import (
event_message,
async_register_command,
)
from homeassistant.components import websocket_api
from homeassistant.core import callback
from .const import (
BROWSER_ID,
DATA_STORE,
WS_CONNECT,
WS_LOG,
WS_RECALL_ID,
WS_REGISTER,
WS_SETTINGS,
WS_UNREGISTER,
WS_UPDATE,
DOMAIN,
)
from .browser import getBrowser, deleteBrowser, getBrowserByConnection
_LOGGER = logging.getLogger(__name__)
async def async_setup_connection(hass):
@websocket_api.websocket_command(
{
vol.Required("type"): WS_CONNECT,
vol.Required("browserID"): str,
}
)
@websocket_api.async_response
async def handle_connect(hass, connection, msg):
"""Connect to Browser Mod and subscribe to settings updates."""
browserID = msg[BROWSER_ID]
store = hass.data[DOMAIN][DATA_STORE]
@callback
def send_update(data):
connection.send_message(event_message(msg["id"], {"result": data}))
store_listener = store.add_listener(send_update)
def close_connection():
store_listener()
dev = getBrowser(hass, browserID, create=False)
if dev:
dev.close_connection(connection)
connection.subscriptions[msg["id"]] = close_connection
connection.send_result(msg["id"])
if store.get_browser(browserID).registered:
dev = getBrowser(hass, browserID)
dev.update_settings(hass, store.get_browser(browserID).asdict())
dev.open_connection(connection, msg["id"])
await store.set_browser(
browserID, last_seen=datetime.now(tz=timezone.utc).isoformat()
)
send_update(store.asdict())
@websocket_api.websocket_command(
{
vol.Required("type"): WS_REGISTER,
vol.Required("browserID"): str,
vol.Optional("data"): dict,
}
)
@websocket_api.async_response
async def handle_register(hass, connection, msg):
"""Register a Browser."""
browserID = msg[BROWSER_ID]
store = hass.data[DOMAIN][DATA_STORE]
browserSettings = {"registered": True}
data = msg.get("data", {})
if "last_seen" in data:
del data["last_seen"]
if BROWSER_ID in data:
# Change ID of registered browser
newBrowserID = data[BROWSER_ID]
del data[BROWSER_ID]
# Copy data from old browser and delete it from store
if oldBrowserSettings := store.get_browser(browserID):
browserSettings = oldBrowserSettings.asdict()
await store.delete_browser(browserID)
# Delete the old Browser device
deleteBrowser(hass, browserID)
# Use the new browserID from now on
browserID = newBrowserID
# Create and/or update Browser device
dev = getBrowser(hass, browserID)
dev.update_settings(hass, data)
# Create or update store data
if data is not None:
browserSettings.update(data)
await store.set_browser(browserID, **browserSettings)
@websocket_api.websocket_command(
{
vol.Required("type"): WS_UNREGISTER,
vol.Required("browserID"): str,
}
)
@websocket_api.async_response
async def handle_unregister(hass, connection, msg):
"""Unregister a Browser."""
browserID = msg[BROWSER_ID]
store = hass.data[DOMAIN][DATA_STORE]
deleteBrowser(hass, browserID)
await store.delete_browser(browserID)
connection.send_result(msg["id"])
@websocket_api.websocket_command(
{
vol.Required("type"): WS_UPDATE,
vol.Required("browserID"): str,
vol.Optional("data"): dict,
}
)
@websocket_api.async_response
async def handle_update(hass, connection, msg):
"""Receive state updates from a Browser."""
browserID = msg[BROWSER_ID]
store = hass.data[DOMAIN][DATA_STORE]
if store.get_browser(browserID).registered:
dev = getBrowser(hass, browserID)
dev.update(hass, msg.get("data", {}))
@websocket_api.websocket_command(
{
vol.Required("type"): WS_SETTINGS,
vol.Required("key"): str,
vol.Optional("value"): vol.Any(int, str, bool, list, object, None),
vol.Optional("user"): str,
}
)
@websocket_api.async_response
async def handle_settings(hass, connection, msg):
"""Change user or global settings."""
store = hass.data[DOMAIN][DATA_STORE]
if "user" in msg:
# Set user setting
await store.set_user_settings(
msg["user"], **{msg["key"]: msg.get("value", None)}
)
else:
# Set global setting
await store.set_global_settings(**{msg["key"]: msg.get("value", None)})
pass
@websocket_api.websocket_command(
{
vol.Required("type"): WS_RECALL_ID,
}
)
def handle_recall_id(hass, connection, msg):
"""Recall browserID of Browser with the current connection."""
dev = getBrowserByConnection(hass, connection)
if dev:
connection.send_message(
websocket_api.result_message(msg["id"], dev.browserID)
)
connection.send_message(websocket_api.result_message(msg["id"], None))
@websocket_api.websocket_command(
{
vol.Required("type"): WS_LOG,
vol.Required("message"): str,
}
)
def handle_log(hass, connection, msg):
"""Print a debug message."""
_LOGGER.info(f"LOG MESSAGE: {msg['message']}")
async_register_command(hass, handle_connect)
async_register_command(hass, handle_register)
async_register_command(hass, handle_unregister)
async_register_command(hass, handle_update)
async_register_command(hass, handle_settings)
async_register_command(hass, handle_recall_id)
async_register_command(hass, handle_log)