""" Rest API for Home Assistant. For more details about the RESTful API, please refer to the documentation at https://home-assistant.io/developers/api/ """ import asyncio import json import logging from aiohttp import web import async_timeout import homeassistant.core as ha import homeassistant.remote as rem from homeassistant.bootstrap import ERROR_LOG_FILENAME from homeassistant.const import ( EVENT_HOMEASSISTANT_STOP, EVENT_TIME_CHANGED, HTTP_BAD_REQUEST, HTTP_CREATED, HTTP_NOT_FOUND, HTTP_UNPROCESSABLE_ENTITY, MATCH_ALL, URL_API, URL_API_COMPONENTS, URL_API_CONFIG, URL_API_DISCOVERY_INFO, URL_API_ERROR_LOG, URL_API_EVENT_FORWARD, URL_API_EVENTS, URL_API_SERVICES, URL_API_STATES, URL_API_STATES_ENTITY, URL_API_STREAM, URL_API_TEMPLATE, __version__) from homeassistant.exceptions import TemplateError from homeassistant.helpers.state import AsyncTrackStates from homeassistant.helpers import template from homeassistant.components.http import HomeAssistantView DOMAIN = 'api' DEPENDENCIES = ['http'] STREAM_PING_PAYLOAD = "ping" STREAM_PING_INTERVAL = 50 # seconds _LOGGER = logging.getLogger(__name__) def setup(hass, config): """Register the API with the HTTP interface.""" 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(APIErrorLogView) hass.http.register_view(APITemplateView) return True class APIStatusView(HomeAssistantView): """View to handle Status requests.""" url = URL_API name = "api:status" @ha.callback def get(self, request): """Retrieve if API is running.""" return self.json_message('API running.') class APIEventStream(HomeAssistantView): """View to handle EventStream requests.""" url = URL_API_STREAM name = "api:stream" @asyncio.coroutine def get(self, request): """Provide a streaming interface for the event bus.""" # pylint: disable=no-self-use hass = request.app['hass'] stop_obj = object() to_write = asyncio.Queue(loop=hass.loop) restrict = request.GET.get('restrict') if restrict: restrict = restrict.split(',') + [EVENT_HOMEASSISTANT_STOP] @asyncio.coroutine def forward_events(event): """Forward events to the open request.""" if event.event_type == EVENT_TIME_CHANGED: return if restrict and event.event_type not in restrict: return _LOGGER.debug('STREAM %s FORWARDING %s', id(stop_obj), event) if event.event_type == EVENT_HOMEASSISTANT_STOP: data = stop_obj else: data = json.dumps(event, cls=rem.JSONEncoder) yield from to_write.put(data) response = web.StreamResponse() response.content_type = 'text/event-stream' yield from response.prepare(request) unsub_stream = hass.bus.async_listen(MATCH_ALL, forward_events) try: _LOGGER.debug('STREAM %s ATTACHED', id(stop_obj)) # Fire off one message so browsers fire open event right away yield from to_write.put(STREAM_PING_PAYLOAD) while True: try: with async_timeout.timeout(STREAM_PING_INTERVAL, loop=hass.loop): payload = yield from to_write.get() if payload is stop_obj: break 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) except asyncio.CancelledError: _LOGGER.debug('STREAM %s ABORT', id(stop_obj)) finally: _LOGGER.debug('STREAM %s RESPONSE CLOSED', id(stop_obj)) unsub_stream() class APIConfigView(HomeAssistantView): """View to handle Config requests.""" url = URL_API_CONFIG name = "api:config" @ha.callback def get(self, request): """Get current configuration.""" return self.json(request.app['hass'].config.as_dict()) class APIDiscoveryView(HomeAssistantView): """View to provide discovery info.""" requires_auth = False url = URL_API_DISCOVERY_INFO name = "api:discovery" @ha.callback def get(self, request): """Get discovery info.""" hass = request.app['hass'] needs_auth = hass.config.api.api_password is not None return self.json({ 'base_url': hass.config.api.base_url, 'location_name': hass.config.location_name, 'requires_api_password': needs_auth, 'version': __version__ }) class APIStatesView(HomeAssistantView): """View to handle States requests.""" url = URL_API_STATES name = "api:states" @ha.callback def get(self, request): """Get current states.""" return self.json(request.app['hass'].states.async_all()) class APIEntityStateView(HomeAssistantView): """View to handle EntityState requests.""" url = "/api/states/{entity_id}" name = "api:entity-state" @ha.callback def get(self, request, entity_id): """Retrieve state of entity.""" state = request.app['hass'].states.get(entity_id) if state: return self.json(state) else: return self.json_message('Entity not found', HTTP_NOT_FOUND) @asyncio.coroutine def post(self, request, entity_id): """Update state of entity.""" hass = request.app['hass'] try: 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: return self.json_message('No state specified', HTTP_BAD_REQUEST) attributes = data.get('attributes') force_update = data.get('force_update', False) is_new_state = hass.states.get(entity_id) is None # Write state hass.states.async_set(entity_id, new_state, attributes, force_update) # Read the state back for our response status_code = HTTP_CREATED if is_new_state else 200 resp = self.json(hass.states.get(entity_id), status_code) resp.headers.add('Location', URL_API_STATES_ENTITY.format(entity_id)) return resp @ha.callback def delete(self, request, entity_id): """Remove entity.""" if request.app['hass'].states.async_remove(entity_id): return self.json_message('Entity removed') else: return self.json_message('Entity not found', HTTP_NOT_FOUND) class APIEventListenersView(HomeAssistantView): """View to handle EventListeners requests.""" url = URL_API_EVENTS name = "api:event-listeners" @ha.callback def get(self, request): """Get event listeners.""" return self.json(async_events_json(request.app['hass'])) class APIEventView(HomeAssistantView): """View to handle Event requests.""" url = '/api/events/{event_type}' name = "api:event" @asyncio.coroutine def post(self, request, event_type): """Fire events.""" body = yield from request.text() event_data = json.loads(body) if body else None 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) # 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 request.app['hass'].bus.async_fire(event_type, event_data, ha.EventOrigin.remote) return self.json_message("Event {} fired.".format(event_type)) class APIServicesView(HomeAssistantView): """View to handle Services requests.""" url = URL_API_SERVICES name = "api:services" @ha.callback def get(self, request): """Get registered services.""" return self.json(async_services_json(request.app['hass'])) class APIDomainServicesView(HomeAssistantView): """View to handle DomainServices requests.""" url = "/api/services/{domain}/{service}" name = "api:domain-services" @asyncio.coroutine def post(self, request, domain, service): """Call a service. Returns a list of changed states. """ hass = request.app['hass'] body = yield from request.text() data = json.loads(body) if body else None with AsyncTrackStates(hass) as changed_states: yield from hass.services.async_call(domain, service, data, True) return self.json(changed_states) class APIEventForwardingView(HomeAssistantView): """View to handle EventForwarding requests.""" url = URL_API_EVENT_FORWARD name = "api:event-forward" event_forwarder = None @asyncio.coroutine def post(self, request): """Setup an event forwarder.""" hass = request.app['hass'] try: data = yield from request.json() except ValueError: return self.json_message("No data received.", HTTP_BAD_REQUEST) try: host = data['host'] api_password = data['api_password'] except KeyError: return self.json_message("No host or api_password received.", HTTP_BAD_REQUEST) try: port = int(data['port']) if 'port' in data else None except ValueError: return self.json_message("Invalid value received for port.", HTTP_UNPROCESSABLE_ENTITY) api = rem.API(host, api_password, port) valid = yield from hass.loop.run_in_executor( None, api.validate_api) if not valid: return self.json_message("Unable to validate API.", HTTP_UNPROCESSABLE_ENTITY) if self.event_forwarder is None: self.event_forwarder = rem.EventForwarder(hass) self.event_forwarder.async_connect(api) return self.json_message("Event forwarding setup.") @asyncio.coroutine def delete(self, request): """Remove event forwarder.""" try: data = yield from request.json() except ValueError: return self.json_message("No data received.", HTTP_BAD_REQUEST) try: host = data['host'] except KeyError: return self.json_message("No host received.", HTTP_BAD_REQUEST) try: port = int(data['port']) if 'port' in data else None except ValueError: return self.json_message("Invalid value received for port.", HTTP_UNPROCESSABLE_ENTITY) if self.event_forwarder is not None: api = rem.API(host, None, port) self.event_forwarder.async_disconnect(api) return self.json_message("Event forwarding cancelled.") class APIComponentsView(HomeAssistantView): """View to handle Components requests.""" url = URL_API_COMPONENTS name = "api:components" @ha.callback def get(self, request): """Get current loaded components.""" return self.json(request.app['hass'].config.components) class APIErrorLogView(HomeAssistantView): """View to handle ErrorLog requests.""" url = URL_API_ERROR_LOG name = "api:error-log" @asyncio.coroutine def get(self, request): """Serve error log.""" resp = yield from self.file( request, request.app['hass'].config.path(ERROR_LOG_FILENAME)) return resp class APITemplateView(HomeAssistantView): """View to handle requests.""" url = URL_API_TEMPLATE name = "api:template" @asyncio.coroutine def post(self, request): """Render a template.""" try: data = yield from request.json() tpl = template.Template(data['template'], request.app['hass']) return tpl.async_render(data.get('variables')) except (ValueError, TemplateError) as ex: return self.json_message('Error rendering template: {}'.format(ex), HTTP_BAD_REQUEST) def async_services_json(hass): """Generate services data to JSONify.""" return [{"domain": key, "services": value} for key, value in hass.services.async_services().items()] def async_events_json(hass): """Generate event data to JSONify.""" return [{"event": key, "listener_count": value} for key, value in hass.bus.async_listeners().items()]