2019-02-14 15:01:46 +00:00
|
|
|
"""HTTP Support for Hass.io."""
|
2021-03-18 08:25:40 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2021-07-29 21:26:05 +00:00
|
|
|
import asyncio
|
2021-10-01 16:27:44 +00:00
|
|
|
from http import HTTPStatus
|
2018-02-20 23:24:31 +00:00
|
|
|
import logging
|
|
|
|
import os
|
|
|
|
import re
|
|
|
|
|
|
|
|
import aiohttp
|
|
|
|
from aiohttp import web
|
2021-08-03 14:48:22 +00:00
|
|
|
from aiohttp.client import ClientTimeout
|
|
|
|
from aiohttp.hdrs import (
|
2022-06-21 15:11:20 +00:00
|
|
|
AUTHORIZATION,
|
2021-08-31 12:45:28 +00:00
|
|
|
CACHE_CONTROL,
|
2021-08-03 14:48:22 +00:00
|
|
|
CONTENT_ENCODING,
|
|
|
|
CONTENT_LENGTH,
|
|
|
|
CONTENT_TYPE,
|
|
|
|
TRANSFER_ENCODING,
|
|
|
|
)
|
2018-02-20 23:24:31 +00:00
|
|
|
from aiohttp.web_exceptions import HTTPBadGateway
|
2022-06-21 15:11:20 +00:00
|
|
|
from multidict import istr
|
2018-02-20 23:24:31 +00:00
|
|
|
|
|
|
|
from homeassistant.components.http import KEY_AUTHENTICATED, HomeAssistantView
|
2020-09-26 07:26:02 +00:00
|
|
|
from homeassistant.components.onboarding import async_is_onboarded
|
2018-02-20 23:24:31 +00:00
|
|
|
|
2022-06-21 15:11:20 +00:00
|
|
|
from .const import X_HASS_IS_ADMIN, X_HASS_USER_ID
|
2018-10-11 08:55:38 +00:00
|
|
|
|
2018-02-20 23:24:31 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2021-07-29 21:26:05 +00:00
|
|
|
MAX_UPLOAD_SIZE = 1024 * 1024 * 1024
|
|
|
|
|
2022-06-06 19:43:47 +00:00
|
|
|
# pylint: disable=implicit-str-concat
|
2018-09-19 10:57:55 +00:00
|
|
|
NO_TIMEOUT = re.compile(
|
2019-07-31 19:25:30 +00:00
|
|
|
r"^(?:"
|
|
|
|
r"|homeassistant/update"
|
|
|
|
r"|hassos/update"
|
|
|
|
r"|hassos/update/cli"
|
|
|
|
r"|supervisor/update"
|
|
|
|
r"|addons/[^/]+/(?:update|install|rebuild)"
|
2021-08-02 09:07:21 +00:00
|
|
|
r"|backups/.+/full"
|
|
|
|
r"|backups/.+/partial"
|
|
|
|
r"|backups/[^/]+/(?:upload|download)"
|
2019-07-31 19:25:30 +00:00
|
|
|
r")$"
|
2018-09-19 10:57:55 +00:00
|
|
|
)
|
|
|
|
|
2021-10-14 08:00:44 +00:00
|
|
|
NO_AUTH_ONBOARDING = re.compile(r"^(?:" r"|supervisor/logs" r"|backups/[^/]+/.+" r")$")
|
2020-09-28 08:10:10 +00:00
|
|
|
|
2020-01-17 21:54:28 +00:00
|
|
|
NO_AUTH = re.compile(
|
2022-06-10 13:41:42 +00:00
|
|
|
r"^(?:" r"|app/.*" r"|[store\/]*addons/[^/]+/(logo|dark_logo|icon|dark_icon)" r")$"
|
2020-01-17 21:54:28 +00:00
|
|
|
)
|
2018-02-20 23:24:31 +00:00
|
|
|
|
2021-08-31 12:45:28 +00:00
|
|
|
NO_STORE = re.compile(r"^(?:" r"|app/entrypoint.js" r")$")
|
2022-06-06 19:43:47 +00:00
|
|
|
# pylint: enable=implicit-str-concat
|
2021-08-31 12:45:28 +00:00
|
|
|
|
2018-02-20 23:24:31 +00:00
|
|
|
|
|
|
|
class HassIOView(HomeAssistantView):
|
|
|
|
"""Hass.io view to handle base part."""
|
|
|
|
|
|
|
|
name = "api:hassio"
|
|
|
|
url = "/api/hassio/{path:.+}"
|
|
|
|
requires_auth = False
|
|
|
|
|
2021-05-20 15:47:30 +00:00
|
|
|
def __init__(self, host: str, websession: aiohttp.ClientSession) -> None:
|
2018-02-20 23:24:31 +00:00
|
|
|
"""Initialize a Hass.io base view."""
|
|
|
|
self._host = host
|
|
|
|
self._websession = websession
|
|
|
|
|
2019-04-01 12:16:16 +00:00
|
|
|
async def _handle(
|
2019-07-31 19:25:30 +00:00
|
|
|
self, request: web.Request, path: str
|
2021-03-18 08:25:40 +00:00
|
|
|
) -> web.Response | web.StreamResponse:
|
2018-02-20 23:24:31 +00:00
|
|
|
"""Route data to Hass.io."""
|
2020-09-26 07:26:02 +00:00
|
|
|
hass = request.app["hass"]
|
|
|
|
if _need_auth(hass, path) and not request[KEY_AUTHENTICATED]:
|
2021-10-01 16:27:44 +00:00
|
|
|
return web.Response(status=HTTPStatus.UNAUTHORIZED)
|
2018-02-20 23:24:31 +00:00
|
|
|
|
2019-04-01 12:16:16 +00:00
|
|
|
return await self._command_proxy(path, request)
|
2018-02-20 23:24:31 +00:00
|
|
|
|
2020-10-08 18:40:45 +00:00
|
|
|
delete = _handle
|
2018-02-20 23:24:31 +00:00
|
|
|
get = _handle
|
|
|
|
post = _handle
|
|
|
|
|
2019-04-01 12:16:16 +00:00
|
|
|
async def _command_proxy(
|
2019-07-31 19:25:30 +00:00
|
|
|
self, path: str, request: web.Request
|
2021-08-03 14:48:22 +00:00
|
|
|
) -> web.StreamResponse:
|
2018-02-20 23:24:31 +00:00
|
|
|
"""Return a client request with proxy origin for Hass.io supervisor.
|
|
|
|
|
|
|
|
This method is a coroutine.
|
|
|
|
"""
|
2019-04-01 12:16:16 +00:00
|
|
|
headers = _init_header(request)
|
2021-10-14 08:00:44 +00:00
|
|
|
if path == "backups/new/upload":
|
2020-09-25 08:02:26 +00:00
|
|
|
# We need to reuse the full content type that includes the boundary
|
|
|
|
headers[
|
2022-06-21 15:11:20 +00:00
|
|
|
CONTENT_TYPE
|
2020-09-25 08:02:26 +00:00
|
|
|
] = request._stored_content_type # pylint: disable=protected-access
|
2021-07-29 21:26:05 +00:00
|
|
|
|
2018-02-20 23:24:31 +00:00
|
|
|
try:
|
2021-08-03 14:48:22 +00:00
|
|
|
client = await self._websession.request(
|
|
|
|
method=request.method,
|
|
|
|
url=f"http://{self._host}/{path}",
|
|
|
|
params=request.query,
|
|
|
|
data=request.content,
|
2019-07-31 19:25:30 +00:00
|
|
|
headers=headers,
|
2021-08-03 14:48:22 +00:00
|
|
|
timeout=_get_timeout(path),
|
2018-02-20 23:24:31 +00:00
|
|
|
)
|
2019-04-01 12:16:16 +00:00
|
|
|
|
2021-07-29 21:26:05 +00:00
|
|
|
# Stream response
|
2021-08-03 14:48:22 +00:00
|
|
|
response = web.StreamResponse(
|
2021-08-31 12:45:28 +00:00
|
|
|
status=client.status, headers=_response_header(client, path)
|
2021-08-03 14:48:22 +00:00
|
|
|
)
|
2019-04-01 12:16:16 +00:00
|
|
|
response.content_type = client.content_type
|
2018-02-20 23:24:31 +00:00
|
|
|
|
2019-04-01 12:16:16 +00:00
|
|
|
await response.prepare(request)
|
|
|
|
async for data in client.content.iter_chunked(4096):
|
|
|
|
await response.write(data)
|
|
|
|
|
|
|
|
return response
|
2018-02-20 23:24:31 +00:00
|
|
|
|
2021-07-29 21:26:05 +00:00
|
|
|
except aiohttp.ClientError as err:
|
2018-02-20 23:24:31 +00:00
|
|
|
_LOGGER.error("Client error on api %s request %s", path, err)
|
|
|
|
|
2021-07-29 21:26:05 +00:00
|
|
|
except asyncio.TimeoutError:
|
|
|
|
_LOGGER.error("Client timeout error on API request %s", path)
|
|
|
|
|
2018-02-20 23:24:31 +00:00
|
|
|
raise HTTPBadGateway()
|
|
|
|
|
|
|
|
|
2022-06-21 15:11:20 +00:00
|
|
|
def _init_header(request: web.Request) -> dict[istr, str]:
|
2019-04-01 12:16:16 +00:00
|
|
|
"""Create initial header."""
|
|
|
|
headers = {
|
2022-06-21 15:11:20 +00:00
|
|
|
AUTHORIZATION: f"Bearer {os.environ.get('SUPERVISOR_TOKEN', '')}",
|
2019-04-01 12:16:16 +00:00
|
|
|
CONTENT_TYPE: request.content_type,
|
|
|
|
}
|
|
|
|
|
|
|
|
# Add user data
|
2021-10-18 13:54:38 +00:00
|
|
|
if request.get("hass_user") is not None:
|
2022-06-21 15:11:20 +00:00
|
|
|
headers[istr(X_HASS_USER_ID)] = request["hass_user"].id
|
|
|
|
headers[istr(X_HASS_IS_ADMIN)] = str(int(request["hass_user"].is_admin))
|
2019-04-01 12:16:16 +00:00
|
|
|
|
|
|
|
return headers
|
2018-02-20 23:24:31 +00:00
|
|
|
|
|
|
|
|
2021-08-31 12:45:28 +00:00
|
|
|
def _response_header(response: aiohttp.ClientResponse, path: str) -> dict[str, str]:
|
2021-08-03 14:48:22 +00:00
|
|
|
"""Create response header."""
|
|
|
|
headers = {}
|
|
|
|
|
|
|
|
for name, value in response.headers.items():
|
|
|
|
if name in (
|
|
|
|
TRANSFER_ENCODING,
|
|
|
|
CONTENT_LENGTH,
|
|
|
|
CONTENT_TYPE,
|
|
|
|
CONTENT_ENCODING,
|
|
|
|
):
|
|
|
|
continue
|
|
|
|
headers[name] = value
|
|
|
|
|
2021-08-31 12:45:28 +00:00
|
|
|
if NO_STORE.match(path):
|
|
|
|
headers[CACHE_CONTROL] = "no-store, max-age=0"
|
|
|
|
|
2021-08-03 14:48:22 +00:00
|
|
|
return headers
|
|
|
|
|
|
|
|
|
|
|
|
def _get_timeout(path: str) -> ClientTimeout:
|
2018-02-20 23:24:31 +00:00
|
|
|
"""Return timeout for a URL path."""
|
2018-09-19 10:57:55 +00:00
|
|
|
if NO_TIMEOUT.match(path):
|
2021-08-03 14:48:22 +00:00
|
|
|
return ClientTimeout(connect=10, total=None)
|
|
|
|
return ClientTimeout(connect=10, total=300)
|
2018-02-20 23:24:31 +00:00
|
|
|
|
|
|
|
|
2020-09-26 07:26:02 +00:00
|
|
|
def _need_auth(hass, path: str) -> bool:
|
2018-02-20 23:24:31 +00:00
|
|
|
"""Return if a path need authentication."""
|
2020-09-28 08:10:10 +00:00
|
|
|
if not async_is_onboarded(hass) and NO_AUTH_ONBOARDING.match(path):
|
2020-09-26 07:26:02 +00:00
|
|
|
return False
|
2018-09-19 10:57:55 +00:00
|
|
|
if NO_AUTH.match(path):
|
|
|
|
return False
|
2018-02-20 23:24:31 +00:00
|
|
|
return True
|