2013-09-25 01:39:58 +00:00
|
|
|
"""
|
2013-12-11 08:07:30 +00:00
|
|
|
homeassistant.components.httpinterface
|
2015-05-13 17:18:30 +00:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
2013-09-25 01:39:58 +00:00
|
|
|
|
2013-09-28 18:09:36 +00:00
|
|
|
This module provides an API and a HTTP interface for debug purposes.
|
|
|
|
|
2013-10-29 07:22:38 +00:00
|
|
|
By default it will run on port 8123.
|
2013-09-28 18:09:36 +00:00
|
|
|
|
2013-10-29 07:22:38 +00:00
|
|
|
All API calls have to be accompanied by an 'api_password' parameter and will
|
|
|
|
return JSON. If successful calls will return status code 200 or 201.
|
|
|
|
|
|
|
|
Other status codes that can occur are:
|
|
|
|
- 400 (Bad Request)
|
|
|
|
- 401 (Unauthorized)
|
|
|
|
- 404 (Not Found)
|
|
|
|
- 405 (Method not allowed)
|
2013-09-28 18:09:36 +00:00
|
|
|
|
|
|
|
The api supports the following actions:
|
|
|
|
|
2014-05-02 06:03:14 +00:00
|
|
|
/api - GET
|
|
|
|
Returns message if API is up and running.
|
|
|
|
Example result:
|
|
|
|
{
|
|
|
|
"message": "API running."
|
|
|
|
}
|
|
|
|
|
2013-10-29 07:22:38 +00:00
|
|
|
/api/states - GET
|
2014-01-20 07:37:40 +00:00
|
|
|
Returns a list of entities for which a state is available
|
2013-10-29 07:22:38 +00:00
|
|
|
Example result:
|
2014-10-17 07:17:02 +00:00
|
|
|
[
|
|
|
|
{ .. state object .. },
|
|
|
|
{ .. state object .. }
|
|
|
|
]
|
2013-10-29 07:22:38 +00:00
|
|
|
|
2014-01-20 07:37:40 +00:00
|
|
|
/api/states/<entity_id> - GET
|
|
|
|
Returns the current state from an entity
|
2013-10-29 07:22:38 +00:00
|
|
|
Example result:
|
|
|
|
{
|
|
|
|
"attributes": {
|
|
|
|
"next_rising": "07:04:15 29-10-2013",
|
|
|
|
"next_setting": "18:00:31 29-10-2013"
|
|
|
|
},
|
2014-01-20 07:37:40 +00:00
|
|
|
"entity_id": "weather.sun",
|
2013-10-29 07:22:38 +00:00
|
|
|
"last_changed": "23:24:33 28-10-2013",
|
|
|
|
"state": "below_horizon"
|
|
|
|
}
|
|
|
|
|
2014-01-20 07:37:40 +00:00
|
|
|
/api/states/<entity_id> - POST
|
|
|
|
Updates the current state of an entity. Returns status code 201 if successful
|
2013-11-01 18:34:43 +00:00
|
|
|
with location header of updated resource and as body the new state.
|
2013-09-28 18:09:36 +00:00
|
|
|
parameter: new_state - string
|
2013-10-29 07:22:38 +00:00
|
|
|
optional parameter: attributes - JSON encoded object
|
2013-11-01 18:34:43 +00:00
|
|
|
Example result:
|
|
|
|
{
|
|
|
|
"attributes": {
|
|
|
|
"next_rising": "07:04:15 29-10-2013",
|
|
|
|
"next_setting": "18:00:31 29-10-2013"
|
|
|
|
},
|
2014-01-20 07:37:40 +00:00
|
|
|
"entity_id": "weather.sun",
|
2013-11-01 18:34:43 +00:00
|
|
|
"last_changed": "23:24:33 28-10-2013",
|
|
|
|
"state": "below_horizon"
|
|
|
|
}
|
2013-09-28 18:09:36 +00:00
|
|
|
|
2013-10-29 07:22:38 +00:00
|
|
|
/api/events/<event_type> - POST
|
|
|
|
Fires an event with event_type
|
|
|
|
optional parameter: event_data - JSON encoded object
|
|
|
|
Example result:
|
|
|
|
{
|
|
|
|
"message": "Event download_file fired."
|
|
|
|
}
|
2013-09-25 01:39:58 +00:00
|
|
|
|
|
|
|
"""
|
|
|
|
|
2013-09-28 18:09:36 +00:00
|
|
|
import json
|
2013-09-20 06:59:49 +00:00
|
|
|
import threading
|
2013-09-22 00:59:31 +00:00
|
|
|
import logging
|
2015-01-30 16:26:06 +00:00
|
|
|
import time
|
|
|
|
import gzip
|
|
|
|
import os
|
2015-05-18 13:54:32 +00:00
|
|
|
import random
|
|
|
|
import string
|
|
|
|
from datetime import timedelta
|
|
|
|
from homeassistant.util import Throttle
|
2014-10-27 01:10:01 +00:00
|
|
|
from http.server import SimpleHTTPRequestHandler, HTTPServer
|
2015-05-18 13:54:32 +00:00
|
|
|
from http import cookies
|
2014-04-29 07:30:31 +00:00
|
|
|
from socketserver import ThreadingMixIn
|
2014-04-14 07:10:24 +00:00
|
|
|
from urllib.parse import urlparse, parse_qs
|
2013-09-20 06:59:49 +00:00
|
|
|
|
2015-08-17 03:44:46 +00:00
|
|
|
import homeassistant.core as ha
|
2015-02-14 02:27:13 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
SERVER_PORT, CONTENT_TYPE_JSON,
|
|
|
|
HTTP_HEADER_HA_AUTH, HTTP_HEADER_CONTENT_TYPE, HTTP_HEADER_ACCEPT_ENCODING,
|
|
|
|
HTTP_HEADER_CONTENT_ENCODING, HTTP_HEADER_VARY, HTTP_HEADER_CONTENT_LENGTH,
|
|
|
|
HTTP_HEADER_CACHE_CONTROL, HTTP_HEADER_EXPIRES, HTTP_OK, HTTP_UNAUTHORIZED,
|
|
|
|
HTTP_NOT_FOUND, HTTP_METHOD_NOT_ALLOWED, HTTP_UNPROCESSABLE_ENTITY)
|
2014-04-29 07:30:31 +00:00
|
|
|
import homeassistant.remote as rem
|
2013-11-04 23:18:39 +00:00
|
|
|
import homeassistant.util as util
|
2015-05-18 13:54:32 +00:00
|
|
|
import homeassistant.util.dt as date_util
|
2015-01-31 17:52:03 +00:00
|
|
|
import homeassistant.bootstrap as bootstrap
|
2014-10-27 01:10:01 +00:00
|
|
|
|
2014-08-13 12:28:45 +00:00
|
|
|
DOMAIN = "http"
|
|
|
|
DEPENDENCIES = []
|
2013-09-20 06:59:49 +00:00
|
|
|
|
2014-08-13 12:28:45 +00:00
|
|
|
CONF_API_PASSWORD = "api_password"
|
|
|
|
CONF_SERVER_HOST = "server_host"
|
|
|
|
CONF_SERVER_PORT = "server_port"
|
2014-10-25 05:59:36 +00:00
|
|
|
CONF_DEVELOPMENT = "development"
|
2015-05-18 13:54:32 +00:00
|
|
|
CONF_SESSIONS_ENABLED = "sessions_enabled"
|
2014-08-13 12:28:45 +00:00
|
|
|
|
2014-11-29 06:27:44 +00:00
|
|
|
DATA_API_PASSWORD = 'api_password'
|
|
|
|
|
2015-05-18 13:54:32 +00:00
|
|
|
# Throttling time in seconds for expired sessions check
|
|
|
|
MIN_SEC_SESSION_CLEARING = timedelta(seconds=20)
|
|
|
|
SESSION_TIMEOUT_SECONDS = 1800
|
|
|
|
SESSION_KEY = 'sessionId'
|
|
|
|
|
2014-11-08 21:57:08 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2014-11-23 20:57:29 +00:00
|
|
|
|
2015-01-18 05:55:33 +00:00
|
|
|
def setup(hass, config=None):
|
2014-04-29 07:30:31 +00:00
|
|
|
""" Sets up the HTTP API and debug interface. """
|
2015-01-18 05:55:33 +00:00
|
|
|
if config is None or DOMAIN not in config:
|
|
|
|
config = {DOMAIN: {}}
|
2014-08-13 12:28:45 +00:00
|
|
|
|
2015-03-08 21:21:35 +00:00
|
|
|
api_password = util.convert(config[DOMAIN].get(CONF_API_PASSWORD), str)
|
2015-01-18 05:55:33 +00:00
|
|
|
|
|
|
|
no_password_set = api_password is None
|
|
|
|
|
|
|
|
if no_password_set:
|
|
|
|
api_password = util.get_random_string()
|
2014-04-29 07:30:31 +00:00
|
|
|
|
|
|
|
# If no server host is given, accept all incoming requests
|
2014-08-13 12:28:45 +00:00
|
|
|
server_host = config[DOMAIN].get(CONF_SERVER_HOST, '0.0.0.0')
|
|
|
|
|
2014-12-07 07:57:02 +00:00
|
|
|
server_port = config[DOMAIN].get(CONF_SERVER_PORT, SERVER_PORT)
|
2013-09-22 00:59:31 +00:00
|
|
|
|
2015-03-06 04:43:20 +00:00
|
|
|
development = str(config[DOMAIN].get(CONF_DEVELOPMENT, "")) == "1"
|
2014-10-25 05:59:36 +00:00
|
|
|
|
2015-05-18 17:57:35 +00:00
|
|
|
sessions_enabled = config[DOMAIN].get(CONF_SESSIONS_ENABLED, True)
|
2015-05-18 13:54:32 +00:00
|
|
|
|
2015-07-26 07:14:55 +00:00
|
|
|
try:
|
|
|
|
server = HomeAssistantHTTPServer(
|
|
|
|
(server_host, server_port), RequestHandler, hass, api_password,
|
|
|
|
development, no_password_set, sessions_enabled)
|
|
|
|
except OSError:
|
|
|
|
# Happens if address already in use
|
|
|
|
_LOGGER.exception("Error setting up HTTP server")
|
|
|
|
return False
|
2013-09-22 00:59:31 +00:00
|
|
|
|
2014-11-29 07:19:59 +00:00
|
|
|
hass.bus.listen_once(
|
2014-04-29 07:30:31 +00:00
|
|
|
ha.EVENT_HOMEASSISTANT_START,
|
|
|
|
lambda event:
|
|
|
|
threading.Thread(target=server.start, daemon=True).start())
|
2013-10-09 02:00:10 +00:00
|
|
|
|
2015-01-30 07:56:04 +00:00
|
|
|
hass.http = server
|
2015-03-22 04:10:46 +00:00
|
|
|
hass.config.api = rem.API(util.get_local_ip(), api_password, server_port)
|
2014-05-02 06:03:14 +00:00
|
|
|
|
2014-08-13 12:28:45 +00:00
|
|
|
return True
|
|
|
|
|
2015-05-18 14:08:02 +00:00
|
|
|
|
2015-05-18 13:54:32 +00:00
|
|
|
# pylint: disable=too-many-instance-attributes
|
2014-04-29 07:30:31 +00:00
|
|
|
class HomeAssistantHTTPServer(ThreadingMixIn, HTTPServer):
|
|
|
|
""" Handle HTTP requests in a threaded fashion. """
|
2014-11-23 20:57:29 +00:00
|
|
|
# pylint: disable=too-few-public-methods
|
2013-09-28 18:09:36 +00:00
|
|
|
|
2014-11-23 17:51:16 +00:00
|
|
|
allow_reuse_address = True
|
2014-11-23 20:57:29 +00:00
|
|
|
daemon_threads = True
|
2014-11-23 17:51:16 +00:00
|
|
|
|
2014-10-25 06:44:00 +00:00
|
|
|
# pylint: disable=too-many-arguments
|
2014-12-07 09:28:52 +00:00
|
|
|
def __init__(self, server_address, request_handler_class,
|
2015-05-18 13:54:32 +00:00
|
|
|
hass, api_password, development, no_password_set,
|
|
|
|
sessions_enabled):
|
2014-12-07 09:28:52 +00:00
|
|
|
super().__init__(server_address, request_handler_class)
|
2013-09-22 00:59:31 +00:00
|
|
|
|
2014-10-25 05:59:36 +00:00
|
|
|
self.server_address = server_address
|
2014-04-29 07:30:31 +00:00
|
|
|
self.hass = hass
|
|
|
|
self.api_password = api_password
|
2014-10-25 05:59:36 +00:00
|
|
|
self.development = development
|
2015-01-18 05:55:33 +00:00
|
|
|
self.no_password_set = no_password_set
|
2015-01-30 07:56:04 +00:00
|
|
|
self.paths = []
|
2015-05-18 17:57:35 +00:00
|
|
|
self.sessions = SessionStore(sessions_enabled)
|
2013-09-22 00:59:31 +00:00
|
|
|
|
2014-04-29 07:30:31 +00:00
|
|
|
# We will lazy init this one if needed
|
|
|
|
self.event_forwarder = None
|
2013-09-22 00:59:31 +00:00
|
|
|
|
2014-10-25 05:59:36 +00:00
|
|
|
if development:
|
2015-01-30 07:56:04 +00:00
|
|
|
_LOGGER.info("running http in development mode")
|
2014-10-25 05:59:36 +00:00
|
|
|
|
2014-04-29 07:30:31 +00:00
|
|
|
def start(self):
|
2015-08-03 15:05:33 +00:00
|
|
|
""" Starts the HTTP server. """
|
|
|
|
def stop_http(event):
|
|
|
|
""" Stops the HTTP server. """
|
|
|
|
self.shutdown()
|
|
|
|
|
|
|
|
self.hass.bus.listen_once(ha.EVENT_HOMEASSISTANT_STOP, stop_http)
|
2014-11-29 04:22:08 +00:00
|
|
|
|
2014-11-08 21:57:08 +00:00
|
|
|
_LOGGER.info(
|
|
|
|
"Starting web interface at http://%s:%d", *self.server_address)
|
2014-04-29 07:30:31 +00:00
|
|
|
|
2015-01-31 17:52:03 +00:00
|
|
|
# 31-1-2015: Refactored frontend/api components out of this component
|
|
|
|
# To prevent stuff from breaking, load the two extracted components
|
|
|
|
bootstrap.setup_component(self.hass, 'api')
|
|
|
|
bootstrap.setup_component(self.hass, 'frontend')
|
|
|
|
|
2014-04-29 07:30:31 +00:00
|
|
|
self.serve_forever()
|
2013-09-22 00:59:31 +00:00
|
|
|
|
2015-01-30 07:56:04 +00:00
|
|
|
def register_path(self, method, url, callback, require_auth=True):
|
2015-08-20 03:09:13 +00:00
|
|
|
""" Registers a path with the server. """
|
2015-01-30 07:56:04 +00:00
|
|
|
self.paths.append((method, url, callback, require_auth))
|
|
|
|
|
2015-09-01 06:12:00 +00:00
|
|
|
def log_message(self, fmt, *args):
|
|
|
|
""" Redirect built-in log to HA logging """
|
|
|
|
# pylint: disable=no-self-use
|
|
|
|
_LOGGER.info(fmt, *args)
|
|
|
|
|
2013-11-11 00:46:48 +00:00
|
|
|
|
2015-01-31 17:52:03 +00:00
|
|
|
# pylint: disable=too-many-public-methods,too-many-locals
|
2014-10-27 01:10:01 +00:00
|
|
|
class RequestHandler(SimpleHTTPRequestHandler):
|
|
|
|
"""
|
|
|
|
Handles incoming HTTP requests
|
|
|
|
|
|
|
|
We extend from SimpleHTTPRequestHandler instead of Base so we
|
|
|
|
can use the guess content type methods.
|
|
|
|
"""
|
|
|
|
|
|
|
|
server_version = "HomeAssistant/1.0"
|
2013-09-20 06:59:49 +00:00
|
|
|
|
2015-05-18 13:54:32 +00:00
|
|
|
def __init__(self, req, client_addr, server):
|
|
|
|
""" Contructor, call the base constructor and set up session """
|
|
|
|
self._session = None
|
2015-05-20 07:13:13 +00:00
|
|
|
SimpleHTTPRequestHandler.__init__(self, req, client_addr, server)
|
2015-05-18 13:54:32 +00:00
|
|
|
|
2015-09-01 06:12:00 +00:00
|
|
|
def log_message(self, fmt, *arguments):
|
|
|
|
""" Redirect built-in log to HA logging """
|
2015-09-24 03:56:34 +00:00
|
|
|
if self.server.no_password_set:
|
|
|
|
_LOGGER.info(fmt, *arguments)
|
|
|
|
else:
|
|
|
|
_LOGGER.info(
|
|
|
|
fmt, *(arg.replace(self.server.api_password, '*******')
|
|
|
|
if isinstance(arg, str) else arg for arg in arguments))
|
2015-09-01 06:12:00 +00:00
|
|
|
|
2013-11-11 00:46:48 +00:00
|
|
|
def _handle_request(self, method): # pylint: disable=too-many-branches
|
2013-10-29 07:22:38 +00:00
|
|
|
""" Does some common checks and calls appropriate method. """
|
|
|
|
url = urlparse(self.path)
|
2013-09-20 06:59:49 +00:00
|
|
|
|
2013-10-29 07:22:38 +00:00
|
|
|
# Read query input
|
|
|
|
data = parse_qs(url.query)
|
2013-09-28 18:09:36 +00:00
|
|
|
|
2014-10-17 07:17:02 +00:00
|
|
|
# parse_qs gives a list for each value, take the latest element
|
|
|
|
for key in data:
|
|
|
|
data[key] = data[key][-1]
|
|
|
|
|
2013-10-29 07:22:38 +00:00
|
|
|
# Did we get post input ?
|
2015-02-14 02:27:13 +00:00
|
|
|
content_length = int(self.headers.get(HTTP_HEADER_CONTENT_LENGTH, 0))
|
2013-09-28 18:09:36 +00:00
|
|
|
|
2013-10-29 07:22:38 +00:00
|
|
|
if content_length:
|
2014-10-17 07:17:02 +00:00
|
|
|
body_content = self.rfile.read(content_length).decode("UTF-8")
|
2013-09-28 18:09:36 +00:00
|
|
|
|
2014-11-29 06:27:44 +00:00
|
|
|
try:
|
|
|
|
data.update(json.loads(body_content))
|
|
|
|
except (TypeError, ValueError):
|
2015-01-30 16:26:06 +00:00
|
|
|
# TypeError if JSON object is not a dict
|
2014-11-29 06:27:44 +00:00
|
|
|
# ValueError if we could not parse JSON
|
2015-01-30 16:26:06 +00:00
|
|
|
_LOGGER.exception(
|
|
|
|
"Exception parsing JSON: %s", body_content)
|
|
|
|
self.write_json_message(
|
2014-11-29 06:27:44 +00:00
|
|
|
"Error parsing JSON", HTTP_UNPROCESSABLE_ENTITY)
|
|
|
|
return
|
2014-10-17 07:17:02 +00:00
|
|
|
|
2015-05-18 13:54:32 +00:00
|
|
|
self._session = self.get_session()
|
2015-01-18 05:55:33 +00:00
|
|
|
if self.server.no_password_set:
|
|
|
|
api_password = self.server.api_password
|
|
|
|
else:
|
2015-02-14 02:27:13 +00:00
|
|
|
api_password = self.headers.get(HTTP_HEADER_HA_AUTH)
|
2014-10-17 07:17:02 +00:00
|
|
|
|
2015-01-18 05:55:33 +00:00
|
|
|
if not api_password and DATA_API_PASSWORD in data:
|
|
|
|
api_password = data[DATA_API_PASSWORD]
|
2013-10-08 06:55:19 +00:00
|
|
|
|
2015-05-18 13:54:32 +00:00
|
|
|
if not api_password and self._session is not None:
|
2015-05-18 17:57:35 +00:00
|
|
|
api_password = self._session.cookie_values.get(
|
|
|
|
CONF_API_PASSWORD)
|
2015-05-18 13:54:32 +00:00
|
|
|
|
2014-04-29 07:30:31 +00:00
|
|
|
if '_METHOD' in data:
|
2014-10-20 01:41:06 +00:00
|
|
|
method = data.pop('_METHOD')
|
2014-04-29 07:30:31 +00:00
|
|
|
|
2013-11-04 23:18:39 +00:00
|
|
|
# Var to keep track if we found a path that matched a handler but
|
|
|
|
# the method was different
|
2013-10-29 07:22:38 +00:00
|
|
|
path_matched_but_not_method = False
|
2013-11-04 23:18:39 +00:00
|
|
|
|
|
|
|
# Var to hold the handler for this path and method if found
|
2013-10-29 07:22:38 +00:00
|
|
|
handle_request_method = False
|
2015-01-30 07:56:04 +00:00
|
|
|
require_auth = True
|
2013-10-24 06:57:08 +00:00
|
|
|
|
2013-11-04 23:18:39 +00:00
|
|
|
# Check every handler to find matching result
|
2015-01-30 07:56:04 +00:00
|
|
|
for t_method, t_path, t_handler, t_auth in self.server.paths:
|
2013-10-29 07:22:38 +00:00
|
|
|
# we either do string-comparison or regular expression matching
|
2014-03-12 05:35:51 +00:00
|
|
|
# pylint: disable=maybe-no-member
|
2013-10-29 07:22:38 +00:00
|
|
|
if isinstance(t_path, str):
|
2013-11-04 23:18:39 +00:00
|
|
|
path_match = url.path == t_path
|
2013-10-29 07:22:38 +00:00
|
|
|
else:
|
2013-11-04 23:18:39 +00:00
|
|
|
path_match = t_path.match(url.path)
|
2013-09-20 06:59:49 +00:00
|
|
|
|
2013-10-29 07:22:38 +00:00
|
|
|
if path_match and method == t_method:
|
|
|
|
# Call the method
|
2015-01-30 07:56:04 +00:00
|
|
|
handle_request_method = t_handler
|
|
|
|
require_auth = t_auth
|
2013-10-29 07:22:38 +00:00
|
|
|
break
|
2013-10-08 06:55:19 +00:00
|
|
|
|
2013-10-29 07:22:38 +00:00
|
|
|
elif path_match:
|
|
|
|
path_matched_but_not_method = True
|
2013-09-20 06:59:49 +00:00
|
|
|
|
2013-11-04 23:18:39 +00:00
|
|
|
# Did we find a handler for the incoming request?
|
2013-10-29 07:22:38 +00:00
|
|
|
if handle_request_method:
|
2013-09-20 06:59:49 +00:00
|
|
|
|
2015-01-30 07:56:04 +00:00
|
|
|
# For some calls we need a valid password
|
|
|
|
if require_auth and api_password != self.server.api_password:
|
2015-01-30 16:26:06 +00:00
|
|
|
self.write_json_message(
|
2014-10-28 07:38:25 +00:00
|
|
|
"API password missing or incorrect.", HTTP_UNAUTHORIZED)
|
2013-11-04 23:18:39 +00:00
|
|
|
|
2014-10-28 07:38:25 +00:00
|
|
|
else:
|
2015-05-18 13:54:32 +00:00
|
|
|
if self._session is None and require_auth:
|
2015-05-19 09:18:41 +00:00
|
|
|
self._session = self.server.sessions.create(
|
2015-05-18 17:57:35 +00:00
|
|
|
api_password)
|
2015-05-18 13:54:32 +00:00
|
|
|
|
2015-01-30 07:56:04 +00:00
|
|
|
handle_request_method(self, path_match, data)
|
2013-09-20 06:59:49 +00:00
|
|
|
|
2013-10-29 07:22:38 +00:00
|
|
|
elif path_matched_but_not_method:
|
|
|
|
self.send_response(HTTP_METHOD_NOT_ALLOWED)
|
2014-11-29 06:27:44 +00:00
|
|
|
self.end_headers()
|
2013-10-13 16:59:13 +00:00
|
|
|
|
2013-10-29 07:22:38 +00:00
|
|
|
else:
|
|
|
|
self.send_response(HTTP_NOT_FOUND)
|
2014-11-29 06:27:44 +00:00
|
|
|
self.end_headers()
|
2013-09-20 06:59:49 +00:00
|
|
|
|
2014-10-27 01:10:01 +00:00
|
|
|
def do_HEAD(self): # pylint: disable=invalid-name
|
|
|
|
""" HEAD request handler. """
|
|
|
|
self._handle_request('HEAD')
|
|
|
|
|
2013-11-11 00:46:48 +00:00
|
|
|
def do_GET(self): # pylint: disable=invalid-name
|
2013-10-29 07:22:38 +00:00
|
|
|
""" GET request handler. """
|
|
|
|
self._handle_request('GET')
|
2013-10-08 06:55:19 +00:00
|
|
|
|
2013-11-11 00:46:48 +00:00
|
|
|
def do_POST(self): # pylint: disable=invalid-name
|
2013-10-29 07:22:38 +00:00
|
|
|
""" POST request handler. """
|
|
|
|
self._handle_request('POST')
|
2013-09-28 18:09:36 +00:00
|
|
|
|
2014-10-17 07:17:02 +00:00
|
|
|
def do_PUT(self): # pylint: disable=invalid-name
|
|
|
|
""" PUT request handler. """
|
|
|
|
self._handle_request('PUT')
|
|
|
|
|
|
|
|
def do_DELETE(self): # pylint: disable=invalid-name
|
|
|
|
""" DELETE request handler. """
|
|
|
|
self._handle_request('DELETE')
|
|
|
|
|
2015-01-30 16:26:06 +00:00
|
|
|
def write_json_message(self, message, status_code=HTTP_OK):
|
2013-10-29 07:22:38 +00:00
|
|
|
""" Helper method to return a message to the caller. """
|
2015-01-30 16:26:06 +00:00
|
|
|
self.write_json({'message': message}, status_code=status_code)
|
2013-09-28 18:09:36 +00:00
|
|
|
|
2015-01-30 16:26:06 +00:00
|
|
|
def write_json(self, data=None, status_code=HTTP_OK, location=None):
|
2013-10-29 07:22:38 +00:00
|
|
|
""" Helper method to return JSON to the caller. """
|
|
|
|
self.send_response(status_code)
|
2015-02-14 02:27:13 +00:00
|
|
|
self.send_header(HTTP_HEADER_CONTENT_TYPE, CONTENT_TYPE_JSON)
|
2013-11-01 18:34:43 +00:00
|
|
|
|
|
|
|
if location:
|
|
|
|
self.send_header('Location', location)
|
|
|
|
|
2015-05-18 13:54:32 +00:00
|
|
|
self.set_session_cookie_header()
|
|
|
|
|
2013-10-29 07:22:38 +00:00
|
|
|
self.end_headers()
|
2013-09-28 18:09:36 +00:00
|
|
|
|
2014-10-17 07:17:02 +00:00
|
|
|
if data is not None:
|
2014-04-14 07:10:24 +00:00
|
|
|
self.wfile.write(
|
2014-04-29 07:30:31 +00:00
|
|
|
json.dumps(data, indent=4, sort_keys=True,
|
|
|
|
cls=rem.JSONEncoder).encode("UTF-8"))
|
2015-01-30 16:26:06 +00:00
|
|
|
|
|
|
|
def write_file(self, path):
|
|
|
|
""" Returns a file to the user. """
|
|
|
|
try:
|
|
|
|
with open(path, 'rb') as inp:
|
|
|
|
self.write_file_pointer(self.guess_type(path), inp)
|
|
|
|
|
|
|
|
except IOError:
|
|
|
|
self.send_response(HTTP_NOT_FOUND)
|
|
|
|
self.end_headers()
|
|
|
|
_LOGGER.exception("Unable to serve %s", path)
|
|
|
|
|
|
|
|
def write_file_pointer(self, content_type, inp):
|
|
|
|
"""
|
|
|
|
Helper function to write a file pointer to the user.
|
|
|
|
Does not do error handling.
|
|
|
|
"""
|
2015-02-14 02:27:13 +00:00
|
|
|
do_gzip = 'gzip' in self.headers.get(HTTP_HEADER_ACCEPT_ENCODING, '')
|
2015-01-30 16:26:06 +00:00
|
|
|
|
|
|
|
self.send_response(HTTP_OK)
|
2015-02-14 02:27:13 +00:00
|
|
|
self.send_header(HTTP_HEADER_CONTENT_TYPE, content_type)
|
2015-01-30 16:26:06 +00:00
|
|
|
|
2015-03-04 05:15:15 +00:00
|
|
|
self.set_cache_header()
|
2015-05-18 13:54:32 +00:00
|
|
|
self.set_session_cookie_header()
|
2015-01-30 16:26:06 +00:00
|
|
|
|
|
|
|
if do_gzip:
|
|
|
|
gzip_data = gzip.compress(inp.read())
|
|
|
|
|
2015-02-14 02:27:13 +00:00
|
|
|
self.send_header(HTTP_HEADER_CONTENT_ENCODING, "gzip")
|
|
|
|
self.send_header(HTTP_HEADER_VARY, HTTP_HEADER_ACCEPT_ENCODING)
|
|
|
|
self.send_header(HTTP_HEADER_CONTENT_LENGTH, str(len(gzip_data)))
|
2015-01-30 16:26:06 +00:00
|
|
|
|
|
|
|
else:
|
|
|
|
fst = os.fstat(inp.fileno())
|
2015-02-14 02:27:13 +00:00
|
|
|
self.send_header(HTTP_HEADER_CONTENT_LENGTH, str(fst[6]))
|
2015-01-30 16:26:06 +00:00
|
|
|
|
|
|
|
self.end_headers()
|
|
|
|
|
|
|
|
if self.command == 'HEAD':
|
|
|
|
return
|
|
|
|
|
|
|
|
elif do_gzip:
|
|
|
|
self.wfile.write(gzip_data)
|
|
|
|
|
|
|
|
else:
|
|
|
|
self.copyfile(inp, self.wfile)
|
2015-03-04 05:15:15 +00:00
|
|
|
|
|
|
|
def set_cache_header(self):
|
|
|
|
""" Add cache headers if not in development """
|
|
|
|
if not self.server.development:
|
|
|
|
# 1 year in seconds
|
|
|
|
cache_time = 365 * 86400
|
|
|
|
|
|
|
|
self.send_header(
|
|
|
|
HTTP_HEADER_CACHE_CONTROL,
|
|
|
|
"public, max-age={}".format(cache_time))
|
|
|
|
self.send_header(
|
|
|
|
HTTP_HEADER_EXPIRES,
|
|
|
|
self.date_time_string(time.time()+cache_time))
|
2015-05-18 13:54:32 +00:00
|
|
|
|
|
|
|
def set_session_cookie_header(self):
|
|
|
|
""" Add the header for the session cookie """
|
2015-05-18 17:57:35 +00:00
|
|
|
if self.server.sessions.enabled and self._session is not None:
|
|
|
|
existing_sess_id = self.get_current_session_id()
|
2015-05-18 13:54:32 +00:00
|
|
|
|
|
|
|
if existing_sess_id != self._session.session_id:
|
|
|
|
self.send_header(
|
|
|
|
'Set-Cookie',
|
|
|
|
SESSION_KEY+'='+self._session.session_id)
|
|
|
|
|
|
|
|
def get_session(self):
|
|
|
|
""" Get the requested session object from cookie value """
|
2015-05-18 17:57:35 +00:00
|
|
|
if self.server.sessions.enabled is not True:
|
2015-05-18 13:54:32 +00:00
|
|
|
return None
|
|
|
|
|
2015-05-18 17:57:35 +00:00
|
|
|
session_id = self.get_current_session_id()
|
|
|
|
if session_id is not None:
|
2015-05-19 09:18:41 +00:00
|
|
|
session = self.server.sessions.get(session_id)
|
2015-05-18 17:57:35 +00:00
|
|
|
if session is not None:
|
|
|
|
session.reset_expiry()
|
|
|
|
return session
|
2015-05-20 07:13:13 +00:00
|
|
|
|
|
|
|
return None
|
2015-05-18 17:57:35 +00:00
|
|
|
|
|
|
|
def get_current_session_id(self):
|
|
|
|
"""
|
|
|
|
Extracts the current session id from the
|
|
|
|
cookie or returns None if not set
|
|
|
|
"""
|
2015-05-18 13:54:32 +00:00
|
|
|
cookie = cookies.SimpleCookie()
|
|
|
|
|
|
|
|
if self.headers.get('Cookie', None) is not None:
|
|
|
|
cookie.load(self.headers.get("Cookie"))
|
|
|
|
|
|
|
|
if cookie.get(SESSION_KEY, False):
|
2015-05-18 17:57:35 +00:00
|
|
|
return cookie[SESSION_KEY].value
|
|
|
|
|
|
|
|
return None
|
2015-05-18 13:54:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ServerSession:
|
|
|
|
""" A very simple session class """
|
|
|
|
def __init__(self, session_id):
|
|
|
|
""" Set up the expiry time on creation """
|
|
|
|
self._expiry = 0
|
|
|
|
self.reset_expiry()
|
|
|
|
self.cookie_values = {}
|
|
|
|
self.session_id = session_id
|
|
|
|
|
|
|
|
def reset_expiry(self):
|
|
|
|
""" Resets the expiry based on current time """
|
|
|
|
self._expiry = date_util.utcnow() + timedelta(
|
|
|
|
seconds=SESSION_TIMEOUT_SECONDS)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_expired(self):
|
|
|
|
""" Return true if the session is expired based on the expiry time """
|
|
|
|
return self._expiry < date_util.utcnow()
|
2015-05-18 17:57:35 +00:00
|
|
|
|
|
|
|
|
2015-08-20 03:09:13 +00:00
|
|
|
class SessionStore(object):
|
2015-05-18 17:57:35 +00:00
|
|
|
""" Responsible for storing and retrieving http sessions """
|
|
|
|
def __init__(self, enabled=True):
|
|
|
|
""" Set up the session store """
|
|
|
|
self._sessions = {}
|
|
|
|
self.enabled = enabled
|
|
|
|
self.session_lock = threading.RLock()
|
|
|
|
|
|
|
|
@Throttle(MIN_SEC_SESSION_CLEARING)
|
2015-05-19 09:18:41 +00:00
|
|
|
def remove_expired(self):
|
2015-05-18 17:57:35 +00:00
|
|
|
""" Remove any expired sessions. """
|
|
|
|
if self.session_lock.acquire(False):
|
|
|
|
try:
|
|
|
|
keys = []
|
|
|
|
for key in self._sessions.keys():
|
|
|
|
keys.append(key)
|
|
|
|
|
|
|
|
for key in keys:
|
|
|
|
if self._sessions[key].is_expired:
|
|
|
|
del self._sessions[key]
|
|
|
|
_LOGGER.info("Cleared expired session %s", key)
|
|
|
|
finally:
|
|
|
|
self.session_lock.release()
|
|
|
|
|
2015-05-19 09:18:41 +00:00
|
|
|
def add(self, key, session):
|
2015-05-18 17:57:35 +00:00
|
|
|
""" Add a new session to the list of tracked sessions """
|
2015-05-19 09:18:41 +00:00
|
|
|
self.remove_expired()
|
2015-05-18 17:57:35 +00:00
|
|
|
with self.session_lock:
|
|
|
|
self._sessions[key] = session
|
|
|
|
|
2015-05-19 09:18:41 +00:00
|
|
|
def get(self, key):
|
2015-05-18 17:57:35 +00:00
|
|
|
""" get a session by key """
|
2015-05-19 09:18:41 +00:00
|
|
|
self.remove_expired()
|
2015-05-18 17:57:35 +00:00
|
|
|
session = self._sessions.get(key, None)
|
|
|
|
if session is not None and session.is_expired:
|
|
|
|
return None
|
|
|
|
return session
|
|
|
|
|
2015-05-19 09:18:41 +00:00
|
|
|
def create(self, api_password):
|
2015-05-18 17:57:35 +00:00
|
|
|
""" Creates a new session and adds it to the sessions """
|
|
|
|
if self.enabled is not True:
|
|
|
|
return None
|
|
|
|
|
|
|
|
chars = string.ascii_letters + string.digits
|
|
|
|
session_id = ''.join([random.choice(chars) for i in range(20)])
|
|
|
|
session = ServerSession(session_id)
|
|
|
|
session.cookie_values[CONF_API_PASSWORD] = api_password
|
2015-05-19 09:18:41 +00:00
|
|
|
self.add(session_id, session)
|
2015-05-18 17:57:35 +00:00
|
|
|
return session
|