"""Handle the frontend for Home Assistant.""" from __future__ import annotations from collections.abc import Iterator from functools import lru_cache import logging import os import pathlib from typing import Any, TypedDict from aiohttp import hdrs, web, web_urldispatcher import jinja2 import voluptuous as vol from yarl import URL from homeassistant.components import onboarding, websocket_api from homeassistant.components.http.view import HomeAssistantView from homeassistant.components.websocket_api.connection import ActiveConnection from homeassistant.config import async_hass_config_yaml from homeassistant.const import CONF_MODE, CONF_NAME, EVENT_THEMES_UPDATED from homeassistant.core import HomeAssistant, ServiceCall, callback from homeassistant.helpers import service import homeassistant.helpers.config_validation as cv from homeassistant.helpers.json import json_dumps_sorted from homeassistant.helpers.storage import Store from homeassistant.helpers.translation import async_get_translations from homeassistant.helpers.typing import ConfigType from homeassistant.loader import async_get_integration, bind_hass from .storage import async_setup_frontend_storage DOMAIN = "frontend" CONF_THEMES = "themes" CONF_THEMES_MODES = "modes" CONF_THEMES_LIGHT = "light" CONF_THEMES_DARK = "dark" 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" 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" THEMES_STORAGE_KEY = f"{DOMAIN}_theme" THEMES_STORAGE_VERSION = 1 THEMES_SAVE_DELAY = 60 DATA_THEMES_STORE = "frontend_themes_store" DATA_THEMES = "frontend_themes" DATA_DEFAULT_THEME = "frontend_default_theme" DATA_DEFAULT_DARK_THEME = "frontend_default_dark_theme" DEFAULT_THEME = "default" VALUE_NO_THEME = "none" PRIMARY_COLOR = "primary-color" _LOGGER = logging.getLogger(__name__) 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, ) ) } ) CONFIG_SCHEMA = vol.Schema( { DOMAIN: vol.Schema( { vol.Optional(CONF_FRONTEND_REPO): cv.isdir, vol.Optional(CONF_THEMES): THEME_SCHEMA, 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, }, ) }, extra=vol.ALLOW_EXTRA, ) SERVICE_SET_THEME = "set_theme" SERVICE_RELOAD_THEMES = "reload_themes" 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_sorted(self.manifest) 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 """ def __init__(self, urls: list[str]) -> None: """Init the url manager.""" self.urls = frozenset(urls) def add(self, url: str) -> None: """Add a url to the set.""" self.urls = frozenset([*self.urls, url]) def remove(self, url: str) -> None: """Remove a url from the set.""" self.urls = self.urls - {url} class Panel: """Abstract class for panels.""" # Name of the webcomponent component_name: str # Icon to show in the sidebar sidebar_icon: str | None = None # Title to show in the sidebar sidebar_title: str | None = None # Url to show the panel in the frontend frontend_url_path: str | None = None # Config to pass to the webcomponent config: dict[str, Any] | None = None # If the panel should only be visible to admins require_admin = False # If the panel is a configuration panel for a integration config_panel_domain: str | None = None def __init__( self, component_name: str, sidebar_title: str | None, sidebar_icon: str | None, frontend_url_path: str | None, config: dict[str, Any] | None, require_admin: bool, config_panel_domain: str | None, ) -> None: """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 self.require_admin = require_admin self.config_panel_domain = config_panel_domain @callback def to_response(self) -> PanelRespons: """Panel as dictionary.""" return { "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, "config_panel_domain": self.config_panel_domain, } @bind_hass @callback def async_register_built_in_panel( 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, *, update: bool = False, config_panel_domain: str | None = None, ) -> None: """Register a built-in panel.""" panel = Panel( component_name, sidebar_title, sidebar_icon, frontend_url_path, config, require_admin, config_panel_domain, ) panels = hass.data.setdefault(DATA_PANELS, {}) if not update and panel.frontend_url_path in panels: raise ValueError(f"Overwriting panel {panel.frontend_url_path}") panels[panel.frontend_url_path] = panel hass.bus.async_fire(EVENT_PANELS_UPDATED) @bind_hass @callback def async_remove_panel(hass: HomeAssistant, frontend_url_path: str) -> None: """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) def add_extra_js_url(hass: HomeAssistant, url: str, es5: bool = False) -> None: """Register extra js or module url to load.""" key = DATA_EXTRA_JS_URL_ES5 if es5 else DATA_EXTRA_MODULE_URL hass.data[key].add(url) def add_manifest_json_key(key: str, val: Any) -> None: """Add a keyval to the manifest.json.""" MANIFEST_JSON.update_key(key, val) def _frontend_root(dev_repo_path: str | None) -> pathlib.Path: """Return root path to the frontend files.""" if dev_repo_path is not None: return pathlib.Path(dev_repo_path) / "hass_frontend" # Keep import here so that we can import frontend without installing reqs # pylint: disable-next=import-outside-toplevel import hass_frontend return hass_frontend.where() async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool: """Set up the serving of the frontend.""" await async_setup_frontend_storage(hass) 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) hass.http.register_view(ManifestJSONView()) conf = config.get(DOMAIN, {}) 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, ) repo_path = conf.get(CONF_FRONTEND_REPO) is_dev = repo_path is not None root_path = _frontend_root(repo_path) for path, should_cache in ( ("service_worker.js", False), ("robots.txt", False), ("onboarding.html", not is_dev), ("static", not is_dev), ("frontend_latest", not is_dev), ("frontend_es5", not is_dev), ): hass.http.register_static_path(f"/{path}", str(root_path / path), should_cache) hass.http.register_static_path( "/auth/authorize", str(root_path / "authorize.html"), False ) # https://wicg.github.io/change-password-url/ hass.http.register_redirect( "/.well-known/change-password", "/profile", redirect_exc=web.HTTPFound ) local = hass.config.path("www") if os.path.isdir(local): hass.http.register_static_path("/local", local, not is_dev) # Can be removed in 2023 hass.http.register_redirect("/config/server_control", "/developer-tools/yaml") hass.http.app.router.register_resource(IndexView(repo_path, hass)) async_register_built_in_panel(hass, "profile") async_register_built_in_panel( hass, "developer-tools", require_admin=True, sidebar_title="developer_tools", sidebar_icon="hass:hammer", ) 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, [])) await _async_setup_themes(hass, conf.get(CONF_THEMES)) return True async def _async_setup_themes( hass: HomeAssistant, themes: dict[str, Any] | None ) -> None: """Set up themes data and services.""" hass.data[DATA_THEMES] = themes or {} store = hass.data[DATA_THEMES_STORE] = Store( hass, THEMES_STORAGE_VERSION, THEMES_STORAGE_KEY ) if not (theme_data := await store.async_load()) or not isinstance(theme_data, dict): theme_data = {} 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 @callback def update_theme_and_fire_event() -> None: """Update theme_color in manifest.""" name = hass.data[DATA_DEFAULT_THEME] themes = hass.data[DATA_THEMES] if name != DEFAULT_THEME: MANIFEST_JSON.update_key( "theme_color", themes[name].get( "app-header-background-color", themes[name].get(PRIMARY_COLOR, DEFAULT_THEME_COLOR), ), ) else: MANIFEST_JSON.update_key("theme_color", DEFAULT_THEME_COLOR) hass.bus.async_fire(EVENT_THEMES_UPDATED) @callback def set_theme(call: ServiceCall) -> None: """Set backend-preferred theme.""" 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 else: _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() async def reload_themes(_: ServiceCall) -> None: """Reload themes.""" config = await async_hass_config_yaml(hass) new_themes = config.get(DOMAIN, {}).get(CONF_THEMES, {}) hass.data[DATA_THEMES] = new_themes if hass.data[DATA_DEFAULT_THEME] not in new_themes: hass.data[DATA_DEFAULT_THEME] = DEFAULT_THEME 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 update_theme_and_fire_event() service.async_register_admin_service( hass, DOMAIN, SERVICE_SET_THEME, set_theme, vol.Schema( { vol.Required(CONF_NAME): cv.string, vol.Optional(CONF_MODE): vol.Any("dark", "light"), } ), ) service.async_register_admin_service( hass, DOMAIN, SERVICE_RELOAD_THEMES, reload_themes ) @callback @lru_cache(maxsize=1) def _async_render_index_cached(template: jinja2.Template, **kwargs: Any) -> str: return template.render(**kwargs) class IndexView(web_urldispatcher.AbstractResource): """Serve the frontend.""" def __init__(self, repo_path: str | None, hass: HomeAssistant) -> None: """Initialize the frontend view.""" super().__init__(name="frontend:index") self.repo_path = repo_path self.hass = hass self._template_cache: jinja2.Template | None = None @property def canonical(self) -> str: """Return resource's canonical path.""" return "/" @property def _route(self) -> web_urldispatcher.ResourceRoute: """Return the index route.""" return web_urldispatcher.ResourceRoute("GET", self.get, self) def url_for(self, **kwargs: str) -> URL: """Construct url for resource with additional params.""" return URL("/") async def resolve( self, request: web.Request ) -> tuple[web_urldispatcher.UrlMappingMatchInfo | None, set[str]]: """Resolve resource. Return (UrlMappingMatchInfo, allowed_methods) pair. """ if ( request.path != "/" and (parts := request.rel_url.parts) and len(parts) > 1 and parts[1] not in self.hass.data[DATA_PANELS] ): return None, set() if request.method != hdrs.METH_GET: return None, {"GET"} return web_urldispatcher.UrlMappingMatchInfo({}, self._route), {"GET"} def add_prefix(self, prefix: str) -> None: """Add a prefix to processed URLs. Required for subapplications support. """ def get_info(self) -> dict[str, list[str]]: # type: ignore[override] """Return a dict with additional info useful for introspection.""" return {"panels": list(self.hass.data[DATA_PANELS])} def raw_match(self, path: str) -> bool: """Perform a raw match against path.""" return False def get_template(self) -> jinja2.Template: """Get template.""" if (tpl := self._template_cache) is None: with (_frontend_root(self.repo_path) / "index.html").open( encoding="utf8" ) as file: tpl = jinja2.Template(file.read()) # Cache template if not running from repository if self.repo_path is None: self._template_cache = tpl return tpl async def get(self, request: web.Request) -> web.Response: """Serve the index page for panel pages.""" hass = request.app["hass"] if not onboarding.async_is_onboarded(hass): return web.Response(status=302, headers={"location": "/onboarding.html"}) template = self._template_cache or await hass.async_add_executor_job( self.get_template ) return web.Response( text=_async_render_index_cached( template, theme_color=MANIFEST_JSON["theme_color"], extra_modules=hass.data[DATA_EXTRA_MODULE_URL].urls, extra_js_es5=hass.data[DATA_EXTRA_JS_URL_ES5].urls, ), content_type="text/html", ) def __len__(self) -> int: """Return length of resource.""" return 1 def __iter__(self) -> Iterator[web_urldispatcher.ResourceRoute]: """Iterate over routes.""" return iter([self._route]) class ManifestJSONView(HomeAssistantView): """View to return a manifest.json.""" requires_auth = False url = "/manifest.json" name = "manifestjson" @callback def get(self, request: web.Request) -> web.Response: """Return the manifest.json.""" return web.Response( text=MANIFEST_JSON.json, content_type="application/manifest+json" ) @callback @websocket_api.websocket_command({"type": "get_panels"}) def websocket_get_panels( hass: HomeAssistant, connection: ActiveConnection, msg: dict[str, Any] ) -> None: """Handle get panels command.""" user_is_admin = connection.user.is_admin panels = { panel_key: panel.to_response() for panel_key, panel in connection.hass.data[DATA_PANELS].items() if user_is_admin or not panel.require_admin } connection.send_message(websocket_api.result_message(msg["id"], panels)) @callback @websocket_api.websocket_command({"type": "frontend/get_themes"}) def websocket_get_themes( hass: HomeAssistant, connection: ActiveConnection, msg: dict[str, Any] ) -> None: """Handle get themes command.""" if hass.config.safe_mode: connection.send_message( websocket_api.result_message( msg["id"], { "themes": { "safe_mode": { "primary-color": "#db4437", "accent-color": "#ffca28", } }, "default_theme": "safe_mode", }, ) ) return connection.send_message( websocket_api.result_message( msg["id"], { "themes": hass.data[DATA_THEMES], "default_theme": hass.data[DATA_DEFAULT_THEME], "default_dark_theme": hass.data.get(DATA_DEFAULT_DARK_THEME), }, ) ) @websocket_api.websocket_command( { "type": "frontend/get_translations", vol.Required("language"): str, vol.Required("category"): str, vol.Optional("integration"): vol.All(cv.ensure_list, [str]), vol.Optional("config_flow"): bool, } ) @websocket_api.async_response async def websocket_get_translations( hass: HomeAssistant, connection: ActiveConnection, msg: dict[str, Any] ) -> None: """Handle get translations command.""" resources = await async_get_translations( hass, msg["language"], msg["category"], msg.get("integration"), msg.get("config_flow"), ) connection.send_message( websocket_api.result_message(msg["id"], {"resources": resources}) ) @websocket_api.websocket_command({"type": "frontend/get_version"}) @websocket_api.async_response async def websocket_get_version( hass: HomeAssistant, connection: ActiveConnection, msg: dict[str, Any] ) -> None: """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.removeprefix("home-assistant-frontend==") if frontend is None: connection.send_error(msg["id"], "unknown_version", "Version not found") else: connection.send_result(msg["id"], {"version": frontend}) 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 config_panel_domain: str | None