193 lines
6.1 KiB
Python
193 lines
6.1 KiB
Python
"""Alexa models."""
|
|
import logging
|
|
from uuid import uuid4
|
|
|
|
from .const import (
|
|
API_CONTEXT,
|
|
API_DIRECTIVE,
|
|
API_ENDPOINT,
|
|
API_EVENT,
|
|
API_HEADER,
|
|
API_PAYLOAD,
|
|
API_SCOPE,
|
|
)
|
|
from .entities import ENTITY_ADAPTERS
|
|
from .errors import AlexaInvalidEndpointError
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
class AlexaDirective:
|
|
"""An incoming Alexa directive."""
|
|
|
|
def __init__(self, request):
|
|
"""Initialize a directive."""
|
|
self._directive = request[API_DIRECTIVE]
|
|
self.namespace = self._directive[API_HEADER]["namespace"]
|
|
self.name = self._directive[API_HEADER]["name"]
|
|
self.payload = self._directive[API_PAYLOAD]
|
|
self.has_endpoint = API_ENDPOINT in self._directive
|
|
|
|
self.entity = self.entity_id = self.endpoint = None
|
|
|
|
def load_entity(self, hass, config):
|
|
"""Set attributes related to the entity for this request.
|
|
|
|
Sets these attributes when self.has_endpoint is True:
|
|
|
|
- entity
|
|
- entity_id
|
|
- endpoint
|
|
|
|
Behavior when self.has_endpoint is False is undefined.
|
|
|
|
Will raise AlexaInvalidEndpointError if the endpoint in the request is
|
|
malformed or nonexistant.
|
|
"""
|
|
_endpoint_id = self._directive[API_ENDPOINT]["endpointId"]
|
|
self.entity_id = _endpoint_id.replace("#", ".")
|
|
|
|
self.entity = hass.states.get(self.entity_id)
|
|
if not self.entity or not config.should_expose(self.entity_id):
|
|
raise AlexaInvalidEndpointError(_endpoint_id)
|
|
|
|
self.endpoint = ENTITY_ADAPTERS[self.entity.domain](hass, config, self.entity)
|
|
|
|
def response(self, name="Response", namespace="Alexa", payload=None):
|
|
"""Create an API formatted response.
|
|
|
|
Async friendly.
|
|
"""
|
|
response = AlexaResponse(name, namespace, payload)
|
|
|
|
token = self._directive[API_HEADER].get("correlationToken")
|
|
if token:
|
|
response.set_correlation_token(token)
|
|
|
|
if self.has_endpoint:
|
|
response.set_endpoint(self._directive[API_ENDPOINT].copy())
|
|
|
|
return response
|
|
|
|
def error(
|
|
self,
|
|
namespace="Alexa",
|
|
error_type="INTERNAL_ERROR",
|
|
error_message="",
|
|
payload=None,
|
|
):
|
|
"""Create a API formatted error response.
|
|
|
|
Async friendly.
|
|
"""
|
|
payload = payload or {}
|
|
payload["type"] = error_type
|
|
payload["message"] = error_message
|
|
|
|
_LOGGER.info(
|
|
"Request %s/%s error %s: %s",
|
|
self._directive[API_HEADER]["namespace"],
|
|
self._directive[API_HEADER]["name"],
|
|
error_type,
|
|
error_message,
|
|
)
|
|
|
|
return self.response(name="ErrorResponse", namespace=namespace, payload=payload)
|
|
|
|
|
|
class AlexaResponse:
|
|
"""Class to hold a response."""
|
|
|
|
def __init__(self, name, namespace, payload=None):
|
|
"""Initialize the response."""
|
|
payload = payload or {}
|
|
self._response = {
|
|
API_EVENT: {
|
|
API_HEADER: {
|
|
"namespace": namespace,
|
|
"name": name,
|
|
"messageId": str(uuid4()),
|
|
"payloadVersion": "3",
|
|
},
|
|
API_PAYLOAD: payload,
|
|
}
|
|
}
|
|
|
|
@property
|
|
def name(self):
|
|
"""Return the name of this response."""
|
|
return self._response[API_EVENT][API_HEADER]["name"]
|
|
|
|
@property
|
|
def namespace(self):
|
|
"""Return the namespace of this response."""
|
|
return self._response[API_EVENT][API_HEADER]["namespace"]
|
|
|
|
def set_correlation_token(self, token):
|
|
"""Set the correlationToken.
|
|
|
|
This should normally mirror the value from a request, and is set by
|
|
AlexaDirective.response() usually.
|
|
"""
|
|
self._response[API_EVENT][API_HEADER]["correlationToken"] = token
|
|
|
|
def set_endpoint_full(self, bearer_token, endpoint_id, cookie=None):
|
|
"""Set the endpoint dictionary.
|
|
|
|
This is used to send proactive messages to Alexa.
|
|
"""
|
|
self._response[API_EVENT][API_ENDPOINT] = {
|
|
API_SCOPE: {"type": "BearerToken", "token": bearer_token}
|
|
}
|
|
|
|
if endpoint_id is not None:
|
|
self._response[API_EVENT][API_ENDPOINT]["endpointId"] = endpoint_id
|
|
|
|
if cookie is not None:
|
|
self._response[API_EVENT][API_ENDPOINT]["cookie"] = cookie
|
|
|
|
def set_endpoint(self, endpoint):
|
|
"""Set the endpoint.
|
|
|
|
This should normally mirror the value from a request, and is set by
|
|
AlexaDirective.response() usually.
|
|
"""
|
|
self._response[API_EVENT][API_ENDPOINT] = endpoint
|
|
|
|
def _properties(self):
|
|
context = self._response.setdefault(API_CONTEXT, {})
|
|
return context.setdefault("properties", [])
|
|
|
|
def add_context_property(self, prop):
|
|
"""Add a property to the response context.
|
|
|
|
The Alexa response includes a list of properties which provides
|
|
feedback on how states have changed. For example if a user asks,
|
|
"Alexa, set thermostat to 20 degrees", the API expects a response with
|
|
the new value of the property, and Alexa will respond to the user
|
|
"Thermostat set to 20 degrees".
|
|
|
|
async_handle_message() will call .merge_context_properties() for every
|
|
request automatically, however often handlers will call services to
|
|
change state but the effects of those changes are applied
|
|
asynchronously. Thus, handlers should call this method to confirm
|
|
changes before returning.
|
|
"""
|
|
self._properties().append(prop)
|
|
|
|
def merge_context_properties(self, endpoint):
|
|
"""Add all properties from given endpoint if not already set.
|
|
|
|
Handlers should be using .add_context_property().
|
|
"""
|
|
properties = self._properties()
|
|
already_set = {(p["namespace"], p["name"]) for p in properties}
|
|
|
|
for prop in endpoint.serialize_properties():
|
|
if (prop["namespace"], prop["name"]) not in already_set:
|
|
self.add_context_property(prop)
|
|
|
|
def serialize(self):
|
|
"""Return response as a JSON-able data structure."""
|
|
return self._response
|