2019-02-13 20:21:14 +00:00
|
|
|
"""Handle the frontend for Home Assistant."""
|
2021-03-17 22:49:01 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2021-06-24 14:01:28 +00:00
|
|
|
from collections.abc import Iterator
|
2021-04-19 04:17:30 +00:00
|
|
|
from functools import lru_cache
|
2016-10-24 06:48:01 +00:00
|
|
|
import json
|
2016-07-17 05:32:25 +00:00
|
|
|
import logging
|
2015-01-30 07:56:04 +00:00
|
|
|
import os
|
2019-05-02 20:59:24 +00:00
|
|
|
import pathlib
|
2022-02-26 23:23:56 +00:00
|
|
|
from typing import Any, TypedDict
|
2015-01-30 07:56:04 +00:00
|
|
|
|
2019-10-20 22:04:56 +00:00
|
|
|
from aiohttp import hdrs, web, web_urldispatcher
|
2017-11-11 23:22:05 +00:00
|
|
|
import jinja2
|
2019-10-20 22:04:56 +00:00
|
|
|
import voluptuous as vol
|
2019-05-31 18:27:05 +00:00
|
|
|
from yarl import URL
|
2016-10-24 06:48:01 +00:00
|
|
|
|
2022-01-14 15:29:48 +00:00
|
|
|
from homeassistant.components import onboarding, websocket_api
|
2019-10-20 22:04:56 +00:00
|
|
|
from homeassistant.components.http.view import HomeAssistantView
|
2021-06-24 14:01:28 +00:00
|
|
|
from homeassistant.components.websocket_api.connection import ActiveConnection
|
2020-01-14 21:03:02 +00:00
|
|
|
from homeassistant.config import async_hass_config_yaml
|
2021-02-08 10:36:45 +00:00
|
|
|
from homeassistant.const import CONF_MODE, CONF_NAME, EVENT_THEMES_UPDATED
|
2021-06-24 14:01:28 +00:00
|
|
|
from homeassistant.core import HomeAssistant, ServiceCall, callback
|
2020-02-10 03:47:16 +00:00
|
|
|
from homeassistant.helpers import service
|
2019-10-20 22:04:56 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2018-03-01 03:31:38 +00:00
|
|
|
from homeassistant.helpers.translation import async_get_translations
|
2021-06-24 14:01:28 +00:00
|
|
|
from homeassistant.helpers.typing import ConfigType
|
2020-04-26 00:30:15 +00:00
|
|
|
from homeassistant.loader import async_get_integration, bind_hass
|
2017-11-01 11:11:32 +00:00
|
|
|
|
2019-02-12 15:38:19 +00:00
|
|
|
from .storage import async_setup_frontend_storage
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DOMAIN = "frontend"
|
|
|
|
CONF_THEMES = "themes"
|
2021-05-25 11:26:24 +00:00
|
|
|
CONF_THEMES_MODES = "modes"
|
|
|
|
CONF_THEMES_LIGHT = "light"
|
|
|
|
CONF_THEMES_DARK = "dark"
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_EXTRA_HTML_URL = "extra_html_url"
|
|
|
|
CONF_EXTRA_HTML_URL_ES5 = "extra_html_url_es5"
|
|
|
|
CONF_EXTRA_MODULE_URL = "extra_module_url"
|
|
|
|
CONF_EXTRA_JS_URL_ES5 = "extra_js_url_es5"
|
|
|
|
CONF_FRONTEND_REPO = "development_repo"
|
|
|
|
CONF_JS_VERSION = "javascript_version"
|
|
|
|
EVENT_PANELS_UPDATED = "panels_updated"
|
|
|
|
|
|
|
|
DEFAULT_THEME_COLOR = "#03A9F4"
|
2017-11-01 11:11:32 +00:00
|
|
|
|
2016-07-20 06:36:46 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DATA_PANELS = "frontend_panels"
|
|
|
|
DATA_JS_VERSION = "frontend_js_version"
|
|
|
|
DATA_EXTRA_MODULE_URL = "frontend_extra_module_url"
|
|
|
|
DATA_EXTRA_JS_URL_ES5 = "frontend_extra_js_url_es5"
|
2020-08-05 15:42:23 +00:00
|
|
|
|
|
|
|
THEMES_STORAGE_KEY = f"{DOMAIN}_theme"
|
|
|
|
THEMES_STORAGE_VERSION = 1
|
|
|
|
THEMES_SAVE_DELAY = 60
|
|
|
|
DATA_THEMES_STORE = "frontend_themes_store"
|
2019-07-31 19:25:30 +00:00
|
|
|
DATA_THEMES = "frontend_themes"
|
|
|
|
DATA_DEFAULT_THEME = "frontend_default_theme"
|
2020-08-05 15:42:23 +00:00
|
|
|
DATA_DEFAULT_DARK_THEME = "frontend_default_dark_theme"
|
2019-07-31 19:25:30 +00:00
|
|
|
DEFAULT_THEME = "default"
|
2020-08-05 15:42:23 +00:00
|
|
|
VALUE_NO_THEME = "none"
|
2019-07-31 19:25:30 +00:00
|
|
|
|
|
|
|
PRIMARY_COLOR = "primary-color"
|
2016-11-25 05:37:56 +00:00
|
|
|
|
2016-07-17 05:32:25 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2021-05-25 11:26:24 +00:00
|
|
|
EXTENDED_THEME_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
# Theme variables that apply to all modes
|
|
|
|
cv.string: cv.string,
|
|
|
|
# Mode specific theme variables
|
|
|
|
vol.Optional(CONF_THEMES_MODES): vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Optional(CONF_THEMES_LIGHT): vol.Schema({cv.string: cv.string}),
|
|
|
|
vol.Optional(CONF_THEMES_DARK): vol.Schema({cv.string: cv.string}),
|
|
|
|
}
|
|
|
|
),
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
THEME_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
cv.string: (
|
|
|
|
vol.Any(
|
|
|
|
# Legacy theme scheme
|
|
|
|
{cv.string: cv.string},
|
|
|
|
# New extended schema with mode support
|
|
|
|
EXTENDED_THEME_SCHEMA,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONFIG_SCHEMA = vol.Schema(
|
|
|
|
{
|
2020-09-08 13:42:50 +00:00
|
|
|
DOMAIN: vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Optional(CONF_FRONTEND_REPO): cv.isdir,
|
2021-05-25 11:26:24 +00:00
|
|
|
vol.Optional(CONF_THEMES): THEME_SCHEMA,
|
2020-09-08 13:42:50 +00:00
|
|
|
vol.Optional(CONF_EXTRA_MODULE_URL): vol.All(
|
|
|
|
cv.ensure_list, [cv.string]
|
|
|
|
),
|
|
|
|
vol.Optional(CONF_EXTRA_JS_URL_ES5): vol.All(
|
|
|
|
cv.ensure_list, [cv.string]
|
|
|
|
),
|
|
|
|
# We no longer use these options.
|
|
|
|
vol.Optional(CONF_EXTRA_HTML_URL): cv.match_all,
|
|
|
|
vol.Optional(CONF_EXTRA_HTML_URL_ES5): cv.match_all,
|
|
|
|
vol.Optional(CONF_JS_VERSION): cv.match_all,
|
|
|
|
},
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
},
|
|
|
|
extra=vol.ALLOW_EXTRA,
|
|
|
|
)
|
|
|
|
|
|
|
|
SERVICE_SET_THEME = "set_theme"
|
|
|
|
SERVICE_RELOAD_THEMES = "reload_themes"
|
2017-07-13 01:08:13 +00:00
|
|
|
|
2016-07-17 05:32:25 +00:00
|
|
|
|
2021-04-19 04:17:30 +00:00
|
|
|
class Manifest:
|
|
|
|
"""Manage the manifest.json contents."""
|
|
|
|
|
|
|
|
def __init__(self, data: dict) -> None:
|
|
|
|
"""Init the manifest manager."""
|
|
|
|
self.manifest = data
|
|
|
|
self._serialize()
|
|
|
|
|
|
|
|
def __getitem__(self, key: str) -> Any:
|
|
|
|
"""Return an item in the manifest."""
|
|
|
|
return self.manifest[key]
|
|
|
|
|
|
|
|
@property
|
|
|
|
def json(self) -> str:
|
|
|
|
"""Return the serialized manifest."""
|
|
|
|
return self._serialized
|
|
|
|
|
|
|
|
def _serialize(self) -> None:
|
|
|
|
self._serialized = json.dumps(self.manifest, sort_keys=True)
|
|
|
|
|
|
|
|
def update_key(self, key: str, val: str) -> None:
|
|
|
|
"""Add a keyval to the manifest.json."""
|
|
|
|
self.manifest[key] = val
|
|
|
|
self._serialize()
|
|
|
|
|
|
|
|
|
|
|
|
MANIFEST_JSON = Manifest(
|
|
|
|
{
|
|
|
|
"background_color": "#FFFFFF",
|
|
|
|
"description": "Home automation platform that puts local control and privacy first.",
|
|
|
|
"dir": "ltr",
|
|
|
|
"display": "standalone",
|
|
|
|
"icons": [
|
|
|
|
{
|
|
|
|
"src": f"/static/icons/favicon-{size}x{size}.png",
|
|
|
|
"sizes": f"{size}x{size}",
|
|
|
|
"type": "image/png",
|
|
|
|
"purpose": "maskable any",
|
|
|
|
}
|
|
|
|
for size in (192, 384, 512, 1024)
|
|
|
|
],
|
|
|
|
"screenshots": [
|
|
|
|
{
|
|
|
|
"src": "/static/images/screenshots/screenshot-1.png",
|
|
|
|
"sizes": "413x792",
|
|
|
|
"type": "image/png",
|
|
|
|
}
|
|
|
|
],
|
|
|
|
"lang": "en-US",
|
|
|
|
"name": "Home Assistant",
|
|
|
|
"short_name": "Assistant",
|
|
|
|
"start_url": "/?homescreen=1",
|
|
|
|
"theme_color": DEFAULT_THEME_COLOR,
|
|
|
|
"prefer_related_applications": True,
|
|
|
|
"related_applications": [
|
|
|
|
{"platform": "play", "id": "io.homeassistant.companion.android"}
|
|
|
|
],
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class UrlManager:
|
|
|
|
"""Manage urls to be used on the frontend.
|
|
|
|
|
|
|
|
This is abstracted into a class because
|
|
|
|
some integrations add a remove these directly
|
|
|
|
on hass.data
|
|
|
|
"""
|
|
|
|
|
2021-06-24 14:01:28 +00:00
|
|
|
def __init__(self, urls: list[str]) -> None:
|
2021-04-19 04:17:30 +00:00
|
|
|
"""Init the url manager."""
|
|
|
|
self.urls = frozenset(urls)
|
|
|
|
|
2021-06-24 14:01:28 +00:00
|
|
|
def add(self, url: str) -> None:
|
2021-04-19 04:17:30 +00:00
|
|
|
"""Add a url to the set."""
|
|
|
|
self.urls = frozenset([*self.urls, url])
|
|
|
|
|
2021-06-24 14:01:28 +00:00
|
|
|
def remove(self, url: str) -> None:
|
2021-04-19 04:17:30 +00:00
|
|
|
"""Remove a url from the set."""
|
|
|
|
self.urls = self.urls - {url}
|
|
|
|
|
|
|
|
|
2018-06-05 14:50:16 +00:00
|
|
|
class Panel:
|
2017-10-25 02:36:27 +00:00
|
|
|
"""Abstract class for panels."""
|
|
|
|
|
|
|
|
# Name of the webcomponent
|
2021-06-24 14:01:28 +00:00
|
|
|
component_name: str
|
2017-10-25 02:36:27 +00:00
|
|
|
|
2019-09-29 17:07:49 +00:00
|
|
|
# Icon to show in the sidebar
|
2021-03-17 22:49:01 +00:00
|
|
|
sidebar_icon: str | None = None
|
2017-10-25 02:36:27 +00:00
|
|
|
|
2019-09-29 17:07:49 +00:00
|
|
|
# Title to show in the sidebar
|
2021-03-17 22:49:01 +00:00
|
|
|
sidebar_title: str | None = None
|
2017-10-25 02:36:27 +00:00
|
|
|
|
|
|
|
# Url to show the panel in the frontend
|
2021-03-17 22:49:01 +00:00
|
|
|
frontend_url_path: str | None = None
|
2017-10-25 02:36:27 +00:00
|
|
|
|
|
|
|
# Config to pass to the webcomponent
|
2021-03-17 22:49:01 +00:00
|
|
|
config: dict[str, Any] | None = None
|
2017-10-25 02:36:27 +00:00
|
|
|
|
2019-03-25 17:04:35 +00:00
|
|
|
# If the panel should only be visible to admins
|
|
|
|
require_admin = False
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
def __init__(
|
|
|
|
self,
|
2021-06-24 14:01:28 +00:00
|
|
|
component_name: str,
|
|
|
|
sidebar_title: str | None,
|
|
|
|
sidebar_icon: str | None,
|
|
|
|
frontend_url_path: str | None,
|
|
|
|
config: dict[str, Any] | None,
|
|
|
|
require_admin: bool,
|
|
|
|
) -> None:
|
2018-06-05 14:50:16 +00:00
|
|
|
"""Initialize a built-in panel."""
|
|
|
|
self.component_name = component_name
|
|
|
|
self.sidebar_title = sidebar_title
|
|
|
|
self.sidebar_icon = sidebar_icon
|
|
|
|
self.frontend_url_path = frontend_url_path or component_name
|
|
|
|
self.config = config
|
2019-03-25 17:04:35 +00:00
|
|
|
self.require_admin = require_admin
|
2017-10-25 02:36:27 +00:00
|
|
|
|
2018-06-05 14:50:16 +00:00
|
|
|
@callback
|
2021-06-24 14:01:28 +00:00
|
|
|
def to_response(self) -> PanelRespons:
|
2018-05-15 18:47:46 +00:00
|
|
|
"""Panel as dictionary."""
|
|
|
|
return {
|
2019-07-31 19:25:30 +00:00
|
|
|
"component_name": self.component_name,
|
|
|
|
"icon": self.sidebar_icon,
|
|
|
|
"title": self.sidebar_title,
|
|
|
|
"config": self.config,
|
|
|
|
"url_path": self.frontend_url_path,
|
|
|
|
"require_admin": self.require_admin,
|
2018-05-15 18:47:46 +00:00
|
|
|
}
|
2017-10-25 02:36:27 +00:00
|
|
|
|
|
|
|
|
2017-07-22 04:38:53 +00:00
|
|
|
@bind_hass
|
2019-05-30 11:37:01 +00:00
|
|
|
@callback
|
2019-07-31 19:25:30 +00:00
|
|
|
def async_register_built_in_panel(
|
2021-06-24 14:01:28 +00:00
|
|
|
hass: HomeAssistant,
|
|
|
|
component_name: str,
|
|
|
|
sidebar_title: str | None = None,
|
|
|
|
sidebar_icon: str | None = None,
|
|
|
|
frontend_url_path: str | None = None,
|
|
|
|
config: dict[str, Any] | None = None,
|
|
|
|
require_admin: bool = False,
|
2020-02-28 20:43:17 +00:00
|
|
|
*,
|
2021-06-24 14:01:28 +00:00
|
|
|
update: bool = False,
|
|
|
|
) -> None:
|
2016-07-17 05:32:25 +00:00
|
|
|
"""Register a built-in panel."""
|
2019-07-31 19:25:30 +00:00
|
|
|
panel = Panel(
|
|
|
|
component_name,
|
|
|
|
sidebar_title,
|
|
|
|
sidebar_icon,
|
|
|
|
frontend_url_path,
|
|
|
|
config,
|
|
|
|
require_admin,
|
|
|
|
)
|
2016-07-17 05:32:25 +00:00
|
|
|
|
2019-05-30 11:37:01 +00:00
|
|
|
panels = hass.data.setdefault(DATA_PANELS, {})
|
2016-07-17 05:32:25 +00:00
|
|
|
|
2020-02-28 20:43:17 +00:00
|
|
|
if not update and panel.frontend_url_path in panels:
|
2020-02-25 19:18:21 +00:00
|
|
|
raise ValueError(f"Overwriting panel {panel.frontend_url_path}")
|
2018-06-05 14:50:16 +00:00
|
|
|
|
|
|
|
panels[panel.frontend_url_path] = panel
|
2016-07-17 05:32:25 +00:00
|
|
|
|
2019-05-30 11:37:01 +00:00
|
|
|
hass.bus.async_fire(EVENT_PANELS_UPDATED)
|
|
|
|
|
|
|
|
|
|
|
|
@bind_hass
|
|
|
|
@callback
|
2021-06-24 14:01:28 +00:00
|
|
|
def async_remove_panel(hass: HomeAssistant, frontend_url_path: str) -> None:
|
2019-05-30 11:37:01 +00:00
|
|
|
"""Remove a built-in panel."""
|
|
|
|
panel = hass.data.get(DATA_PANELS, {}).pop(frontend_url_path, None)
|
|
|
|
|
|
|
|
if panel is None:
|
|
|
|
_LOGGER.warning("Removing unknown panel %s", frontend_url_path)
|
|
|
|
|
|
|
|
hass.bus.async_fire(EVENT_PANELS_UPDATED)
|
|
|
|
|
2015-06-24 06:22:32 +00:00
|
|
|
|
2021-06-24 14:01:28 +00:00
|
|
|
def add_extra_js_url(hass: HomeAssistant, url: str, es5: bool = False) -> None:
|
2019-06-21 20:16:28 +00:00
|
|
|
"""Register extra js or module url to load."""
|
|
|
|
key = DATA_EXTRA_JS_URL_ES5 if es5 else DATA_EXTRA_MODULE_URL
|
2021-04-19 04:17:30 +00:00
|
|
|
hass.data[key].add(url)
|
2019-06-21 20:16:28 +00:00
|
|
|
|
|
|
|
|
2021-06-24 14:01:28 +00:00
|
|
|
def add_manifest_json_key(key: str, val: Any) -> None:
|
2016-08-14 08:10:07 +00:00
|
|
|
"""Add a keyval to the manifest.json."""
|
2021-04-19 04:17:30 +00:00
|
|
|
MANIFEST_JSON.update_key(key, val)
|
2016-08-14 08:10:07 +00:00
|
|
|
|
|
|
|
|
2021-06-24 14:01:28 +00:00
|
|
|
def _frontend_root(dev_repo_path: str | None) -> pathlib.Path:
|
2019-05-02 20:59:24 +00:00
|
|
|
"""Return root path to the frontend files."""
|
|
|
|
if dev_repo_path is not None:
|
2019-07-31 19:25:30 +00:00
|
|
|
return pathlib.Path(dev_repo_path) / "hass_frontend"
|
2019-10-25 16:02:40 +00:00
|
|
|
# Keep import here so that we can import frontend without installing reqs
|
2019-12-05 12:44:59 +00:00
|
|
|
# pylint: disable=import-outside-toplevel
|
2019-10-25 16:02:40 +00:00
|
|
|
import hass_frontend
|
2019-05-02 20:59:24 +00:00
|
|
|
|
2022-02-26 23:23:56 +00:00
|
|
|
return hass_frontend.where()
|
2019-05-02 20:59:24 +00:00
|
|
|
|
|
|
|
|
2021-06-24 14:01:28 +00:00
|
|
|
async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool:
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Set up the serving of the frontend."""
|
2019-02-12 15:38:19 +00:00
|
|
|
await async_setup_frontend_storage(hass)
|
2022-01-11 16:26:25 +00:00
|
|
|
websocket_api.async_register_command(hass, websocket_get_panels)
|
|
|
|
websocket_api.async_register_command(hass, websocket_get_themes)
|
|
|
|
websocket_api.async_register_command(hass, websocket_get_translations)
|
|
|
|
websocket_api.async_register_command(hass, websocket_get_version)
|
2021-06-24 14:01:28 +00:00
|
|
|
hass.http.register_view(ManifestJSONView())
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2017-10-25 02:36:27 +00:00
|
|
|
conf = config.get(DOMAIN, {})
|
|
|
|
|
2020-09-08 13:42:50 +00:00
|
|
|
for key in (CONF_EXTRA_HTML_URL, CONF_EXTRA_HTML_URL_ES5, CONF_JS_VERSION):
|
|
|
|
if key in conf:
|
|
|
|
_LOGGER.error(
|
|
|
|
"Please remove %s from your frontend config. It is no longer supported",
|
|
|
|
key,
|
|
|
|
)
|
|
|
|
|
2017-10-25 02:36:27 +00:00
|
|
|
repo_path = conf.get(CONF_FRONTEND_REPO)
|
|
|
|
is_dev = repo_path is not None
|
2019-05-02 20:59:24 +00:00
|
|
|
root_path = _frontend_root(repo_path)
|
|
|
|
|
|
|
|
for path, should_cache in (
|
2019-07-31 19:25:30 +00:00
|
|
|
("service_worker.js", False),
|
|
|
|
("robots.txt", False),
|
2021-01-29 23:05:06 +00:00
|
|
|
("onboarding.html", not is_dev),
|
|
|
|
("static", not is_dev),
|
|
|
|
("frontend_latest", not is_dev),
|
|
|
|
("frontend_es5", not is_dev),
|
2019-05-02 20:59:24 +00:00
|
|
|
):
|
2019-08-23 16:53:33 +00:00
|
|
|
hass.http.register_static_path(f"/{path}", str(root_path / path), should_cache)
|
2017-10-25 02:36:27 +00:00
|
|
|
|
2017-11-11 07:02:06 +00:00
|
|
|
hass.http.register_static_path(
|
2019-07-31 19:25:30 +00:00
|
|
|
"/auth/authorize", str(root_path / "authorize.html"), False
|
|
|
|
)
|
2020-09-03 16:13:33 +00:00
|
|
|
# https://wicg.github.io/change-password-url/
|
|
|
|
hass.http.register_redirect(
|
|
|
|
"/.well-known/change-password", "/profile", redirect_exc=web.HTTPFound
|
|
|
|
)
|
2016-10-24 06:48:01 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
local = hass.config.path("www")
|
2016-10-24 06:48:01 +00:00
|
|
|
if os.path.isdir(local):
|
2017-11-01 12:07:16 +00:00
|
|
|
hass.http.register_static_path("/local", local, not is_dev)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2019-05-31 18:27:05 +00:00
|
|
|
hass.http.app.router.register_resource(IndexView(repo_path, hass))
|
2016-11-25 05:37:56 +00:00
|
|
|
|
2020-03-13 18:55:53 +00:00
|
|
|
async_register_built_in_panel(hass, "profile")
|
2017-08-27 16:07:58 +00:00
|
|
|
|
2022-04-28 21:09:54 +00:00
|
|
|
# Can be removed in 2023
|
|
|
|
hass.http.register_redirect("/config/server_control", "/developer-tools/yaml")
|
2019-06-28 15:34:53 +00:00
|
|
|
|
|
|
|
async_register_built_in_panel(
|
2019-07-31 19:25:30 +00:00
|
|
|
hass,
|
|
|
|
"developer-tools",
|
|
|
|
require_admin=True,
|
|
|
|
sidebar_title="developer_tools",
|
|
|
|
sidebar_icon="hass:hammer",
|
|
|
|
)
|
2017-07-13 01:08:13 +00:00
|
|
|
|
2021-04-19 04:17:30 +00:00
|
|
|
hass.data[DATA_EXTRA_MODULE_URL] = UrlManager(conf.get(CONF_EXTRA_MODULE_URL, []))
|
|
|
|
hass.data[DATA_EXTRA_JS_URL_ES5] = UrlManager(conf.get(CONF_EXTRA_JS_URL_ES5, []))
|
2019-06-21 20:16:28 +00:00
|
|
|
|
2020-08-05 15:42:23 +00:00
|
|
|
await _async_setup_themes(hass, conf.get(CONF_THEMES))
|
2018-03-01 03:31:38 +00:00
|
|
|
|
2015-01-30 07:56:04 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
2021-06-24 14:01:28 +00:00
|
|
|
async def _async_setup_themes(
|
|
|
|
hass: HomeAssistant, themes: dict[str, Any] | None
|
|
|
|
) -> None:
|
2017-07-13 01:08:13 +00:00
|
|
|
"""Set up themes data and services."""
|
2020-02-10 03:47:16 +00:00
|
|
|
hass.data[DATA_THEMES] = themes or {}
|
2017-07-13 01:08:13 +00:00
|
|
|
|
2020-08-05 15:42:23 +00:00
|
|
|
store = hass.data[DATA_THEMES_STORE] = hass.helpers.storage.Store(
|
|
|
|
THEMES_STORAGE_VERSION, THEMES_STORAGE_KEY
|
|
|
|
)
|
|
|
|
|
|
|
|
theme_data = await store.async_load() or {}
|
|
|
|
theme_name = theme_data.get(DATA_DEFAULT_THEME, DEFAULT_THEME)
|
|
|
|
dark_theme_name = theme_data.get(DATA_DEFAULT_DARK_THEME)
|
|
|
|
|
|
|
|
if theme_name == DEFAULT_THEME or theme_name in hass.data[DATA_THEMES]:
|
|
|
|
hass.data[DATA_DEFAULT_THEME] = theme_name
|
|
|
|
else:
|
|
|
|
hass.data[DATA_DEFAULT_THEME] = DEFAULT_THEME
|
|
|
|
|
|
|
|
if dark_theme_name == DEFAULT_THEME or dark_theme_name in hass.data[DATA_THEMES]:
|
|
|
|
hass.data[DATA_DEFAULT_DARK_THEME] = dark_theme_name
|
|
|
|
|
2017-07-13 01:08:13 +00:00
|
|
|
@callback
|
2021-06-24 14:01:28 +00:00
|
|
|
def update_theme_and_fire_event() -> None:
|
2017-07-13 01:08:13 +00:00
|
|
|
"""Update theme_color in manifest."""
|
|
|
|
name = hass.data[DATA_DEFAULT_THEME]
|
|
|
|
themes = hass.data[DATA_THEMES]
|
2020-02-03 22:09:25 +00:00
|
|
|
if name != DEFAULT_THEME:
|
2021-04-19 04:17:30 +00:00
|
|
|
MANIFEST_JSON.update_key(
|
|
|
|
"theme_color",
|
|
|
|
themes[name].get(
|
|
|
|
"app-header-background-color",
|
|
|
|
themes[name].get(PRIMARY_COLOR, DEFAULT_THEME_COLOR),
|
|
|
|
),
|
2020-02-03 22:09:25 +00:00
|
|
|
)
|
2021-04-19 04:17:30 +00:00
|
|
|
else:
|
|
|
|
MANIFEST_JSON.update_key("theme_color", DEFAULT_THEME_COLOR)
|
2020-02-10 03:47:16 +00:00
|
|
|
hass.bus.async_fire(EVENT_THEMES_UPDATED)
|
2017-07-13 01:08:13 +00:00
|
|
|
|
|
|
|
@callback
|
2021-06-24 14:01:28 +00:00
|
|
|
def set_theme(call: ServiceCall) -> None:
|
2018-01-27 19:58:27 +00:00
|
|
|
"""Set backend-preferred theme."""
|
2020-08-05 15:42:23 +00:00
|
|
|
name = call.data[CONF_NAME]
|
|
|
|
mode = call.data.get("mode", "light")
|
|
|
|
|
|
|
|
if (
|
|
|
|
name not in (DEFAULT_THEME, VALUE_NO_THEME)
|
|
|
|
and name not in hass.data[DATA_THEMES]
|
|
|
|
):
|
|
|
|
_LOGGER.warning("Theme %s not found", name)
|
|
|
|
return
|
|
|
|
|
|
|
|
light_mode = mode == "light"
|
|
|
|
|
|
|
|
theme_key = DATA_DEFAULT_THEME if light_mode else DATA_DEFAULT_DARK_THEME
|
|
|
|
|
|
|
|
if name == VALUE_NO_THEME:
|
|
|
|
to_set = DEFAULT_THEME if light_mode else None
|
2017-07-13 01:08:13 +00:00
|
|
|
else:
|
2020-08-05 15:42:23 +00:00
|
|
|
_LOGGER.info("Theme %s set as default %s theme", name, mode)
|
|
|
|
to_set = name
|
|
|
|
|
|
|
|
hass.data[theme_key] = to_set
|
|
|
|
store.async_delay_save(
|
|
|
|
lambda: {
|
|
|
|
DATA_DEFAULT_THEME: hass.data[DATA_DEFAULT_THEME],
|
|
|
|
DATA_DEFAULT_DARK_THEME: hass.data.get(DATA_DEFAULT_DARK_THEME),
|
|
|
|
},
|
|
|
|
THEMES_SAVE_DELAY,
|
|
|
|
)
|
|
|
|
update_theme_and_fire_event()
|
2017-07-13 01:08:13 +00:00
|
|
|
|
2021-06-24 14:01:28 +00:00
|
|
|
async def reload_themes(_: ServiceCall) -> None:
|
2017-07-13 01:08:13 +00:00
|
|
|
"""Reload themes."""
|
2020-01-14 21:03:02 +00:00
|
|
|
config = await async_hass_config_yaml(hass)
|
|
|
|
new_themes = config[DOMAIN].get(CONF_THEMES, {})
|
2017-07-13 01:08:13 +00:00
|
|
|
hass.data[DATA_THEMES] = new_themes
|
|
|
|
if hass.data[DATA_DEFAULT_THEME] not in new_themes:
|
|
|
|
hass.data[DATA_DEFAULT_THEME] = DEFAULT_THEME
|
2020-08-05 15:42:23 +00:00
|
|
|
if (
|
|
|
|
hass.data.get(DATA_DEFAULT_DARK_THEME)
|
|
|
|
and hass.data.get(DATA_DEFAULT_DARK_THEME) not in new_themes
|
|
|
|
):
|
|
|
|
hass.data[DATA_DEFAULT_DARK_THEME] = None
|
2017-07-13 01:08:13 +00:00
|
|
|
update_theme_and_fire_event()
|
|
|
|
|
2020-02-10 03:47:16 +00:00
|
|
|
service.async_register_admin_service(
|
|
|
|
hass,
|
|
|
|
DOMAIN,
|
|
|
|
SERVICE_SET_THEME,
|
|
|
|
set_theme,
|
2020-08-05 15:42:23 +00:00
|
|
|
vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_NAME): cv.string,
|
|
|
|
vol.Optional(CONF_MODE): vol.Any("dark", "light"),
|
|
|
|
}
|
|
|
|
),
|
2020-02-10 03:47:16 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
service.async_register_admin_service(
|
|
|
|
hass, DOMAIN, SERVICE_RELOAD_THEMES, reload_themes
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
|
|
|
|
2021-04-19 04:17:30 +00:00
|
|
|
@callback
|
|
|
|
@lru_cache(maxsize=1)
|
2021-06-24 14:01:28 +00:00
|
|
|
def _async_render_index_cached(template: jinja2.Template, **kwargs: Any) -> str:
|
2021-04-19 04:17:30 +00:00
|
|
|
return template.render(**kwargs)
|
|
|
|
|
|
|
|
|
2019-05-31 18:27:05 +00:00
|
|
|
class IndexView(web_urldispatcher.AbstractResource):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Serve the frontend."""
|
|
|
|
|
2021-06-24 14:01:28 +00:00
|
|
|
def __init__(self, repo_path: str | None, hass: HomeAssistant) -> None:
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Initialize the frontend view."""
|
2019-05-31 18:27:05 +00:00
|
|
|
super().__init__(name="frontend:index")
|
2017-11-11 23:22:05 +00:00
|
|
|
self.repo_path = repo_path
|
2019-05-31 18:27:05 +00:00
|
|
|
self.hass = hass
|
2021-06-24 14:01:28 +00:00
|
|
|
self._template_cache: jinja2.Template | None = None
|
2017-11-11 23:22:05 +00:00
|
|
|
|
2019-05-31 18:27:05 +00:00
|
|
|
@property
|
|
|
|
def canonical(self) -> str:
|
|
|
|
"""Return resource's canonical path."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return "/"
|
2019-05-31 18:27:05 +00:00
|
|
|
|
|
|
|
@property
|
2021-06-24 14:01:28 +00:00
|
|
|
def _route(self) -> web_urldispatcher.ResourceRoute:
|
2019-05-31 18:27:05 +00:00
|
|
|
"""Return the index route."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return web_urldispatcher.ResourceRoute("GET", self.get, self)
|
2019-05-31 18:27:05 +00:00
|
|
|
|
|
|
|
def url_for(self, **kwargs: str) -> URL:
|
|
|
|
"""Construct url for resource with additional params."""
|
|
|
|
return URL("/")
|
|
|
|
|
2019-09-20 15:23:34 +00:00
|
|
|
async def resolve(
|
|
|
|
self, request: web.Request
|
2021-03-17 22:49:01 +00:00
|
|
|
) -> tuple[web_urldispatcher.UrlMappingMatchInfo | None, set[str]]:
|
2019-05-31 18:27:05 +00:00
|
|
|
"""Resolve resource.
|
|
|
|
|
|
|
|
Return (UrlMappingMatchInfo, allowed_methods) pair.
|
|
|
|
"""
|
2019-07-31 19:25:30 +00:00
|
|
|
if (
|
|
|
|
request.path != "/"
|
2022-04-14 18:57:08 +00:00
|
|
|
and len(request.url.parts) > 1
|
2019-07-31 19:25:30 +00:00
|
|
|
and request.url.parts[1] not in self.hass.data[DATA_PANELS]
|
|
|
|
):
|
2019-05-31 18:27:05 +00:00
|
|
|
return None, set()
|
|
|
|
|
|
|
|
if request.method != hdrs.METH_GET:
|
2019-07-31 19:25:30 +00:00
|
|
|
return None, {"GET"}
|
2019-05-31 18:27:05 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
return web_urldispatcher.UrlMappingMatchInfo({}, self._route), {"GET"}
|
2019-05-31 18:27:05 +00:00
|
|
|
|
|
|
|
def add_prefix(self, prefix: str) -> None:
|
|
|
|
"""Add a prefix to processed URLs.
|
|
|
|
|
|
|
|
Required for subapplications support.
|
|
|
|
"""
|
|
|
|
|
2021-06-24 14:01:28 +00:00
|
|
|
def get_info(self) -> dict[str, list[str]]: # type: ignore[override]
|
2019-05-31 18:27:05 +00:00
|
|
|
"""Return a dict with additional info useful for introspection."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return {"panels": list(self.hass.data[DATA_PANELS])}
|
2019-05-31 18:27:05 +00:00
|
|
|
|
|
|
|
def freeze(self) -> None:
|
|
|
|
"""Freeze the resource."""
|
|
|
|
|
|
|
|
def raw_match(self, path: str) -> bool:
|
|
|
|
"""Perform a raw match against path."""
|
|
|
|
|
2021-06-24 14:01:28 +00:00
|
|
|
def get_template(self) -> jinja2.Template:
|
2017-11-11 23:22:05 +00:00
|
|
|
"""Get template."""
|
2021-10-17 18:05:11 +00:00
|
|
|
if (tpl := self._template_cache) is None:
|
2021-07-28 07:41:45 +00:00
|
|
|
with (_frontend_root(self.repo_path) / "index.html").open(
|
|
|
|
encoding="utf8"
|
|
|
|
) as file:
|
2017-11-11 23:22:05 +00:00
|
|
|
tpl = jinja2.Template(file.read())
|
|
|
|
|
|
|
|
# Cache template if not running from repository
|
|
|
|
if self.repo_path is None:
|
2019-05-02 20:59:24 +00:00
|
|
|
self._template_cache = tpl
|
2017-11-11 23:22:05 +00:00
|
|
|
|
|
|
|
return tpl
|
2016-05-12 05:55:24 +00:00
|
|
|
|
2019-09-20 15:23:34 +00:00
|
|
|
async def get(self, request: web.Request) -> web.Response:
|
2019-05-31 18:27:05 +00:00
|
|
|
"""Serve the index page for panel pages."""
|
2019-07-31 19:25:30 +00:00
|
|
|
hass = request.app["hass"]
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2022-01-14 15:29:48 +00:00
|
|
|
if not onboarding.async_is_onboarded(hass):
|
2019-07-31 19:25:30 +00:00
|
|
|
return web.Response(status=302, headers={"location": "/onboarding.html"})
|
2018-07-17 08:49:15 +00:00
|
|
|
|
2021-04-19 04:17:30 +00:00
|
|
|
template = self._template_cache or await hass.async_add_executor_job(
|
|
|
|
self.get_template
|
|
|
|
)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2019-05-02 20:59:24 +00:00
|
|
|
return web.Response(
|
2021-04-19 04:17:30 +00:00
|
|
|
text=_async_render_index_cached(
|
|
|
|
template,
|
2019-07-31 19:25:30 +00:00
|
|
|
theme_color=MANIFEST_JSON["theme_color"],
|
2021-04-19 04:17:30 +00:00
|
|
|
extra_modules=hass.data[DATA_EXTRA_MODULE_URL].urls,
|
|
|
|
extra_js_es5=hass.data[DATA_EXTRA_JS_URL_ES5].urls,
|
2019-05-02 20:59:24 +00:00
|
|
|
),
|
2019-07-31 19:25:30 +00:00
|
|
|
content_type="text/html",
|
2017-11-11 23:22:05 +00:00
|
|
|
)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2019-05-31 18:27:05 +00:00
|
|
|
def __len__(self) -> int:
|
|
|
|
"""Return length of resource."""
|
|
|
|
return 1
|
|
|
|
|
2021-06-24 14:01:28 +00:00
|
|
|
def __iter__(self) -> Iterator[web_urldispatcher.ResourceRoute]:
|
2019-05-31 18:27:05 +00:00
|
|
|
"""Iterate over routes."""
|
|
|
|
return iter([self._route])
|
|
|
|
|
2016-08-14 08:10:07 +00:00
|
|
|
|
|
|
|
class ManifestJSONView(HomeAssistantView):
|
|
|
|
"""View to return a manifest.json."""
|
|
|
|
|
|
|
|
requires_auth = False
|
2019-07-31 19:25:30 +00:00
|
|
|
url = "/manifest.json"
|
|
|
|
name = "manifestjson"
|
2016-08-14 08:10:07 +00:00
|
|
|
|
2018-06-05 14:50:16 +00:00
|
|
|
@callback
|
2022-04-25 14:41:01 +00:00
|
|
|
def get(self, request: web.Request) -> web.Response:
|
2016-08-14 08:10:07 +00:00
|
|
|
"""Return the manifest.json."""
|
2021-04-19 04:17:30 +00:00
|
|
|
return web.Response(
|
|
|
|
text=MANIFEST_JSON.json, content_type="application/manifest+json"
|
|
|
|
)
|
2017-07-13 01:08:13 +00:00
|
|
|
|
|
|
|
|
2018-05-03 20:02:59 +00:00
|
|
|
@callback
|
2020-02-10 03:47:16 +00:00
|
|
|
@websocket_api.websocket_command({"type": "get_panels"})
|
2021-06-24 14:01:28 +00:00
|
|
|
def websocket_get_panels(
|
|
|
|
hass: HomeAssistant, connection: ActiveConnection, msg: dict
|
|
|
|
) -> None:
|
2020-04-26 00:30:15 +00:00
|
|
|
"""Handle get panels command."""
|
2019-03-25 17:04:35 +00:00
|
|
|
user_is_admin = connection.user.is_admin
|
2018-05-01 17:35:23 +00:00
|
|
|
panels = {
|
2019-03-25 17:04:35 +00:00
|
|
|
panel_key: panel.to_response()
|
|
|
|
for panel_key, panel in connection.hass.data[DATA_PANELS].items()
|
2019-07-31 19:25:30 +00:00
|
|
|
if user_is_admin or not panel.require_admin
|
|
|
|
}
|
2018-05-01 17:35:23 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
connection.send_message(websocket_api.result_message(msg["id"], panels))
|
2018-06-06 08:12:43 +00:00
|
|
|
|
|
|
|
|
|
|
|
@callback
|
2020-02-10 03:47:16 +00:00
|
|
|
@websocket_api.websocket_command({"type": "frontend/get_themes"})
|
2021-06-24 14:01:28 +00:00
|
|
|
def websocket_get_themes(
|
|
|
|
hass: HomeAssistant, connection: ActiveConnection, msg: dict
|
|
|
|
) -> None:
|
2020-04-26 00:30:15 +00:00
|
|
|
"""Handle get themes command."""
|
2020-02-18 19:52:38 +00:00
|
|
|
if hass.config.safe_mode:
|
|
|
|
connection.send_message(
|
|
|
|
websocket_api.result_message(
|
|
|
|
msg["id"],
|
|
|
|
{
|
|
|
|
"themes": {
|
|
|
|
"safe_mode": {
|
|
|
|
"primary-color": "#db4437",
|
2020-09-23 16:57:35 +00:00
|
|
|
"accent-color": "#ffca28",
|
2020-02-18 19:52:38 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
"default_theme": "safe_mode",
|
|
|
|
},
|
|
|
|
)
|
|
|
|
)
|
|
|
|
return
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
connection.send_message(
|
|
|
|
websocket_api.result_message(
|
|
|
|
msg["id"],
|
|
|
|
{
|
|
|
|
"themes": hass.data[DATA_THEMES],
|
|
|
|
"default_theme": hass.data[DATA_DEFAULT_THEME],
|
2020-08-05 15:42:23 +00:00
|
|
|
"default_dark_theme": hass.data.get(DATA_DEFAULT_DARK_THEME),
|
2019-07-31 19:25:30 +00:00
|
|
|
},
|
|
|
|
)
|
|
|
|
)
|
2018-06-06 08:12:43 +00:00
|
|
|
|
|
|
|
|
2020-02-10 03:47:16 +00:00
|
|
|
@websocket_api.websocket_command(
|
2020-04-19 00:13:13 +00:00
|
|
|
{
|
|
|
|
"type": "frontend/get_translations",
|
|
|
|
vol.Required("language"): str,
|
|
|
|
vol.Required("category"): str,
|
|
|
|
vol.Optional("integration"): str,
|
|
|
|
vol.Optional("config_flow"): bool,
|
|
|
|
}
|
2020-02-10 03:47:16 +00:00
|
|
|
)
|
2018-10-01 14:09:31 +00:00
|
|
|
@websocket_api.async_response
|
2021-06-24 14:01:28 +00:00
|
|
|
async def websocket_get_translations(
|
|
|
|
hass: HomeAssistant, connection: ActiveConnection, msg: dict
|
|
|
|
) -> None:
|
2020-04-26 00:30:15 +00:00
|
|
|
"""Handle get translations command."""
|
2020-04-19 00:13:13 +00:00
|
|
|
resources = await async_get_translations(
|
|
|
|
hass,
|
|
|
|
msg["language"],
|
|
|
|
msg["category"],
|
|
|
|
msg.get("integration"),
|
|
|
|
msg.get("config_flow"),
|
|
|
|
)
|
2019-07-31 19:25:30 +00:00
|
|
|
connection.send_message(
|
|
|
|
websocket_api.result_message(msg["id"], {"resources": resources})
|
|
|
|
)
|
2020-04-26 00:30:15 +00:00
|
|
|
|
|
|
|
|
|
|
|
@websocket_api.websocket_command({"type": "frontend/get_version"})
|
|
|
|
@websocket_api.async_response
|
2021-06-24 14:01:28 +00:00
|
|
|
async def websocket_get_version(
|
|
|
|
hass: HomeAssistant, connection: ActiveConnection, msg: dict
|
|
|
|
) -> None:
|
2020-04-26 00:30:15 +00:00
|
|
|
"""Handle get version command."""
|
|
|
|
integration = await async_get_integration(hass, "frontend")
|
|
|
|
|
|
|
|
frontend = None
|
|
|
|
|
|
|
|
for req in integration.requirements:
|
|
|
|
if req.startswith("home-assistant-frontend=="):
|
|
|
|
frontend = req.split("==", 1)[1]
|
|
|
|
|
|
|
|
if frontend is None:
|
|
|
|
connection.send_error(msg["id"], "unknown_version", "Version not found")
|
|
|
|
else:
|
|
|
|
connection.send_result(msg["id"], {"version": frontend})
|
2021-06-24 14:01:28 +00:00
|
|
|
|
|
|
|
|
|
|
|
class PanelRespons(TypedDict):
|
|
|
|
"""Represent the panel response type."""
|
|
|
|
|
|
|
|
component_name: str
|
|
|
|
icon: str | None
|
|
|
|
title: str | None
|
|
|
|
config: dict[str, Any] | None
|
|
|
|
url_path: str | None
|
|
|
|
require_admin: bool
|