2015-01-30 16:26:06 +00:00
|
|
|
"""
|
|
|
|
homeassistant.components.frontend
|
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
|
|
|
Provides a frontend for Home Assistant.
|
|
|
|
"""
|
2015-01-30 07:56:04 +00:00
|
|
|
import re
|
|
|
|
import os
|
2015-01-30 16:26:06 +00:00
|
|
|
import logging
|
2015-01-30 07:56:04 +00:00
|
|
|
|
2015-01-30 16:26:06 +00:00
|
|
|
from . import version
|
2015-01-30 07:56:04 +00:00
|
|
|
import homeassistant.util as util
|
|
|
|
|
|
|
|
DOMAIN = 'frontend'
|
|
|
|
DEPENDENCIES = ['api']
|
|
|
|
|
|
|
|
HTTP_OK = 200
|
|
|
|
HTTP_CREATED = 201
|
|
|
|
HTTP_MOVED_PERMANENTLY = 301
|
|
|
|
HTTP_BAD_REQUEST = 400
|
|
|
|
HTTP_UNAUTHORIZED = 401
|
|
|
|
HTTP_NOT_FOUND = 404
|
|
|
|
HTTP_METHOD_NOT_ALLOWED = 405
|
|
|
|
HTTP_UNPROCESSABLE_ENTITY = 422
|
|
|
|
|
|
|
|
|
|
|
|
URL_ROOT = "/"
|
|
|
|
|
2015-01-30 16:26:06 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2015-01-30 07:56:04 +00:00
|
|
|
|
|
|
|
def setup(hass, config):
|
|
|
|
""" Setup serving the frontend. """
|
|
|
|
if 'http' not in hass.components:
|
2015-01-30 16:26:06 +00:00
|
|
|
_LOGGER.error('Dependency http is not loaded')
|
2015-01-30 07:56:04 +00:00
|
|
|
return False
|
|
|
|
|
|
|
|
hass.http.register_path('GET', URL_ROOT, _handle_get_root, False)
|
|
|
|
|
|
|
|
# Static files
|
|
|
|
hass.http.register_path(
|
|
|
|
'GET', re.compile(r'/static/(?P<file>[a-zA-Z\._\-0-9/]+)'),
|
|
|
|
_handle_get_static, False)
|
|
|
|
hass.http.register_path(
|
|
|
|
'HEAD', re.compile(r'/static/(?P<file>[a-zA-Z\._\-0-9/]+)'),
|
|
|
|
_handle_get_static, False)
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
def _handle_get_root(handler, path_match, data):
|
|
|
|
""" Renders the debug interface. """
|
|
|
|
|
2015-02-07 23:54:58 +00:00
|
|
|
def write(txt):
|
|
|
|
""" Helper to write text to the output. """
|
|
|
|
handler.wfile.write((txt + "\n").encode("UTF-8"))
|
2015-01-30 07:56:04 +00:00
|
|
|
|
|
|
|
handler.send_response(HTTP_OK)
|
|
|
|
handler.send_header('Content-type', 'text/html; charset=utf-8')
|
|
|
|
handler.end_headers()
|
|
|
|
|
|
|
|
if handler.server.development:
|
2015-02-04 07:16:53 +00:00
|
|
|
app_url = "polymer/home-assistant.html"
|
2015-01-30 07:56:04 +00:00
|
|
|
else:
|
2015-01-30 16:26:06 +00:00
|
|
|
app_url = "frontend-{}.html".format(version.VERSION)
|
2015-01-30 07:56:04 +00:00
|
|
|
|
|
|
|
# auto login if no password was set, else check api_password param
|
|
|
|
auth = (handler.server.api_password if handler.server.no_password_set
|
|
|
|
else data.get('api_password', ''))
|
|
|
|
|
|
|
|
write(("<!doctype html>"
|
|
|
|
"<html>"
|
|
|
|
"<head><title>Home Assistant</title>"
|
|
|
|
"<meta name='mobile-web-app-capable' content='yes'>"
|
|
|
|
"<link rel='shortcut icon' href='/static/favicon.ico' />"
|
|
|
|
"<link rel='icon' type='image/png' "
|
|
|
|
" href='/static/favicon-192x192.png' sizes='192x192'>"
|
|
|
|
"<meta name='viewport' content='width=device-width, "
|
|
|
|
" user-scalable=no, initial-scale=1.0, "
|
|
|
|
" minimum-scale=1.0, maximum-scale=1.0' />"
|
|
|
|
"<meta name='theme-color' content='#03a9f4'>"
|
|
|
|
"</head>"
|
|
|
|
"<body fullbleed>"
|
|
|
|
"<h3 id='init' align='center'>Initializing Home Assistant</h3>"
|
|
|
|
"<script"
|
|
|
|
" src='/static/webcomponents.min.js'></script>"
|
|
|
|
"<link rel='import' href='/static/{}' />"
|
2015-02-04 07:16:53 +00:00
|
|
|
"<home-assistant auth='{}'></home-assistant>"
|
2015-01-30 07:56:04 +00:00
|
|
|
"</body></html>").format(app_url, auth))
|
|
|
|
|
|
|
|
|
|
|
|
def _handle_get_static(handler, path_match, data):
|
2015-01-30 16:26:06 +00:00
|
|
|
""" Returns a static file for the frontend. """
|
2015-01-30 07:56:04 +00:00
|
|
|
req_file = util.sanitize_path(path_match.group('file'))
|
|
|
|
|
|
|
|
# Strip md5 hash out of frontend filename
|
|
|
|
if re.match(r'^frontend-[A-Za-z0-9]{32}\.html$', req_file):
|
|
|
|
req_file = "frontend.html"
|
|
|
|
|
|
|
|
path = os.path.join(os.path.dirname(__file__), 'www_static', req_file)
|
|
|
|
|
2015-01-30 16:26:06 +00:00
|
|
|
handler.write_file(path)
|