mirror of
https://github.com/home-assistant/supervisor.git
synced 2025-06-22 09:56:30 +00:00

* Improve gdbus error handling * Fix logging type * Detect no dbus * Fix issue with complex * Update hassio/dbus/__init__.py Co-Authored-By: Franck Nijhof <frenck@frenck.nl> * Update hassio/dbus/hostname.py Co-Authored-By: Franck Nijhof <frenck@frenck.nl> * Update hassio/dbus/rauc.py Co-Authored-By: Franck Nijhof <frenck@frenck.nl> * Update hassio/dbus/systemd.py Co-Authored-By: Franck Nijhof <frenck@frenck.nl> * Fix black
254 lines
8.8 KiB
Python
254 lines
8.8 KiB
Python
"""Utils for Home Assistant Proxy."""
|
|
import asyncio
|
|
from contextlib import asynccontextmanager
|
|
import logging
|
|
|
|
import aiohttp
|
|
from aiohttp import web
|
|
from aiohttp.web_exceptions import HTTPBadGateway, HTTPUnauthorized
|
|
from aiohttp.client_exceptions import ClientConnectorError
|
|
from aiohttp.hdrs import CONTENT_TYPE, AUTHORIZATION
|
|
import async_timeout
|
|
|
|
from ..const import HEADER_HA_ACCESS
|
|
from ..coresys import CoreSysAttributes
|
|
from ..exceptions import HomeAssistantAuthError, HomeAssistantAPIError, APIError
|
|
|
|
_LOGGER: logging.Logger = logging.getLogger(__name__)
|
|
|
|
|
|
class APIProxy(CoreSysAttributes):
|
|
"""API Proxy for Home Assistant."""
|
|
|
|
def _check_access(self, request: web.Request):
|
|
"""Check the Hass.io token."""
|
|
if AUTHORIZATION in request.headers:
|
|
bearer = request.headers[AUTHORIZATION]
|
|
hassio_token = bearer.split(" ")[-1]
|
|
else:
|
|
hassio_token = request.headers.get(HEADER_HA_ACCESS)
|
|
|
|
addon = self.sys_addons.from_token(hassio_token)
|
|
if not addon:
|
|
_LOGGER.warning("Unknown Home Assistant API access!")
|
|
elif not addon.access_homeassistant_api:
|
|
_LOGGER.warning("Not permitted API access: %s", addon.slug)
|
|
else:
|
|
_LOGGER.debug("%s access from %s", request.path, addon.slug)
|
|
return
|
|
|
|
raise HTTPUnauthorized()
|
|
|
|
@asynccontextmanager
|
|
async def _api_client(self, request: web.Request, path: str, timeout: int = 300):
|
|
"""Return a client request with proxy origin for Home Assistant."""
|
|
try:
|
|
# read data
|
|
with async_timeout.timeout(30):
|
|
data = await request.read()
|
|
|
|
if data:
|
|
content_type = request.content_type
|
|
else:
|
|
content_type = None
|
|
|
|
async with self.sys_homeassistant.make_request(
|
|
request.method.lower(),
|
|
f"api/{path}",
|
|
content_type=content_type,
|
|
data=data,
|
|
timeout=timeout,
|
|
params=request.query,
|
|
) as resp:
|
|
yield resp
|
|
return
|
|
|
|
except HomeAssistantAuthError:
|
|
_LOGGER.error("Authenticate error on API for request %s", path)
|
|
except HomeAssistantAPIError:
|
|
_LOGGER.error("Error on API for request %s", path)
|
|
except aiohttp.ClientError as err:
|
|
_LOGGER.error("Client error on API %s request %s", path, err)
|
|
except asyncio.TimeoutError:
|
|
_LOGGER.error("Client timeout error on API request %s", path)
|
|
|
|
raise HTTPBadGateway()
|
|
|
|
async def stream(self, request: web.Request):
|
|
"""Proxy HomeAssistant EventStream Requests."""
|
|
self._check_access(request)
|
|
|
|
_LOGGER.info("Home Assistant EventStream start")
|
|
async with self._api_client(request, "stream", timeout=None) as client:
|
|
response = web.StreamResponse()
|
|
response.content_type = request.headers.get(CONTENT_TYPE)
|
|
try:
|
|
await response.prepare(request)
|
|
async for data in client.content:
|
|
await response.write(data)
|
|
|
|
except (aiohttp.ClientError, aiohttp.ClientPayloadError):
|
|
pass
|
|
|
|
_LOGGER.info("Home Assistant EventStream close")
|
|
return response
|
|
|
|
async def api(self, request: web.Request):
|
|
"""Proxy Home Assistant API Requests."""
|
|
self._check_access(request)
|
|
|
|
# Normal request
|
|
path = request.match_info.get("path", "")
|
|
async with self._api_client(request, path) as client:
|
|
data = await client.read()
|
|
return web.Response(
|
|
body=data, status=client.status, content_type=client.content_type
|
|
)
|
|
|
|
async def _websocket_client(self):
|
|
"""Initialize a WebSocket API connection."""
|
|
url = f"{self.sys_homeassistant.api_url}/api/websocket"
|
|
|
|
try:
|
|
client = await self.sys_websession_ssl.ws_connect(
|
|
url, heartbeat=30, verify_ssl=False
|
|
)
|
|
|
|
# Handle authentication
|
|
data = await client.receive_json()
|
|
|
|
if data.get("type") == "auth_ok":
|
|
return client
|
|
|
|
if data.get("type") != "auth_required":
|
|
# Invalid protocol
|
|
_LOGGER.error("Got unexpected response from HA WebSocket: %s", data)
|
|
raise APIError()
|
|
|
|
if self.sys_homeassistant.refresh_token:
|
|
await self.sys_homeassistant.ensure_access_token()
|
|
await client.send_json(
|
|
{
|
|
"type": "auth",
|
|
"access_token": self.sys_homeassistant.access_token,
|
|
}
|
|
)
|
|
else:
|
|
await client.send_json(
|
|
{
|
|
"type": "auth",
|
|
"api_password": self.sys_homeassistant.api_password,
|
|
}
|
|
)
|
|
|
|
data = await client.receive_json()
|
|
|
|
if data.get("type") == "auth_ok":
|
|
return client
|
|
|
|
# Renew the Token is invalid
|
|
if (
|
|
data.get("type") == "invalid_auth"
|
|
and self.sys_homeassistant.refresh_token
|
|
):
|
|
self.sys_homeassistant.access_token = None
|
|
return await self._websocket_client()
|
|
|
|
raise HomeAssistantAuthError()
|
|
|
|
except (RuntimeError, ValueError, ClientConnectorError) as err:
|
|
_LOGGER.error("Client error on WebSocket API %s.", err)
|
|
except HomeAssistantAuthError:
|
|
_LOGGER.error("Failed authentication to Home Assistant WebSocket")
|
|
|
|
raise APIError()
|
|
|
|
async def websocket(self, request: web.Request):
|
|
"""Initialize a WebSocket API connection."""
|
|
_LOGGER.info("Home Assistant WebSocket API request initialize")
|
|
|
|
# init server
|
|
server = web.WebSocketResponse(heartbeat=30)
|
|
await server.prepare(request)
|
|
|
|
# handle authentication
|
|
try:
|
|
await server.send_json(
|
|
{"type": "auth_required", "ha_version": self.sys_homeassistant.version}
|
|
)
|
|
|
|
# Check API access
|
|
response = await server.receive_json()
|
|
hassio_token = response.get("api_password") or response.get("access_token")
|
|
addon = self.sys_addons.from_token(hassio_token)
|
|
|
|
if not addon or not addon.access_homeassistant_api:
|
|
_LOGGER.warning("Unauthorized WebSocket access!")
|
|
await server.send_json(
|
|
{"type": "auth_invalid", "message": "Invalid access"}
|
|
)
|
|
return server
|
|
|
|
_LOGGER.info("WebSocket access from %s", addon.slug)
|
|
|
|
await server.send_json(
|
|
{"type": "auth_ok", "ha_version": self.sys_homeassistant.version}
|
|
)
|
|
except (RuntimeError, ValueError) as err:
|
|
_LOGGER.error("Can't initialize handshake: %s", err)
|
|
return server
|
|
|
|
# init connection to hass
|
|
try:
|
|
client = await self._websocket_client()
|
|
except APIError:
|
|
return server
|
|
|
|
_LOGGER.info("Home Assistant WebSocket API request running")
|
|
try:
|
|
client_read = None
|
|
server_read = None
|
|
while not server.closed and not client.closed:
|
|
if not client_read:
|
|
client_read = self.sys_create_task(client.receive_str())
|
|
if not server_read:
|
|
server_read = self.sys_create_task(server.receive_str())
|
|
|
|
# wait until data need to be processed
|
|
await asyncio.wait(
|
|
[client_read, server_read], return_when=asyncio.FIRST_COMPLETED
|
|
)
|
|
|
|
# server
|
|
if server_read.done() and not client.closed:
|
|
server_read.exception()
|
|
await client.send_str(server_read.result())
|
|
server_read = None
|
|
|
|
# client
|
|
if client_read.done() and not server.closed:
|
|
client_read.exception()
|
|
await server.send_str(client_read.result())
|
|
client_read = None
|
|
|
|
except asyncio.CancelledError:
|
|
pass
|
|
|
|
except (RuntimeError, ConnectionError, TypeError) as err:
|
|
_LOGGER.info("Home Assistant WebSocket API error: %s", err)
|
|
|
|
finally:
|
|
if client_read:
|
|
client_read.cancel()
|
|
if server_read:
|
|
server_read.cancel()
|
|
|
|
# close connections
|
|
if not client.closed:
|
|
await client.close()
|
|
if not server.closed:
|
|
await server.close()
|
|
|
|
_LOGGER.info("Home Assistant WebSocket API connection is closed")
|
|
return server
|