2016-03-09 10:15:04 +00:00
|
|
|
"""Handle the frontend for Home Assistant."""
|
2016-07-17 05:32:25 +00:00
|
|
|
import logging
|
2015-01-30 07:56:04 +00:00
|
|
|
import os
|
|
|
|
|
2016-07-17 05:32:25 +00:00
|
|
|
from homeassistant.const import EVENT_HOMEASSISTANT_START
|
2016-02-14 08:04:08 +00:00
|
|
|
from homeassistant.components import api
|
2016-05-14 07:58:36 +00:00
|
|
|
from homeassistant.components.http import HomeAssistantView
|
2016-07-17 05:32:25 +00:00
|
|
|
from .version import FINGERPRINTS
|
2015-01-30 07:56:04 +00:00
|
|
|
|
|
|
|
DOMAIN = 'frontend'
|
|
|
|
DEPENDENCIES = ['api']
|
2016-07-17 05:32:25 +00:00
|
|
|
URL_PANEL_COMPONENT = '/frontend/panels/{}.html'
|
|
|
|
URL_PANEL_COMPONENT_FP = '/frontend/panels/{}-{}.html'
|
|
|
|
STATIC_PATH = os.path.join(os.path.dirname(__file__), 'www_static')
|
2016-07-20 06:36:46 +00:00
|
|
|
PANELS = {}
|
|
|
|
|
|
|
|
# To keep track we don't register a component twice (gives a warning)
|
|
|
|
_REGISTERED_COMPONENTS = set()
|
2016-07-17 05:32:25 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
def register_built_in_panel(hass, component_name, title=None, icon=None,
|
|
|
|
url_name=None, config=None):
|
|
|
|
"""Register a built-in panel."""
|
2016-07-17 06:45:38 +00:00
|
|
|
# pylint: disable=too-many-arguments
|
2016-07-17 05:32:25 +00:00
|
|
|
path = 'panels/ha-panel-{}.html'.format(component_name)
|
|
|
|
|
|
|
|
register_panel(hass, component_name, os.path.join(STATIC_PATH, path),
|
|
|
|
FINGERPRINTS[path], title, icon, url_name, config)
|
|
|
|
|
|
|
|
|
|
|
|
def register_panel(hass, component_name, path, md5, title=None, icon=None,
|
|
|
|
url_name=None, config=None):
|
|
|
|
"""Register a panel for the frontend.
|
|
|
|
|
|
|
|
component_name: name of the web component
|
|
|
|
path: path to the HTML of the web component
|
|
|
|
md5: the md5 hash of the web component (for versioning)
|
|
|
|
title: title to show in the sidebar (optional)
|
|
|
|
icon: icon to show next to title in sidebar (optional)
|
|
|
|
url_name: name to use in the url (defaults to component_name)
|
|
|
|
config: config to be passed into the web component
|
|
|
|
|
|
|
|
Warning: this API will probably change. Use at own risk.
|
|
|
|
"""
|
2016-07-17 06:45:38 +00:00
|
|
|
# pylint: disable=too-many-arguments
|
2016-07-17 05:32:25 +00:00
|
|
|
if url_name is None:
|
|
|
|
url_name = component_name
|
|
|
|
|
|
|
|
if url_name in PANELS:
|
|
|
|
_LOGGER.warning('Overwriting component %s', url_name)
|
|
|
|
if not os.path.isfile(path):
|
|
|
|
_LOGGER.warning('Panel %s component does not exist: %s',
|
|
|
|
component_name, path)
|
|
|
|
|
|
|
|
data = {
|
|
|
|
'url_name': url_name,
|
|
|
|
'component_name': component_name,
|
|
|
|
}
|
|
|
|
|
|
|
|
if title:
|
|
|
|
data['title'] = title
|
|
|
|
if icon:
|
|
|
|
data['icon'] = icon
|
|
|
|
if config is not None:
|
|
|
|
data['config'] = config
|
|
|
|
|
|
|
|
if hass.wsgi.development:
|
|
|
|
data['url'] = ('/static/home-assistant-polymer/panels/'
|
|
|
|
'{0}/ha-panel-{0}.html'.format(component_name))
|
|
|
|
else:
|
|
|
|
url = URL_PANEL_COMPONENT.format(component_name)
|
2016-07-20 06:36:46 +00:00
|
|
|
|
|
|
|
if url not in _REGISTERED_COMPONENTS:
|
|
|
|
hass.wsgi.register_static_path(url, path)
|
|
|
|
_REGISTERED_COMPONENTS.add(url)
|
|
|
|
|
2016-07-17 05:32:25 +00:00
|
|
|
fprinted_url = URL_PANEL_COMPONENT_FP.format(component_name, md5)
|
|
|
|
data['url'] = fprinted_url
|
|
|
|
|
|
|
|
PANELS[url_name] = data
|
|
|
|
|
2015-06-24 06:22:32 +00:00
|
|
|
|
2015-01-30 07:56:04 +00:00
|
|
|
def setup(hass, config):
|
2016-03-09 10:15:04 +00:00
|
|
|
"""Setup serving the frontend."""
|
2016-05-10 01:09:38 +00:00
|
|
|
hass.wsgi.register_view(BootstrapView)
|
|
|
|
|
|
|
|
if hass.wsgi.development:
|
|
|
|
sw_path = "home-assistant-polymer/build/service_worker.js"
|
|
|
|
else:
|
|
|
|
sw_path = "service_worker.js"
|
|
|
|
|
2016-07-17 05:32:25 +00:00
|
|
|
hass.wsgi.register_static_path("/service_worker.js",
|
|
|
|
os.path.join(STATIC_PATH, sw_path), 0)
|
|
|
|
hass.wsgi.register_static_path("/robots.txt",
|
|
|
|
os.path.join(STATIC_PATH, "robots.txt"))
|
|
|
|
hass.wsgi.register_static_path("/static", STATIC_PATH)
|
2016-05-10 01:09:38 +00:00
|
|
|
hass.wsgi.register_static_path("/local", hass.config.path('www'))
|
|
|
|
|
2016-07-17 05:32:25 +00:00
|
|
|
register_built_in_panel(hass, 'map', 'Map', 'mdi:account-location')
|
|
|
|
|
|
|
|
for panel in ('dev-event', 'dev-info', 'dev-service', 'dev-state',
|
|
|
|
'dev-template'):
|
|
|
|
register_built_in_panel(hass, panel)
|
|
|
|
|
|
|
|
def register_frontend_index(event):
|
|
|
|
"""Register the frontend index urls.
|
|
|
|
|
|
|
|
Done when Home Assistant is started so that all panels are known.
|
|
|
|
"""
|
|
|
|
hass.wsgi.register_view(IndexView(
|
|
|
|
hass, ['/{}'.format(name) for name in PANELS]))
|
|
|
|
|
|
|
|
hass.bus.listen_once(EVENT_HOMEASSISTANT_START, register_frontend_index)
|
|
|
|
|
2015-01-30 07:56:04 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
class BootstrapView(HomeAssistantView):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""View to bootstrap frontend with all needed data."""
|
|
|
|
|
2016-05-28 04:45:38 +00:00
|
|
|
url = "/api/bootstrap"
|
2016-05-10 01:09:38 +00:00
|
|
|
name = "api:bootstrap"
|
|
|
|
|
|
|
|
def get(self, request):
|
|
|
|
"""Return all data needed to bootstrap Home Assistant."""
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json({
|
2016-05-10 01:09:38 +00:00
|
|
|
'config': self.hass.config.as_dict(),
|
|
|
|
'states': self.hass.states.all(),
|
|
|
|
'events': api.events_json(self.hass),
|
|
|
|
'services': api.services_json(self.hass),
|
2016-07-17 05:32:25 +00:00
|
|
|
'panels': PANELS,
|
2016-05-14 07:58:36 +00:00
|
|
|
})
|
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 = '/'
|
2016-05-10 01:09:38 +00:00
|
|
|
name = "frontend:index"
|
2016-05-14 07:58:36 +00:00
|
|
|
requires_auth = False
|
2016-07-17 05:32:25 +00:00
|
|
|
extra_urls = ['/states', '/states/<entity:entity_id>']
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2016-07-17 05:32:25 +00:00
|
|
|
def __init__(self, hass, extra_urls):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Initialize the frontend view."""
|
2016-05-12 05:55:24 +00:00
|
|
|
super().__init__(hass)
|
|
|
|
|
|
|
|
from jinja2 import FileSystemLoader, Environment
|
|
|
|
|
2016-07-17 05:32:25 +00:00
|
|
|
self.extra_urls = self.extra_urls + extra_urls
|
2016-05-14 07:58:36 +00:00
|
|
|
self.templates = Environment(
|
2016-05-12 05:55:24 +00:00
|
|
|
loader=FileSystemLoader(
|
|
|
|
os.path.join(os.path.dirname(__file__), 'templates/')
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
2016-05-14 22:07:20 +00:00
|
|
|
def get(self, request, entity_id=None):
|
2016-05-12 05:55:24 +00:00
|
|
|
"""Serve the index view."""
|
2016-05-14 22:07:20 +00:00
|
|
|
if self.hass.wsgi.development:
|
2016-07-17 05:32:25 +00:00
|
|
|
core_url = '/static/home-assistant-polymer/build/core.js'
|
2016-07-12 07:10:05 +00:00
|
|
|
ui_url = '/static/home-assistant-polymer/src/home-assistant.html'
|
2016-05-14 22:07:20 +00:00
|
|
|
else:
|
2016-07-17 05:32:25 +00:00
|
|
|
core_url = '/static/core-{}.js'.format(
|
|
|
|
FINGERPRINTS['core.js'])
|
|
|
|
ui_url = '/static/frontend-{}.html'.format(
|
|
|
|
FINGERPRINTS['frontend.html'])
|
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]
|
|
|
|
|
|
|
|
panel_url = PANELS[panel]['url'] if panel != 'states' else ''
|
|
|
|
|
2016-05-28 04:45:38 +00:00
|
|
|
# auto login if no password was set
|
2016-07-17 05:32:25 +00:00
|
|
|
no_auth = 'false' if self.hass.config.api.api_password else 'true'
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2016-07-17 05:32:25 +00:00
|
|
|
icons_url = '/static/mdi-{}.html'.format(FINGERPRINTS['mdi.html'])
|
2016-05-14 07:58:36 +00:00
|
|
|
template = self.templates.get_template('index.html')
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2016-05-16 07:25:47 +00:00
|
|
|
# pylint is wrong
|
|
|
|
# pylint: disable=no-member
|
2016-05-29 01:38:46 +00:00
|
|
|
resp = template.render(
|
2016-07-17 05:32:25 +00:00
|
|
|
core_url=core_url, ui_url=ui_url, no_auth=no_auth,
|
2016-07-20 06:36:46 +00:00
|
|
|
icons_url=icons_url, icons=FINGERPRINTS['mdi.html'],
|
|
|
|
panel_url=panel_url)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2016-05-29 01:38:46 +00:00
|
|
|
return self.Response(resp, mimetype='text/html')
|