2018-10-01 14:09:31 +00:00
|
|
|
"""Connection session."""
|
2019-06-25 05:04:31 +00:00
|
|
|
import asyncio
|
2020-01-03 20:37:11 +00:00
|
|
|
from typing import Any, Callable, Dict, Hashable, Optional
|
2019-09-29 17:07:49 +00:00
|
|
|
|
2018-10-01 14:09:31 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2019-12-08 11:19:15 +00:00
|
|
|
from homeassistant.core import Context, callback
|
2020-10-25 20:10:13 +00:00
|
|
|
from homeassistant.exceptions import HomeAssistantError, Unauthorized
|
2018-10-01 14:09:31 +00:00
|
|
|
|
|
|
|
from . import const, messages
|
|
|
|
|
2019-09-29 17:07:49 +00:00
|
|
|
# mypy: allow-untyped-calls, allow-untyped-defs
|
|
|
|
|
|
|
|
|
2018-10-01 14:09:31 +00:00
|
|
|
class ActiveConnection:
|
|
|
|
"""Handle an active websocket client connection."""
|
|
|
|
|
|
|
|
def __init__(self, logger, hass, send_message, user, refresh_token):
|
|
|
|
"""Initialize an active connection."""
|
|
|
|
self.logger = logger
|
|
|
|
self.hass = hass
|
|
|
|
self.send_message = send_message
|
|
|
|
self.user = user
|
|
|
|
if refresh_token:
|
|
|
|
self.refresh_token_id = refresh_token.id
|
|
|
|
else:
|
|
|
|
self.refresh_token_id = None
|
|
|
|
|
2019-09-29 17:07:49 +00:00
|
|
|
self.subscriptions: Dict[Hashable, Callable[[], Any]] = {}
|
2018-10-01 14:09:31 +00:00
|
|
|
self.last_id = 0
|
|
|
|
|
|
|
|
def context(self, msg):
|
|
|
|
"""Return a context."""
|
|
|
|
user = self.user
|
|
|
|
if user is None:
|
|
|
|
return Context()
|
|
|
|
return Context(user_id=user.id)
|
|
|
|
|
2019-02-09 18:41:40 +00:00
|
|
|
@callback
|
2020-01-03 20:37:11 +00:00
|
|
|
def send_result(self, msg_id: int, result: Optional[Any] = None) -> None:
|
2019-02-09 18:41:40 +00:00
|
|
|
"""Send a result message."""
|
|
|
|
self.send_message(messages.result_message(msg_id, result))
|
|
|
|
|
2019-05-14 03:57:47 +00:00
|
|
|
async def send_big_result(self, msg_id, result):
|
|
|
|
"""Send a result message that would be expensive to JSON serialize."""
|
|
|
|
content = await self.hass.async_add_executor_job(
|
|
|
|
const.JSON_DUMP, messages.result_message(msg_id, result)
|
|
|
|
)
|
|
|
|
self.send_message(content)
|
|
|
|
|
2019-02-09 18:41:40 +00:00
|
|
|
@callback
|
2020-01-03 20:37:11 +00:00
|
|
|
def send_error(self, msg_id: int, code: str, message: str) -> None:
|
2019-02-09 18:41:40 +00:00
|
|
|
"""Send a error message."""
|
|
|
|
self.send_message(messages.error_message(msg_id, code, message))
|
|
|
|
|
2018-10-01 14:09:31 +00:00
|
|
|
@callback
|
|
|
|
def async_handle(self, msg):
|
|
|
|
"""Handle a single incoming message."""
|
|
|
|
handlers = self.hass.data[const.DOMAIN]
|
|
|
|
|
|
|
|
try:
|
|
|
|
msg = messages.MINIMAL_MESSAGE_SCHEMA(msg)
|
2019-07-31 19:25:30 +00:00
|
|
|
cur_id = msg["id"]
|
2018-10-01 14:09:31 +00:00
|
|
|
except vol.Invalid:
|
2019-07-31 19:25:30 +00:00
|
|
|
self.logger.error("Received invalid command", msg)
|
|
|
|
self.send_message(
|
|
|
|
messages.error_message(
|
|
|
|
msg.get("id"),
|
|
|
|
const.ERR_INVALID_FORMAT,
|
|
|
|
"Message incorrectly formatted.",
|
|
|
|
)
|
|
|
|
)
|
2018-10-01 14:09:31 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
if cur_id <= self.last_id:
|
2019-07-31 19:25:30 +00:00
|
|
|
self.send_message(
|
|
|
|
messages.error_message(
|
|
|
|
cur_id, const.ERR_ID_REUSE, "Identifier values have to increase."
|
|
|
|
)
|
|
|
|
)
|
2018-10-01 14:09:31 +00:00
|
|
|
return
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
if msg["type"] not in handlers:
|
|
|
|
self.logger.error("Received invalid command: {}".format(msg["type"]))
|
|
|
|
self.send_message(
|
|
|
|
messages.error_message(
|
|
|
|
cur_id, const.ERR_UNKNOWN_COMMAND, "Unknown command."
|
|
|
|
)
|
|
|
|
)
|
2018-10-01 14:09:31 +00:00
|
|
|
return
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
handler, schema = handlers[msg["type"]]
|
2018-10-01 14:09:31 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
handler(self.hass, self, schema(msg))
|
2018-11-27 09:12:31 +00:00
|
|
|
except Exception as err: # pylint: disable=broad-except
|
|
|
|
self.async_handle_exception(msg, err)
|
2018-10-01 14:09:31 +00:00
|
|
|
|
|
|
|
self.last_id = cur_id
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_close(self):
|
|
|
|
"""Close down connection."""
|
2019-03-11 03:07:09 +00:00
|
|
|
for unsub in self.subscriptions.values():
|
2018-10-01 14:09:31 +00:00
|
|
|
unsub()
|
2018-11-27 09:12:31 +00:00
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_handle_exception(self, msg, err):
|
|
|
|
"""Handle an exception while processing a handler."""
|
2019-11-08 09:06:16 +00:00
|
|
|
log_handler = self.logger.error
|
|
|
|
|
2018-11-27 09:12:31 +00:00
|
|
|
if isinstance(err, Unauthorized):
|
|
|
|
code = const.ERR_UNAUTHORIZED
|
2019-07-31 19:25:30 +00:00
|
|
|
err_message = "Unauthorized"
|
2018-11-27 09:12:31 +00:00
|
|
|
elif isinstance(err, vol.Invalid):
|
|
|
|
code = const.ERR_INVALID_FORMAT
|
2019-03-27 17:40:39 +00:00
|
|
|
err_message = vol.humanize.humanize_error(msg, err)
|
2019-06-25 05:04:31 +00:00
|
|
|
elif isinstance(err, asyncio.TimeoutError):
|
|
|
|
code = const.ERR_TIMEOUT
|
2019-07-31 19:25:30 +00:00
|
|
|
err_message = "Timeout"
|
2020-10-25 20:10:13 +00:00
|
|
|
elif isinstance(err, HomeAssistantError):
|
|
|
|
code = const.ERR_UNKNOWN_ERROR
|
|
|
|
err_message = str(err)
|
2018-11-27 09:12:31 +00:00
|
|
|
else:
|
|
|
|
code = const.ERR_UNKNOWN_ERROR
|
2019-07-31 19:25:30 +00:00
|
|
|
err_message = "Unknown error"
|
2019-11-08 09:06:16 +00:00
|
|
|
log_handler = self.logger.exception
|
|
|
|
|
|
|
|
log_handler("Error handling message: %s", err_message)
|
2018-11-27 09:12:31 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
self.send_message(messages.error_message(msg["id"], code, err_message))
|