2015-01-30 16:26:06 +00:00
|
|
|
"""
|
2016-02-23 20:06:50 +00:00
|
|
|
Rest API for Home Assistant.
|
2015-10-25 14:30:56 +00:00
|
|
|
|
|
|
|
For more details about the RESTful API, please refer to the documentation at
|
2015-11-09 12:12:18 +00:00
|
|
|
https://home-assistant.io/developers/api/
|
2015-01-30 16:26:06 +00:00
|
|
|
"""
|
2016-09-13 02:16:14 +00:00
|
|
|
import asyncio
|
2016-02-19 05:27:50 +00:00
|
|
|
import json
|
2015-01-30 16:26:06 +00:00
|
|
|
import logging
|
2016-10-24 06:48:01 +00:00
|
|
|
|
|
|
|
from aiohttp import web
|
|
|
|
import async_timeout
|
2015-01-30 07:56:04 +00:00
|
|
|
|
2015-08-17 03:44:46 +00:00
|
|
|
import homeassistant.core as ha
|
2015-01-30 07:56:04 +00:00
|
|
|
import homeassistant.remote as rem
|
2015-11-07 09:44:02 +00:00
|
|
|
from homeassistant.bootstrap import ERROR_LOG_FILENAME
|
2015-01-30 07:56:04 +00:00
|
|
|
from homeassistant.const import (
|
2016-05-14 07:58:36 +00:00
|
|
|
EVENT_HOMEASSISTANT_STOP, EVENT_TIME_CHANGED,
|
|
|
|
HTTP_BAD_REQUEST, HTTP_CREATED, HTTP_NOT_FOUND,
|
|
|
|
HTTP_UNPROCESSABLE_ENTITY, MATCH_ALL, URL_API, URL_API_COMPONENTS,
|
2016-05-07 05:11:35 +00:00
|
|
|
URL_API_CONFIG, URL_API_DISCOVERY_INFO, URL_API_ERROR_LOG,
|
2016-05-16 07:25:47 +00:00
|
|
|
URL_API_EVENT_FORWARD, URL_API_EVENTS, URL_API_SERVICES,
|
2016-05-07 05:11:35 +00:00
|
|
|
URL_API_STATES, URL_API_STATES_ENTITY, URL_API_STREAM, URL_API_TEMPLATE,
|
|
|
|
__version__)
|
2016-02-19 05:27:50 +00:00
|
|
|
from homeassistant.exceptions import TemplateError
|
2016-10-24 06:48:01 +00:00
|
|
|
from homeassistant.helpers.state import AsyncTrackStates
|
2016-02-23 20:06:50 +00:00
|
|
|
from homeassistant.helpers import template
|
2016-05-14 07:58:36 +00:00
|
|
|
from homeassistant.components.http import HomeAssistantView
|
2015-01-30 07:56:04 +00:00
|
|
|
|
|
|
|
DOMAIN = 'api'
|
2016-05-14 07:58:36 +00:00
|
|
|
DEPENDENCIES = ['http']
|
2015-01-30 07:56:04 +00:00
|
|
|
|
2015-02-19 08:15:21 +00:00
|
|
|
STREAM_PING_PAYLOAD = "ping"
|
|
|
|
STREAM_PING_INTERVAL = 50 # seconds
|
|
|
|
|
2015-01-30 16:26:06 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2015-01-30 07:56:04 +00:00
|
|
|
|
|
|
|
def setup(hass, config):
|
2016-02-23 20:06:50 +00:00
|
|
|
"""Register the API with the HTTP interface."""
|
2016-10-24 06:48:01 +00:00
|
|
|
hass.http.register_view(APIStatusView)
|
|
|
|
hass.http.register_view(APIEventStream)
|
|
|
|
hass.http.register_view(APIConfigView)
|
|
|
|
hass.http.register_view(APIDiscoveryView)
|
|
|
|
hass.http.register_view(APIStatesView)
|
|
|
|
hass.http.register_view(APIEntityStateView)
|
|
|
|
hass.http.register_view(APIEventListenersView)
|
|
|
|
hass.http.register_view(APIEventView)
|
|
|
|
hass.http.register_view(APIServicesView)
|
|
|
|
hass.http.register_view(APIDomainServicesView)
|
|
|
|
hass.http.register_view(APIEventForwardingView)
|
|
|
|
hass.http.register_view(APIComponentsView)
|
|
|
|
hass.http.register_view(APITemplateView)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2017-03-30 07:50:53 +00:00
|
|
|
hass.http.register_static_path(
|
|
|
|
URL_API_ERROR_LOG, hass.config.path(ERROR_LOG_FILENAME), False)
|
|
|
|
|
2015-01-30 07:56:04 +00:00
|
|
|
return True
|
|
|
|
|
2015-01-30 16:26:06 +00:00
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
class APIStatusView(HomeAssistantView):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""View to handle Status requests."""
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
url = URL_API
|
|
|
|
name = "api:status"
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@ha.callback
|
2016-05-10 01:09:38 +00:00
|
|
|
def get(self, request):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Retrieve if API is running."""
|
|
|
|
return self.json_message('API running.')
|
2015-01-30 07:56:04 +00:00
|
|
|
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
class APIEventStream(HomeAssistantView):
|
2016-05-17 02:47:15 +00:00
|
|
|
"""View to handle EventStream requests."""
|
2015-02-14 02:59:42 +00:00
|
|
|
|
2016-05-14 07:58:36 +00:00
|
|
|
url = URL_API_STREAM
|
|
|
|
name = "api:stream"
|
2015-09-24 04:35:23 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@asyncio.coroutine
|
2016-05-14 07:58:36 +00:00
|
|
|
def get(self, request):
|
|
|
|
"""Provide a streaming interface for the event bus."""
|
2016-11-25 21:04:06 +00:00
|
|
|
# pylint: disable=no-self-use
|
|
|
|
hass = request.app['hass']
|
2016-05-14 07:58:36 +00:00
|
|
|
stop_obj = object()
|
2016-11-25 21:04:06 +00:00
|
|
|
to_write = asyncio.Queue(loop=hass.loop)
|
2016-05-14 07:58:36 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
restrict = request.GET.get('restrict')
|
2016-05-14 07:58:36 +00:00
|
|
|
if restrict:
|
2016-06-30 16:02:12 +00:00
|
|
|
restrict = restrict.split(',') + [EVENT_HOMEASSISTANT_STOP]
|
2016-05-14 07:58:36 +00:00
|
|
|
|
2016-09-13 02:16:14 +00:00
|
|
|
@asyncio.coroutine
|
2016-06-30 16:02:12 +00:00
|
|
|
def forward_events(event):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Forward events to the open request."""
|
|
|
|
if event.event_type == EVENT_TIME_CHANGED:
|
2016-05-16 06:54:14 +00:00
|
|
|
return
|
|
|
|
|
2016-05-22 01:24:03 +00:00
|
|
|
if restrict and event.event_type not in restrict:
|
|
|
|
return
|
|
|
|
|
2016-05-16 06:54:14 +00:00
|
|
|
_LOGGER.debug('STREAM %s FORWARDING %s', id(stop_obj), event)
|
|
|
|
|
|
|
|
if event.event_type == EVENT_HOMEASSISTANT_STOP:
|
|
|
|
data = stop_obj
|
2016-05-14 07:58:36 +00:00
|
|
|
else:
|
2016-05-16 06:54:14 +00:00
|
|
|
data = json.dumps(event, cls=rem.JSONEncoder)
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
yield from to_write.put(data)
|
2016-05-14 07:58:36 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
response = web.StreamResponse()
|
|
|
|
response.content_type = 'text/event-stream'
|
|
|
|
yield from response.prepare(request)
|
2016-05-14 07:58:36 +00:00
|
|
|
|
2016-11-25 21:04:06 +00:00
|
|
|
unsub_stream = hass.bus.async_listen(MATCH_ALL, forward_events)
|
2015-02-19 08:15:21 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
try:
|
|
|
|
_LOGGER.debug('STREAM %s ATTACHED', id(stop_obj))
|
2016-05-17 02:47:15 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
# Fire off one message so browsers fire open event right away
|
|
|
|
yield from to_write.put(STREAM_PING_PAYLOAD)
|
2015-02-14 02:59:42 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
while True:
|
|
|
|
try:
|
|
|
|
with async_timeout.timeout(STREAM_PING_INTERVAL,
|
2016-11-25 21:04:06 +00:00
|
|
|
loop=hass.loop):
|
2016-10-24 06:48:01 +00:00
|
|
|
payload = yield from to_write.get()
|
2015-02-14 02:59:42 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
if payload is stop_obj:
|
2016-09-07 13:59:59 +00:00
|
|
|
break
|
2015-02-20 07:31:06 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
msg = "data: {}\n\n".format(payload)
|
|
|
|
_LOGGER.debug('STREAM %s WRITING %s', id(stop_obj),
|
|
|
|
msg.strip())
|
|
|
|
response.write(msg.encode("UTF-8"))
|
|
|
|
yield from response.drain()
|
|
|
|
except asyncio.TimeoutError:
|
|
|
|
yield from to_write.put(STREAM_PING_PAYLOAD)
|
|
|
|
|
2017-01-15 16:36:24 +00:00
|
|
|
except asyncio.CancelledError:
|
|
|
|
_LOGGER.debug('STREAM %s ABORT', id(stop_obj))
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
finally:
|
|
|
|
_LOGGER.debug('STREAM %s RESPONSE CLOSED', id(stop_obj))
|
|
|
|
unsub_stream()
|
2015-02-14 02:59:42 +00:00
|
|
|
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
class APIConfigView(HomeAssistantView):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""View to handle Config requests."""
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
url = URL_API_CONFIG
|
|
|
|
name = "api:config"
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@ha.callback
|
2016-05-10 01:09:38 +00:00
|
|
|
def get(self, request):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Get current configuration."""
|
2016-11-25 21:04:06 +00:00
|
|
|
return self.json(request.app['hass'].config.as_dict())
|
2015-05-02 01:24:32 +00:00
|
|
|
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
class APIDiscoveryView(HomeAssistantView):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""View to provide discovery info."""
|
|
|
|
|
|
|
|
requires_auth = False
|
2016-05-10 01:09:38 +00:00
|
|
|
url = URL_API_DISCOVERY_INFO
|
|
|
|
name = "api:discovery"
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@ha.callback
|
2016-05-10 01:09:38 +00:00
|
|
|
def get(self, request):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Get discovery info."""
|
2016-11-25 21:04:06 +00:00
|
|
|
hass = request.app['hass']
|
|
|
|
needs_auth = hass.config.api.api_password is not None
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json({
|
2016-11-25 21:04:06 +00:00
|
|
|
'base_url': hass.config.api.base_url,
|
|
|
|
'location_name': hass.config.location_name,
|
2016-05-14 07:58:36 +00:00
|
|
|
'requires_api_password': needs_auth,
|
2016-06-02 05:37:16 +00:00
|
|
|
'version': __version__
|
2016-05-14 07:58:36 +00:00
|
|
|
})
|
2016-05-07 05:11:35 +00:00
|
|
|
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
class APIStatesView(HomeAssistantView):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""View to handle States requests."""
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
url = URL_API_STATES
|
|
|
|
name = "api:states"
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@ha.callback
|
2016-05-10 01:09:38 +00:00
|
|
|
def get(self, request):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Get current states."""
|
2016-11-25 21:04:06 +00:00
|
|
|
return self.json(request.app['hass'].states.async_all())
|
2015-01-30 07:56:04 +00:00
|
|
|
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
class APIEntityStateView(HomeAssistantView):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""View to handle EntityState requests."""
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
url = "/api/states/{entity_id}"
|
2016-05-10 01:09:38 +00:00
|
|
|
name = "api:entity-state"
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@ha.callback
|
2016-05-10 01:09:38 +00:00
|
|
|
def get(self, request, entity_id):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Retrieve state of entity."""
|
2016-11-25 21:04:06 +00:00
|
|
|
state = request.app['hass'].states.get(entity_id)
|
2016-05-10 01:09:38 +00:00
|
|
|
if state:
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json(state)
|
2016-05-10 01:09:38 +00:00
|
|
|
else:
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json_message('Entity not found', HTTP_NOT_FOUND)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@asyncio.coroutine
|
2016-05-10 01:09:38 +00:00
|
|
|
def post(self, request, entity_id):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Update state of entity."""
|
2016-11-25 21:04:06 +00:00
|
|
|
hass = request.app['hass']
|
2016-05-10 01:09:38 +00:00
|
|
|
try:
|
2016-10-24 06:48:01 +00:00
|
|
|
data = yield from request.json()
|
|
|
|
except ValueError:
|
|
|
|
return self.json_message('Invalid JSON specified',
|
|
|
|
HTTP_BAD_REQUEST)
|
|
|
|
|
|
|
|
new_state = data.get('state')
|
|
|
|
|
|
|
|
if not new_state:
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json_message('No state specified', HTTP_BAD_REQUEST)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
attributes = data.get('attributes')
|
|
|
|
force_update = data.get('force_update', False)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2016-11-25 21:04:06 +00:00
|
|
|
is_new_state = hass.states.get(entity_id) is None
|
2016-05-10 01:09:38 +00:00
|
|
|
|
|
|
|
# Write state
|
2016-11-25 21:04:06 +00:00
|
|
|
hass.states.async_set(entity_id, new_state, attributes, force_update)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
|
|
|
# Read the state back for our response
|
2016-10-24 06:48:01 +00:00
|
|
|
status_code = HTTP_CREATED if is_new_state else 200
|
2016-11-25 21:04:06 +00:00
|
|
|
resp = self.json(hass.states.get(entity_id), status_code)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
|
|
|
resp.headers.add('Location', URL_API_STATES_ENTITY.format(entity_id))
|
|
|
|
|
|
|
|
return resp
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@ha.callback
|
2016-05-10 01:09:38 +00:00
|
|
|
def delete(self, request, entity_id):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Remove entity."""
|
2016-11-25 21:04:06 +00:00
|
|
|
if request.app['hass'].states.async_remove(entity_id):
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json_message('Entity removed')
|
2016-05-10 01:09:38 +00:00
|
|
|
else:
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json_message('Entity not found', HTTP_NOT_FOUND)
|
2016-02-14 07:00:38 +00:00
|
|
|
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
class APIEventListenersView(HomeAssistantView):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""View to handle EventListeners requests."""
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
url = URL_API_EVENTS
|
|
|
|
name = "api:event-listeners"
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@ha.callback
|
2016-05-10 01:09:38 +00:00
|
|
|
def get(self, request):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Get event listeners."""
|
2016-11-25 21:04:06 +00:00
|
|
|
return self.json(async_events_json(request.app['hass']))
|
2015-01-30 07:56:04 +00:00
|
|
|
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
class APIEventView(HomeAssistantView):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""View to handle Event requests."""
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
url = '/api/events/{event_type}'
|
2016-05-10 01:09:38 +00:00
|
|
|
name = "api:event"
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@asyncio.coroutine
|
2016-05-10 01:09:38 +00:00
|
|
|
def post(self, request, event_type):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Fire events."""
|
2016-10-24 06:48:01 +00:00
|
|
|
body = yield from request.text()
|
|
|
|
event_data = json.loads(body) if body else None
|
2016-05-14 07:58:36 +00:00
|
|
|
|
|
|
|
if event_data is not None and not isinstance(event_data, dict):
|
|
|
|
return self.json_message('Event data should be a JSON object',
|
|
|
|
HTTP_BAD_REQUEST)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
|
|
|
# Special case handling for event STATE_CHANGED
|
|
|
|
# We will try to convert state dicts back to State objects
|
|
|
|
if event_type == ha.EVENT_STATE_CHANGED and event_data:
|
|
|
|
for key in ('old_state', 'new_state'):
|
|
|
|
state = ha.State.from_dict(event_data.get(key))
|
|
|
|
|
|
|
|
if state:
|
|
|
|
event_data[key] = state
|
|
|
|
|
2016-11-25 21:04:06 +00:00
|
|
|
request.app['hass'].bus.async_fire(event_type, event_data,
|
|
|
|
ha.EventOrigin.remote)
|
2015-01-30 07:56:04 +00:00
|
|
|
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json_message("Event {} fired.".format(event_type))
|
2015-01-30 07:56:04 +00:00
|
|
|
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
class APIServicesView(HomeAssistantView):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""View to handle Services requests."""
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
url = URL_API_SERVICES
|
|
|
|
name = "api:services"
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@ha.callback
|
2016-05-10 01:09:38 +00:00
|
|
|
def get(self, request):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Get registered services."""
|
2016-11-25 21:04:06 +00:00
|
|
|
return self.json(async_services_json(request.app['hass']))
|
2015-01-30 07:56:04 +00:00
|
|
|
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
class APIDomainServicesView(HomeAssistantView):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""View to handle DomainServices requests."""
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
url = "/api/services/{domain}/{service}"
|
2016-05-10 01:09:38 +00:00
|
|
|
name = "api:domain-services"
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@asyncio.coroutine
|
2016-05-14 07:58:36 +00:00
|
|
|
def post(self, request, domain, service):
|
|
|
|
"""Call a service.
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2016-05-14 07:58:36 +00:00
|
|
|
Returns a list of changed states.
|
|
|
|
"""
|
2016-11-25 21:04:06 +00:00
|
|
|
hass = request.app['hass']
|
2016-10-24 06:48:01 +00:00
|
|
|
body = yield from request.text()
|
|
|
|
data = json.loads(body) if body else None
|
|
|
|
|
2016-11-25 21:04:06 +00:00
|
|
|
with AsyncTrackStates(hass) as changed_states:
|
|
|
|
yield from hass.services.async_call(domain, service, data, True)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json(changed_states)
|
2015-01-30 07:56:04 +00:00
|
|
|
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
class APIEventForwardingView(HomeAssistantView):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""View to handle EventForwarding requests."""
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
url = URL_API_EVENT_FORWARD
|
|
|
|
name = "api:event-forward"
|
2016-05-14 07:58:36 +00:00
|
|
|
event_forwarder = None
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@asyncio.coroutine
|
2016-05-10 01:09:38 +00:00
|
|
|
def post(self, request):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Setup an event forwarder."""
|
2017-03-15 06:58:15 +00:00
|
|
|
_LOGGER.warning('Event forwarding is deprecated. '
|
|
|
|
'Will be removed by 0.43')
|
2016-11-25 21:04:06 +00:00
|
|
|
hass = request.app['hass']
|
2016-10-24 06:48:01 +00:00
|
|
|
try:
|
|
|
|
data = yield from request.json()
|
|
|
|
except ValueError:
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json_message("No data received.", HTTP_BAD_REQUEST)
|
2016-10-24 06:48:01 +00:00
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
try:
|
2016-05-14 07:58:36 +00:00
|
|
|
host = data['host']
|
|
|
|
api_password = data['api_password']
|
2016-05-10 01:09:38 +00:00
|
|
|
except KeyError:
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json_message("No host or api_password received.",
|
|
|
|
HTTP_BAD_REQUEST)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
port = int(data['port']) if 'port' in data else None
|
|
|
|
except ValueError:
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json_message("Invalid value received for port.",
|
|
|
|
HTTP_UNPROCESSABLE_ENTITY)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
|
|
|
api = rem.API(host, api_password, port)
|
|
|
|
|
2016-11-25 21:04:06 +00:00
|
|
|
valid = yield from hass.loop.run_in_executor(
|
2016-10-24 06:48:01 +00:00
|
|
|
None, api.validate_api)
|
|
|
|
if not valid:
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json_message("Unable to validate API.",
|
|
|
|
HTTP_UNPROCESSABLE_ENTITY)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2016-05-14 07:58:36 +00:00
|
|
|
if self.event_forwarder is None:
|
2016-11-25 21:04:06 +00:00
|
|
|
self.event_forwarder = rem.EventForwarder(hass)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
self.event_forwarder.async_connect(api)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json_message("Event forwarding setup.")
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@asyncio.coroutine
|
2016-05-10 01:09:38 +00:00
|
|
|
def delete(self, request):
|
2016-10-24 06:48:01 +00:00
|
|
|
"""Remove event forwarder."""
|
|
|
|
try:
|
|
|
|
data = yield from request.json()
|
|
|
|
except ValueError:
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json_message("No data received.", HTTP_BAD_REQUEST)
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
try:
|
2016-05-14 07:58:36 +00:00
|
|
|
host = data['host']
|
2016-05-10 01:09:38 +00:00
|
|
|
except KeyError:
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json_message("No host received.", HTTP_BAD_REQUEST)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
port = int(data['port']) if 'port' in data else None
|
|
|
|
except ValueError:
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json_message("Invalid value received for port.",
|
|
|
|
HTTP_UNPROCESSABLE_ENTITY)
|
2016-05-10 01:09:38 +00:00
|
|
|
|
2016-05-14 07:58:36 +00:00
|
|
|
if self.event_forwarder is not None:
|
2016-05-10 01:09:38 +00:00
|
|
|
api = rem.API(host, None, port)
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
self.event_forwarder.async_disconnect(api)
|
2015-01-30 07:56:04 +00:00
|
|
|
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json_message("Event forwarding cancelled.")
|
2015-02-01 03:08:50 +00:00
|
|
|
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
class APIComponentsView(HomeAssistantView):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""View to handle Components requests."""
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
url = URL_API_COMPONENTS
|
|
|
|
name = "api:components"
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@ha.callback
|
2016-05-10 01:09:38 +00:00
|
|
|
def get(self, request):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Get current loaded components."""
|
2016-11-25 21:04:06 +00:00
|
|
|
return self.json(request.app['hass'].config.components)
|
2015-05-02 02:02:29 +00:00
|
|
|
|
2015-05-02 03:56:10 +00:00
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
class APITemplateView(HomeAssistantView):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""View to handle requests."""
|
|
|
|
|
2016-05-10 01:09:38 +00:00
|
|
|
url = URL_API_TEMPLATE
|
|
|
|
name = "api:template"
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
@asyncio.coroutine
|
2016-05-10 01:09:38 +00:00
|
|
|
def post(self, request):
|
2016-05-14 07:58:36 +00:00
|
|
|
"""Render a template."""
|
|
|
|
try:
|
2016-10-24 06:48:01 +00:00
|
|
|
data = yield from request.json()
|
2016-11-25 21:04:06 +00:00
|
|
|
tpl = template.Template(data['template'], request.app['hass'])
|
2016-10-24 06:48:01 +00:00
|
|
|
return tpl.async_render(data.get('variables'))
|
|
|
|
except (ValueError, TemplateError) as ex:
|
2016-05-14 07:58:36 +00:00
|
|
|
return self.json_message('Error rendering template: {}'.format(ex),
|
|
|
|
HTTP_BAD_REQUEST)
|
2015-12-10 07:56:20 +00:00
|
|
|
|
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
def async_services_json(hass):
|
2016-02-23 20:06:50 +00:00
|
|
|
"""Generate services data to JSONify."""
|
2015-05-02 02:02:29 +00:00
|
|
|
return [{"domain": key, "services": value}
|
2016-10-24 06:48:01 +00:00
|
|
|
for key, value in hass.services.async_services().items()]
|
2015-05-02 02:02:29 +00:00
|
|
|
|
2015-05-02 03:56:10 +00:00
|
|
|
|
2016-10-24 06:48:01 +00:00
|
|
|
def async_events_json(hass):
|
2016-02-23 20:06:50 +00:00
|
|
|
"""Generate event data to JSONify."""
|
2015-05-02 02:02:29 +00:00
|
|
|
return [{"event": key, "listener_count": value}
|
2016-10-24 06:48:01 +00:00
|
|
|
for key, value in hass.bus.async_listeners().items()]
|