2017-11-01 11:11:32 +00:00
|
|
|
"""
|
|
|
|
Handle the frontend for Home Assistant.
|
|
|
|
|
|
|
|
For more details about this component, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/frontend/
|
|
|
|
"""
|
2016-10-24 06:48:01 +00:00
|
|
|
import asyncio
|
2016-07-29 07:49:58 +00:00
|
|
|
import hashlib
|
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
|
2017-11-11 07:02:06 +00:00
|
|
|
from urllib.parse import urlparse
|
2015-01-30 07:56:04 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
from aiohttp import web
|
2017-07-13 01:08:13 +00:00
|
|
|
import voluptuous as vol
|
2017-11-11 23:22:05 +00:00
|
|
|
import jinja2
|
2016-10-24 06:48:01 +00:00
|
|
|
|
2017-11-01 11:11:32 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2018-05-01 17:35:23 +00:00
|
|
|
from homeassistant.components.http.view import HomeAssistantView
|
2018-02-15 21:06:14 +00:00
|
|
|
from homeassistant.components.http.const import KEY_AUTHENTICATED
|
2018-05-01 17:35:23 +00:00
|
|
|
from homeassistant.components import websocket_api
|
2017-07-13 01:08:13 +00:00
|
|
|
from homeassistant.config import find_config_file, load_yaml_config_file
|
|
|
|
from homeassistant.const import CONF_NAME, EVENT_THEMES_UPDATED
|
2016-10-24 06:48:01 +00:00
|
|
|
from homeassistant.core import callback
|
2018-03-01 03:31:38 +00:00
|
|
|
from homeassistant.helpers.translation import async_get_translations
|
2017-07-22 04:38:53 +00:00
|
|
|
from homeassistant.loader import bind_hass
|
2017-11-01 11:11:32 +00:00
|
|
|
|
2018-06-03 16:29:57 +00:00
|
|
|
REQUIREMENTS = ['home-assistant-frontend==20180603.0']
|
2015-01-30 07:56:04 +00:00
|
|
|
|
|
|
|
DOMAIN = 'frontend'
|
2017-11-15 04:35:56 +00:00
|
|
|
DEPENDENCIES = ['api', 'websocket_api', 'http', 'system_log']
|
2017-04-30 05:04:49 +00:00
|
|
|
|
2016-07-17 05:32:25 +00:00
|
|
|
URL_PANEL_COMPONENT_FP = '/frontend/panels/{}-{}.html'
|
2017-04-30 05:04:49 +00:00
|
|
|
|
2017-10-25 02:36:27 +00:00
|
|
|
CONF_THEMES = 'themes'
|
|
|
|
CONF_EXTRA_HTML_URL = 'extra_html_url'
|
2017-11-29 06:53:12 +00:00
|
|
|
CONF_EXTRA_HTML_URL_ES5 = 'extra_html_url_es5'
|
2017-10-25 02:36:27 +00:00
|
|
|
CONF_FRONTEND_REPO = 'development_repo'
|
2017-11-11 07:02:06 +00:00
|
|
|
CONF_JS_VERSION = 'javascript_version'
|
2017-12-08 17:16:26 +00:00
|
|
|
JS_DEFAULT_OPTION = 'auto'
|
2017-11-11 07:02:06 +00:00
|
|
|
JS_OPTIONS = ['es5', 'latest', 'auto']
|
2017-11-01 11:11:32 +00:00
|
|
|
|
2017-07-13 01:08:13 +00:00
|
|
|
DEFAULT_THEME_COLOR = '#03A9F4'
|
2017-11-01 11:11:32 +00:00
|
|
|
|
2016-08-14 08:10:07 +00:00
|
|
|
MANIFEST_JSON = {
|
2017-04-30 05:04:49 +00:00
|
|
|
'background_color': '#FFFFFF',
|
|
|
|
'description': 'Open-source home automation platform running on Python 3.',
|
|
|
|
'dir': 'ltr',
|
|
|
|
'display': 'standalone',
|
|
|
|
'icons': [],
|
|
|
|
'lang': 'en-US',
|
|
|
|
'name': 'Home Assistant',
|
2017-07-01 04:57:38 +00:00
|
|
|
'short_name': 'Assistant',
|
2017-12-23 05:05:15 +00:00
|
|
|
'start_url': '/states',
|
2017-07-13 01:08:13 +00:00
|
|
|
'theme_color': DEFAULT_THEME_COLOR
|
2016-08-14 08:10:07 +00:00
|
|
|
}
|
2016-07-20 06:36:46 +00:00
|
|
|
|
2016-11-25 05:37:56 +00:00
|
|
|
for size in (192, 384, 512, 1024):
|
|
|
|
MANIFEST_JSON['icons'].append({
|
2017-04-30 05:04:49 +00:00
|
|
|
'src': '/static/icons/favicon-{}x{}.png'.format(size, size),
|
|
|
|
'sizes': '{}x{}'.format(size, size),
|
|
|
|
'type': 'image/png'
|
2016-11-25 05:37:56 +00:00
|
|
|
})
|
|
|
|
|
2017-10-25 02:36:27 +00:00
|
|
|
DATA_FINALIZE_PANEL = 'frontend_finalize_panel'
|
2016-11-25 05:37:56 +00:00
|
|
|
DATA_PANELS = 'frontend_panels'
|
2017-11-11 07:02:06 +00:00
|
|
|
DATA_JS_VERSION = 'frontend_js_version'
|
2017-08-27 16:07:58 +00:00
|
|
|
DATA_EXTRA_HTML_URL = 'frontend_extra_html_url'
|
2017-11-29 06:53:12 +00:00
|
|
|
DATA_EXTRA_HTML_URL_ES5 = 'frontend_extra_html_url_es5'
|
2017-07-13 01:08:13 +00:00
|
|
|
DATA_THEMES = 'frontend_themes'
|
|
|
|
DATA_DEFAULT_THEME = 'frontend_default_theme'
|
|
|
|
DEFAULT_THEME = 'default'
|
|
|
|
|
|
|
|
PRIMARY_COLOR = 'primary-color'
|
2016-11-25 05:37:56 +00:00
|
|
|
|
2016-07-17 05:32:25 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2017-07-13 01:08:13 +00:00
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
|
|
|
DOMAIN: vol.Schema({
|
2017-10-25 02:36:27 +00:00
|
|
|
vol.Optional(CONF_FRONTEND_REPO): cv.isdir,
|
|
|
|
vol.Optional(CONF_THEMES): vol.Schema({
|
2017-07-13 01:08:13 +00:00
|
|
|
cv.string: {cv.string: cv.string}
|
|
|
|
}),
|
2017-10-25 02:36:27 +00:00
|
|
|
vol.Optional(CONF_EXTRA_HTML_URL):
|
2017-08-27 16:07:58 +00:00
|
|
|
vol.All(cv.ensure_list, [cv.string]),
|
2017-11-29 06:53:12 +00:00
|
|
|
vol.Optional(CONF_EXTRA_HTML_URL_ES5):
|
|
|
|
vol.All(cv.ensure_list, [cv.string]),
|
2017-11-11 07:02:06 +00:00
|
|
|
vol.Optional(CONF_JS_VERSION, default=JS_DEFAULT_OPTION):
|
|
|
|
vol.In(JS_OPTIONS)
|
2017-07-13 01:08:13 +00:00
|
|
|
}),
|
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
|
|
|
|
|
|
|
SERVICE_SET_THEME = 'set_theme'
|
|
|
|
SERVICE_RELOAD_THEMES = 'reload_themes'
|
|
|
|
SERVICE_SET_THEME_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(CONF_NAME): cv.string,
|
|
|
|
})
|
2018-05-01 17:35:23 +00:00
|
|
|
WS_TYPE_GET_PANELS = 'get_panels'
|
|
|
|
SCHEMA_GET_PANELS = websocket_api.BASE_COMMAND_MESSAGE_SCHEMA.extend({
|
|
|
|
vol.Required('type'): WS_TYPE_GET_PANELS,
|
|
|
|
})
|
2017-07-13 01:08:13 +00:00
|
|
|
|
2016-07-17 05:32:25 +00:00
|
|
|
|
2017-10-25 02:36:27 +00:00
|
|
|
class AbstractPanel:
|
|
|
|
"""Abstract class for panels."""
|
|
|
|
|
|
|
|
# Name of the webcomponent
|
|
|
|
component_name = None
|
|
|
|
|
|
|
|
# Icon to show in the sidebar (optional)
|
|
|
|
sidebar_icon = None
|
|
|
|
|
|
|
|
# Title to show in the sidebar (optional)
|
|
|
|
sidebar_title = None
|
|
|
|
|
2017-11-11 07:02:06 +00:00
|
|
|
# Url to the webcomponent (depending on JS version)
|
|
|
|
webcomponent_url_es5 = None
|
|
|
|
webcomponent_url_latest = None
|
2017-10-25 02:36:27 +00:00
|
|
|
|
|
|
|
# Url to show the panel in the frontend
|
|
|
|
frontend_url_path = None
|
|
|
|
|
|
|
|
# Config to pass to the webcomponent
|
|
|
|
config = None
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def async_register(self, hass):
|
|
|
|
"""Register panel with HASS."""
|
|
|
|
panels = hass.data.get(DATA_PANELS)
|
|
|
|
if panels is None:
|
|
|
|
panels = hass.data[DATA_PANELS] = {}
|
|
|
|
|
|
|
|
if self.frontend_url_path in panels:
|
|
|
|
_LOGGER.warning("Overwriting component %s", self.frontend_url_path)
|
|
|
|
|
|
|
|
if DATA_FINALIZE_PANEL in hass.data:
|
|
|
|
yield from hass.data[DATA_FINALIZE_PANEL](self)
|
|
|
|
|
|
|
|
panels[self.frontend_url_path] = self
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_register_index_routes(self, router, index_view):
|
|
|
|
"""Register routes for panel to be served by index view."""
|
|
|
|
router.add_route(
|
|
|
|
'get', '/{}'.format(self.frontend_url_path), index_view.get)
|
|
|
|
router.add_route(
|
|
|
|
'get', '/{}/{{extra:.+}}'.format(self.frontend_url_path),
|
|
|
|
index_view.get)
|
|
|
|
|
|
|
|
|
|
|
|
class BuiltInPanel(AbstractPanel):
|
|
|
|
"""Panel that is part of hass_frontend."""
|
|
|
|
|
|
|
|
def __init__(self, component_name, sidebar_title, sidebar_icon,
|
|
|
|
frontend_url_path, config):
|
|
|
|
"""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
|
|
|
|
|
2018-05-15 18:47:46 +00:00
|
|
|
def to_response(self, hass, request):
|
|
|
|
"""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,
|
|
|
|
}
|
2017-10-25 02:36:27 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ExternalPanel(AbstractPanel):
|
|
|
|
"""Panel that is added by a custom component."""
|
|
|
|
|
|
|
|
REGISTERED_COMPONENTS = set()
|
|
|
|
|
|
|
|
def __init__(self, component_name, path, md5, sidebar_title, sidebar_icon,
|
|
|
|
frontend_url_path, config):
|
|
|
|
"""Initialize an external panel."""
|
|
|
|
self.component_name = component_name
|
|
|
|
self.path = path
|
|
|
|
self.md5 = md5
|
|
|
|
self.sidebar_title = sidebar_title
|
|
|
|
self.sidebar_icon = sidebar_icon
|
|
|
|
self.frontend_url_path = frontend_url_path or component_name
|
|
|
|
self.config = config
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def async_finalize(self, hass, frontend_repository_path):
|
|
|
|
"""Finalize this panel for usage.
|
|
|
|
|
|
|
|
frontend_repository_path is set, will be prepended to path of built-in
|
|
|
|
components.
|
|
|
|
"""
|
|
|
|
try:
|
|
|
|
if self.md5 is None:
|
2017-11-11 07:02:06 +00:00
|
|
|
self.md5 = yield from hass.async_add_job(
|
|
|
|
_fingerprint, self.path)
|
2017-10-25 02:36:27 +00:00
|
|
|
except OSError:
|
|
|
|
_LOGGER.error('Cannot find or access %s at %s',
|
|
|
|
self.component_name, self.path)
|
|
|
|
hass.data[DATA_PANELS].pop(self.frontend_url_path)
|
2017-11-11 07:02:06 +00:00
|
|
|
return
|
2017-10-25 02:36:27 +00:00
|
|
|
|
2017-11-11 07:02:06 +00:00
|
|
|
self.webcomponent_url_es5 = self.webcomponent_url_latest = \
|
2017-10-25 02:36:27 +00:00
|
|
|
URL_PANEL_COMPONENT_FP.format(self.component_name, self.md5)
|
|
|
|
|
|
|
|
if self.component_name not in self.REGISTERED_COMPONENTS:
|
2017-11-01 12:07:16 +00:00
|
|
|
hass.http.register_static_path(
|
2017-11-11 07:02:06 +00:00
|
|
|
self.webcomponent_url_latest, self.path,
|
2017-11-01 12:07:16 +00:00
|
|
|
# if path is None, we're in prod mode, so cache static assets
|
|
|
|
frontend_repository_path is None)
|
2017-10-25 02:36:27 +00:00
|
|
|
self.REGISTERED_COMPONENTS.add(self.component_name)
|
|
|
|
|
2018-05-15 18:47:46 +00:00
|
|
|
def to_response(self, hass, request):
|
|
|
|
"""Panel as dictionary."""
|
|
|
|
result = {
|
|
|
|
'component_name': self.component_name,
|
|
|
|
'icon': self.sidebar_icon,
|
|
|
|
'title': self.sidebar_title,
|
|
|
|
'url_path': self.frontend_url_path,
|
|
|
|
'config': self.config,
|
|
|
|
}
|
|
|
|
if _is_latest(hass.data[DATA_JS_VERSION], request):
|
|
|
|
result['url'] = self.webcomponent_url_latest
|
|
|
|
else:
|
|
|
|
result['url'] = self.webcomponent_url_es5
|
|
|
|
return result
|
|
|
|
|
2017-10-25 02:36:27 +00:00
|
|
|
|
2017-07-22 04:38:53 +00:00
|
|
|
@bind_hass
|
2017-10-25 02:36:27 +00:00
|
|
|
@asyncio.coroutine
|
|
|
|
def async_register_built_in_panel(hass, component_name, sidebar_title=None,
|
|
|
|
sidebar_icon=None, frontend_url_path=None,
|
|
|
|
config=None):
|
2016-07-17 05:32:25 +00:00
|
|
|
"""Register a built-in panel."""
|
2017-10-25 02:36:27 +00:00
|
|
|
panel = BuiltInPanel(component_name, sidebar_title, sidebar_icon,
|
|
|
|
frontend_url_path, config)
|
|
|
|
yield from panel.async_register(hass)
|
2016-07-17 05:32:25 +00:00
|
|
|
|
|
|
|
|
2017-07-22 04:38:53 +00:00
|
|
|
@bind_hass
|
2017-10-25 02:36:27 +00:00
|
|
|
@asyncio.coroutine
|
|
|
|
def async_register_panel(hass, component_name, path, md5=None,
|
|
|
|
sidebar_title=None, sidebar_icon=None,
|
|
|
|
frontend_url_path=None, config=None):
|
2016-07-17 05:32:25 +00:00
|
|
|
"""Register a panel for the frontend.
|
|
|
|
|
|
|
|
component_name: name of the web component
|
|
|
|
path: path to the HTML of the web component
|
2017-08-31 04:21:24 +00:00
|
|
|
(required unless url is provided)
|
2018-01-21 06:35:38 +00:00
|
|
|
md5: the md5 hash of the web component (for versioning in URL, optional)
|
2016-08-08 04:56:17 +00:00
|
|
|
sidebar_title: title to show in the sidebar (optional)
|
|
|
|
sidebar_icon: icon to show next to title in sidebar (optional)
|
2018-01-21 06:35:38 +00:00
|
|
|
url_path: name to use in the URL (defaults to component_name)
|
2016-07-17 05:32:25 +00:00
|
|
|
config: config to be passed into the web component
|
|
|
|
"""
|
2017-10-25 02:36:27 +00:00
|
|
|
panel = ExternalPanel(component_name, path, md5, sidebar_title,
|
|
|
|
sidebar_icon, frontend_url_path, config)
|
|
|
|
yield from panel.async_register(hass)
|
2016-07-17 05:32:25 +00:00
|
|
|
|
2015-06-24 06:22:32 +00:00
|
|
|
|
2017-08-27 16:07:58 +00:00
|
|
|
@bind_hass
|
2017-10-25 02:36:27 +00:00
|
|
|
@callback
|
2017-11-29 06:53:12 +00:00
|
|
|
def add_extra_html_url(hass, url, es5=False):
|
2017-08-27 16:07:58 +00:00
|
|
|
"""Register extra html url to load."""
|
2017-11-29 06:53:12 +00:00
|
|
|
key = DATA_EXTRA_HTML_URL_ES5 if es5 else DATA_EXTRA_HTML_URL
|
|
|
|
url_set = hass.data.get(key)
|
2017-08-27 16:07:58 +00:00
|
|
|
if url_set is None:
|
2017-11-29 06:53:12 +00:00
|
|
|
url_set = hass.data[key] = set()
|
2017-08-27 16:07:58 +00:00
|
|
|
url_set.add(url)
|
|
|
|
|
|
|
|
|
2016-08-14 08:10:07 +00:00
|
|
|
def add_manifest_json_key(key, val):
|
|
|
|
"""Add a keyval to the manifest.json."""
|
|
|
|
MANIFEST_JSON[key] = val
|
|
|
|
|
|
|
|
|
2017-10-25 02:36:27 +00:00
|
|
|
@asyncio.coroutine
|
|
|
|
def async_setup(hass, config):
|
2017-04-30 05:04:49 +00:00
|
|
|
"""Set up the serving of the frontend."""
|
2018-05-10 08:38:11 +00:00
|
|
|
if list(hass.auth.async_auth_providers):
|
|
|
|
client = yield from hass.auth.async_create_client(
|
|
|
|
'Home Assistant Frontend',
|
|
|
|
redirect_uris=['/'],
|
|
|
|
no_secret=True,
|
|
|
|
)
|
|
|
|
else:
|
|
|
|
client = None
|
|
|
|
|
2018-05-01 17:35:23 +00:00
|
|
|
hass.components.websocket_api.async_register_command(
|
|
|
|
WS_TYPE_GET_PANELS, websocket_handle_get_panels, SCHEMA_GET_PANELS)
|
2016-10-24 06:48:01 +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, {})
|
|
|
|
|
|
|
|
repo_path = conf.get(CONF_FRONTEND_REPO)
|
|
|
|
is_dev = repo_path is not None
|
2017-11-11 07:02:06 +00:00
|
|
|
hass.data[DATA_JS_VERSION] = js_version = conf.get(CONF_JS_VERSION)
|
2017-10-25 02:36:27 +00:00
|
|
|
|
|
|
|
if is_dev:
|
2018-05-18 01:29:37 +00:00
|
|
|
hass_frontend_path = os.path.join(repo_path, 'hass_frontend')
|
|
|
|
hass_frontend_es5_path = os.path.join(repo_path, 'hass_frontend_es5')
|
2016-05-10 01:09:38 +00:00
|
|
|
else:
|
2017-10-27 05:28:07 +00:00
|
|
|
import hass_frontend
|
2017-11-11 07:02:06 +00:00
|
|
|
import hass_frontend_es5
|
2018-05-18 01:29:37 +00:00
|
|
|
hass_frontend_path = hass_frontend.where()
|
|
|
|
hass_frontend_es5_path = hass_frontend_es5.where()
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2017-11-11 07:02:06 +00:00
|
|
|
hass.http.register_static_path(
|
2018-05-18 01:29:37 +00:00
|
|
|
"/service_worker_es5.js",
|
|
|
|
os.path.join(hass_frontend_es5_path, "service_worker.js"), False)
|
2017-11-11 07:02:06 +00:00
|
|
|
hass.http.register_static_path(
|
2018-05-18 01:29:37 +00:00
|
|
|
"/service_worker.js",
|
|
|
|
os.path.join(hass_frontend_path, "service_worker.js"), False)
|
2017-11-01 12:07:16 +00:00
|
|
|
hass.http.register_static_path(
|
2018-05-18 01:29:37 +00:00
|
|
|
"/robots.txt",
|
|
|
|
os.path.join(hass_frontend_path, "robots.txt"), False)
|
|
|
|
hass.http.register_static_path("/static", hass_frontend_path, not is_dev)
|
2017-11-11 07:02:06 +00:00
|
|
|
hass.http.register_static_path(
|
2018-05-18 01:29:37 +00:00
|
|
|
"/frontend_latest", hass_frontend_path, not is_dev)
|
2017-11-11 07:02:06 +00:00
|
|
|
hass.http.register_static_path(
|
2018-05-18 01:29:37 +00:00
|
|
|
"/frontend_es5", hass_frontend_es5_path, not is_dev)
|
2016-10-24 06:48:01 +00:00
|
|
|
|
|
|
|
local = hass.config.path('www')
|
|
|
|
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
|
|
|
|
2018-05-10 08:38:11 +00:00
|
|
|
index_view = IndexView(repo_path, js_version, client)
|
2016-11-25 05:37:56 +00:00
|
|
|
hass.http.register_view(index_view)
|
|
|
|
|
2018-05-15 18:47:46 +00:00
|
|
|
async def finalize_panel(panel):
|
2017-10-25 02:36:27 +00:00
|
|
|
"""Finalize setup of a panel."""
|
2018-05-15 18:47:46 +00:00
|
|
|
if hasattr(panel, 'async_finalize'):
|
|
|
|
await panel.async_finalize(hass, repo_path)
|
2017-10-25 02:36:27 +00:00
|
|
|
panel.async_register_index_routes(hass.http.app.router, index_view)
|
2016-11-25 05:37:56 +00:00
|
|
|
|
2017-10-25 02:36:27 +00:00
|
|
|
yield from asyncio.wait([
|
|
|
|
async_register_built_in_panel(hass, panel)
|
|
|
|
for panel in ('dev-event', 'dev-info', 'dev-service', 'dev-state',
|
|
|
|
'dev-template', 'dev-mqtt', 'kiosk')], loop=hass.loop)
|
2017-08-27 16:07:58 +00:00
|
|
|
|
2017-10-25 02:36:27 +00:00
|
|
|
hass.data[DATA_FINALIZE_PANEL] = finalize_panel
|
2016-07-17 05:32:25 +00:00
|
|
|
|
2017-10-25 02:36:27 +00:00
|
|
|
# Finalize registration of panels that registered before frontend was setup
|
|
|
|
# This includes the built-in panels from line above.
|
|
|
|
yield from asyncio.wait(
|
|
|
|
[finalize_panel(panel) for panel in hass.data[DATA_PANELS].values()],
|
|
|
|
loop=hass.loop)
|
2017-07-13 01:08:13 +00:00
|
|
|
|
2017-10-25 02:36:27 +00:00
|
|
|
if DATA_EXTRA_HTML_URL not in hass.data:
|
|
|
|
hass.data[DATA_EXTRA_HTML_URL] = set()
|
2017-11-29 06:53:12 +00:00
|
|
|
if DATA_EXTRA_HTML_URL_ES5 not in hass.data:
|
|
|
|
hass.data[DATA_EXTRA_HTML_URL_ES5] = set()
|
2017-10-25 02:36:27 +00:00
|
|
|
|
|
|
|
for url in conf.get(CONF_EXTRA_HTML_URL, []):
|
2017-11-29 06:53:12 +00:00
|
|
|
add_extra_html_url(hass, url, False)
|
|
|
|
for url in conf.get(CONF_EXTRA_HTML_URL_ES5, []):
|
|
|
|
add_extra_html_url(hass, url, True)
|
2017-08-27 16:07:58 +00:00
|
|
|
|
2018-01-07 22:54:16 +00:00
|
|
|
async_setup_themes(hass, conf.get(CONF_THEMES))
|
2017-10-25 02:36:27 +00:00
|
|
|
|
2018-03-01 03:31:38 +00:00
|
|
|
hass.http.register_view(TranslationsView)
|
|
|
|
|
2015-01-30 07:56:04 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
2017-10-25 02:36:27 +00:00
|
|
|
def async_setup_themes(hass, themes):
|
2017-07-13 01:08:13 +00:00
|
|
|
"""Set up themes data and services."""
|
|
|
|
hass.http.register_view(ThemesView)
|
2017-07-14 18:26:26 +00:00
|
|
|
hass.data[DATA_DEFAULT_THEME] = DEFAULT_THEME
|
|
|
|
if themes is None:
|
|
|
|
hass.data[DATA_THEMES] = {}
|
|
|
|
return
|
|
|
|
|
|
|
|
hass.data[DATA_THEMES] = themes
|
2017-07-13 01:08:13 +00:00
|
|
|
|
|
|
|
@callback
|
|
|
|
def update_theme_and_fire_event():
|
|
|
|
"""Update theme_color in manifest."""
|
|
|
|
name = hass.data[DATA_DEFAULT_THEME]
|
|
|
|
themes = hass.data[DATA_THEMES]
|
|
|
|
if name != DEFAULT_THEME and PRIMARY_COLOR in themes[name]:
|
|
|
|
MANIFEST_JSON['theme_color'] = themes[name][PRIMARY_COLOR]
|
|
|
|
else:
|
|
|
|
MANIFEST_JSON['theme_color'] = DEFAULT_THEME_COLOR
|
|
|
|
hass.bus.async_fire(EVENT_THEMES_UPDATED, {
|
|
|
|
'themes': themes,
|
|
|
|
'default_theme': name,
|
|
|
|
})
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def set_theme(call):
|
2018-01-27 19:58:27 +00:00
|
|
|
"""Set backend-preferred theme."""
|
2017-07-13 01:08:13 +00:00
|
|
|
data = call.data
|
|
|
|
name = data[CONF_NAME]
|
|
|
|
if name == DEFAULT_THEME or name in hass.data[DATA_THEMES]:
|
|
|
|
_LOGGER.info("Theme %s set as default", name)
|
|
|
|
hass.data[DATA_DEFAULT_THEME] = name
|
|
|
|
update_theme_and_fire_event()
|
|
|
|
else:
|
|
|
|
_LOGGER.warning("Theme %s is not defined.", name)
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def reload_themes(_):
|
|
|
|
"""Reload themes."""
|
|
|
|
path = find_config_file(hass.config.config_dir)
|
2017-10-25 02:36:27 +00:00
|
|
|
new_themes = load_yaml_config_file(path)[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
|
|
|
|
update_theme_and_fire_event()
|
|
|
|
|
2018-01-07 22:54:16 +00:00
|
|
|
hass.services.async_register(
|
|
|
|
DOMAIN, SERVICE_SET_THEME, set_theme, schema=SERVICE_SET_THEME_SCHEMA)
|
|
|
|
hass.services.async_register(DOMAIN, SERVICE_RELOAD_THEMES, reload_themes)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
|
|
|
|
|
|
|
class IndexView(HomeAssistantView):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Serve the frontend."""
|
|
|
|
|
2016-05-28 04:45:38 +00:00
|
|
|
url = '/'
|
2017-04-30 05:04:49 +00:00
|
|
|
name = 'frontend:index'
|
2016-05-14 07:58:36 +00:00
|
|
|
requires_auth = False
|
2017-07-07 03:58:21 +00:00
|
|
|
extra_urls = ['/states', '/states/{extra}']
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2018-05-10 08:38:11 +00:00
|
|
|
def __init__(self, repo_path, js_option, client):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Initialize the frontend view."""
|
2017-11-11 23:22:05 +00:00
|
|
|
self.repo_path = repo_path
|
2017-11-11 07:02:06 +00:00
|
|
|
self.js_option = js_option
|
2018-05-10 08:38:11 +00:00
|
|
|
self.client = client
|
2017-11-11 23:22:05 +00:00
|
|
|
self._template_cache = {}
|
|
|
|
|
|
|
|
def get_template(self, latest):
|
|
|
|
"""Get template."""
|
|
|
|
if self.repo_path is not None:
|
|
|
|
root = self.repo_path
|
|
|
|
elif latest:
|
|
|
|
import hass_frontend
|
|
|
|
root = hass_frontend.where()
|
|
|
|
else:
|
|
|
|
import hass_frontend_es5
|
|
|
|
root = hass_frontend_es5.where()
|
|
|
|
|
|
|
|
tpl = self._template_cache.get(root)
|
|
|
|
|
|
|
|
if tpl is None:
|
|
|
|
with open(os.path.join(root, 'index.html')) as file:
|
|
|
|
tpl = jinja2.Template(file.read())
|
|
|
|
|
|
|
|
# Cache template if not running from repository
|
|
|
|
if self.repo_path is None:
|
|
|
|
self._template_cache[root] = tpl
|
|
|
|
|
|
|
|
return tpl
|
2016-05-12 05:55:24 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@asyncio.coroutine
|
2017-07-07 03:58:21 +00:00
|
|
|
def get(self, request, extra=None):
|
2016-05-12 05:55:24 +00:00
|
|
|
"""Serve the index view."""
|
2016-11-25 21:04:06 +00:00
|
|
|
hass = request.app['hass']
|
2017-12-01 20:53:15 +00:00
|
|
|
latest = self.repo_path is not None or \
|
|
|
|
_is_latest(self.js_option, request)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2016-07-20 06:36:46 +00:00
|
|
|
if request.path == '/':
|
|
|
|
panel = 'states'
|
|
|
|
else:
|
|
|
|
panel = request.path.split('/')[1]
|
|
|
|
|
2016-11-25 05:37:56 +00:00
|
|
|
if panel == 'states':
|
|
|
|
panel_url = ''
|
2017-11-11 07:02:06 +00:00
|
|
|
elif latest:
|
|
|
|
panel_url = hass.data[DATA_PANELS][panel].webcomponent_url_latest
|
2016-11-25 05:37:56 +00:00
|
|
|
else:
|
2017-11-11 07:02:06 +00:00
|
|
|
panel_url = hass.data[DATA_PANELS][panel].webcomponent_url_es5
|
2016-07-20 06:36:46 +00:00
|
|
|
|
2017-11-20 14:16:36 +00:00
|
|
|
no_auth = '1'
|
2018-02-15 21:06:14 +00:00
|
|
|
if hass.config.api.api_password and not request[KEY_AUTHENTICATED]:
|
2017-10-25 02:36:27 +00:00
|
|
|
# do not try to auto connect on load
|
2017-11-20 14:16:36 +00:00
|
|
|
no_auth = '0'
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2017-11-11 23:22:05 +00:00
|
|
|
template = yield from hass.async_add_job(self.get_template, latest)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2017-11-29 06:53:12 +00:00
|
|
|
extra_key = DATA_EXTRA_HTML_URL if latest else DATA_EXTRA_HTML_URL_ES5
|
|
|
|
|
2018-05-10 08:38:11 +00:00
|
|
|
template_params = dict(
|
2017-11-11 23:22:05 +00:00
|
|
|
no_auth=no_auth,
|
|
|
|
panel_url=panel_url,
|
|
|
|
panels=hass.data[DATA_PANELS],
|
2017-08-27 16:07:58 +00:00
|
|
|
theme_color=MANIFEST_JSON['theme_color'],
|
2017-11-29 06:53:12 +00:00
|
|
|
extra_urls=hass.data[extra_key],
|
2017-11-11 23:22:05 +00:00
|
|
|
)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2018-05-10 08:38:11 +00:00
|
|
|
if self.client is not None:
|
|
|
|
template_params['client_id'] = self.client.id
|
|
|
|
|
|
|
|
return web.Response(text=template.render(**template_params),
|
|
|
|
content_type='text/html')
|
2016-08-14 08:10:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ManifestJSONView(HomeAssistantView):
|
|
|
|
"""View to return a manifest.json."""
|
|
|
|
|
|
|
|
requires_auth = False
|
2017-04-30 05:04:49 +00:00
|
|
|
url = '/manifest.json'
|
|
|
|
name = 'manifestjson'
|
2016-08-14 08:10:07 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@asyncio.coroutine
|
|
|
|
def get(self, request): # pylint: disable=no-self-use
|
2016-08-14 08:10:07 +00:00
|
|
|
"""Return the manifest.json."""
|
2017-11-11 23:22:05 +00:00
|
|
|
msg = json.dumps(MANIFEST_JSON, sort_keys=True)
|
|
|
|
return web.Response(text=msg, content_type="application/manifest+json")
|
2017-07-13 01:08:13 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ThemesView(HomeAssistantView):
|
|
|
|
"""View to return defined themes."""
|
|
|
|
|
|
|
|
requires_auth = False
|
|
|
|
url = '/api/themes'
|
|
|
|
name = 'api:themes'
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def get(self, request):
|
|
|
|
"""Return themes."""
|
|
|
|
hass = request.app['hass']
|
|
|
|
|
|
|
|
return self.json({
|
|
|
|
'themes': hass.data[DATA_THEMES],
|
|
|
|
'default_theme': hass.data[DATA_DEFAULT_THEME],
|
|
|
|
})
|
2017-10-25 02:36:27 +00:00
|
|
|
|
|
|
|
|
2018-03-01 03:31:38 +00:00
|
|
|
class TranslationsView(HomeAssistantView):
|
|
|
|
"""View to return backend defined translations."""
|
|
|
|
|
|
|
|
url = '/api/translations/{language}'
|
|
|
|
name = 'api:translations'
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def get(self, request, language):
|
|
|
|
"""Return translations."""
|
|
|
|
hass = request.app['hass']
|
|
|
|
|
|
|
|
resources = yield from async_get_translations(hass, language)
|
|
|
|
return self.json({
|
|
|
|
'resources': resources,
|
|
|
|
})
|
|
|
|
|
|
|
|
|
2017-10-25 02:36:27 +00:00
|
|
|
def _fingerprint(path):
|
|
|
|
"""Fingerprint a file."""
|
|
|
|
with open(path) as fil:
|
|
|
|
return hashlib.md5(fil.read().encode('utf-8')).hexdigest()
|
2017-11-11 07:02:06 +00:00
|
|
|
|
|
|
|
|
|
|
|
def _is_latest(js_option, request):
|
|
|
|
"""
|
|
|
|
Return whether we should serve latest untranspiled code.
|
|
|
|
|
|
|
|
Set according to user's preference and URL override.
|
|
|
|
"""
|
2018-02-28 21:26:49 +00:00
|
|
|
import hass_frontend
|
|
|
|
|
2017-11-11 07:02:06 +00:00
|
|
|
if request is None:
|
|
|
|
return js_option == 'latest'
|
2017-11-29 06:53:12 +00:00
|
|
|
|
|
|
|
# latest in query
|
|
|
|
if 'latest' in request.query or (
|
|
|
|
request.headers.get('Referer') and
|
|
|
|
'latest' in urlparse(request.headers['Referer']).query):
|
|
|
|
return True
|
|
|
|
|
|
|
|
# es5 in query
|
|
|
|
if 'es5' in request.query or (
|
|
|
|
request.headers.get('Referer') and
|
|
|
|
'es5' in urlparse(request.headers['Referer']).query):
|
|
|
|
return False
|
|
|
|
|
|
|
|
# non-auto option in config
|
|
|
|
if js_option != 'auto':
|
|
|
|
return js_option == 'latest'
|
|
|
|
|
2018-01-03 00:42:41 +00:00
|
|
|
useragent = request.headers.get('User-Agent')
|
|
|
|
|
2018-02-28 21:26:49 +00:00
|
|
|
return useragent and hass_frontend.version(useragent)
|
2018-05-01 17:35:23 +00:00
|
|
|
|
|
|
|
|
2018-05-03 20:02:59 +00:00
|
|
|
@callback
|
2018-05-01 17:35:23 +00:00
|
|
|
def websocket_handle_get_panels(hass, connection, msg):
|
|
|
|
"""Handle get panels command.
|
|
|
|
|
|
|
|
Async friendly.
|
|
|
|
"""
|
|
|
|
panels = {
|
|
|
|
panel:
|
|
|
|
connection.hass.data[DATA_PANELS][panel].to_response(
|
|
|
|
connection.hass, connection.request)
|
|
|
|
for panel in connection.hass.data[DATA_PANELS]}
|
|
|
|
|
|
|
|
connection.to_write.put_nowait(websocket_api.result_message(
|
|
|
|
msg['id'], panels))
|