* Filter some Alexa reports that are duplicate * When state changes during reporting, only report last state, not all state changes
310 lines
9.8 KiB
Python
310 lines
9.8 KiB
Python
"""Alexa state report code."""
|
|
import asyncio
|
|
import json
|
|
import logging
|
|
from typing import Dict, Optional
|
|
|
|
import aiohttp
|
|
import async_timeout
|
|
|
|
from homeassistant.const import HTTP_ACCEPTED, MATCH_ALL, STATE_ON
|
|
from homeassistant.core import State
|
|
import homeassistant.util.dt as dt_util
|
|
|
|
from .const import API_CHANGE, Cause
|
|
from .entities import ENTITY_ADAPTERS, AlexaEntity, generate_alexa_id
|
|
from .messages import AlexaResponse
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
DEFAULT_TIMEOUT = 10
|
|
|
|
|
|
async def async_enable_proactive_mode(hass, smart_home_config):
|
|
"""Enable the proactive mode.
|
|
|
|
Proactive mode makes this component report state changes to Alexa.
|
|
"""
|
|
# Validate we can get access token.
|
|
await smart_home_config.async_get_access_token()
|
|
|
|
progress: Dict[str, AlexaEntity] = {}
|
|
|
|
async def async_entity_state_listener(
|
|
changed_entity: str,
|
|
old_state: Optional[State],
|
|
new_state: Optional[State],
|
|
):
|
|
if not hass.is_running:
|
|
return
|
|
|
|
if not new_state:
|
|
return
|
|
|
|
if new_state.domain not in ENTITY_ADAPTERS:
|
|
return
|
|
|
|
if not smart_home_config.should_expose(changed_entity):
|
|
_LOGGER.debug("Not exposing %s because filtered by config", changed_entity)
|
|
return
|
|
|
|
alexa_changed_entity: AlexaEntity = ENTITY_ADAPTERS[new_state.domain](
|
|
hass, smart_home_config, new_state
|
|
)
|
|
|
|
# Queue up entity to be sent later.
|
|
# If two states come in while we are reporting the state, only the last one will be reported.
|
|
if changed_entity in progress:
|
|
progress[changed_entity] = alexa_changed_entity
|
|
return
|
|
|
|
# Determine how entity should be reported on
|
|
should_report = False
|
|
should_doorbell = False
|
|
|
|
for interface in alexa_changed_entity.interfaces():
|
|
if not should_report and interface.properties_proactively_reported():
|
|
should_report = True
|
|
|
|
if (
|
|
interface.name() == "Alexa.DoorbellEventSource"
|
|
and new_state.state == STATE_ON
|
|
):
|
|
should_doorbell = True
|
|
break
|
|
|
|
if not should_report and not should_doorbell:
|
|
return
|
|
|
|
if should_doorbell:
|
|
should_report = False
|
|
|
|
# Store current state change information
|
|
last_state: Optional[AlexaEntity] = None
|
|
if old_state:
|
|
last_state = ENTITY_ADAPTERS[old_state.domain](
|
|
hass, smart_home_config, old_state
|
|
)
|
|
progress[changed_entity] = alexa_changed_entity
|
|
|
|
# Start reporting on entity. Keep reporting as long as new states come in
|
|
# while we were reporting a state.
|
|
while last_state != progress[changed_entity]:
|
|
to_report = progress[changed_entity]
|
|
alexa_properties = None
|
|
|
|
if should_report:
|
|
# this sends all the properties of the Alexa Entity, whether they have
|
|
# changed or not. this should be improved, and properties that have not
|
|
# changed should be moved to the 'context' object
|
|
alexa_properties = list(alexa_changed_entity.serialize_properties())
|
|
|
|
if last_state and last_state.entity.state == to_report.entity.state:
|
|
old_alexa_properties = list(last_state.serialize_properties())
|
|
if old_alexa_properties == alexa_properties:
|
|
return
|
|
|
|
try:
|
|
if should_report:
|
|
await async_send_changereport_message(
|
|
hass, smart_home_config, alexa_changed_entity, alexa_properties
|
|
)
|
|
|
|
elif should_doorbell:
|
|
await async_send_doorbell_event_message(
|
|
hass, smart_home_config, alexa_changed_entity
|
|
)
|
|
except Exception:
|
|
progress.pop(changed_entity)
|
|
raise
|
|
|
|
last_state = to_report
|
|
|
|
progress.pop(changed_entity)
|
|
|
|
return hass.helpers.event.async_track_state_change(
|
|
MATCH_ALL, async_entity_state_listener
|
|
)
|
|
|
|
|
|
async def async_send_changereport_message(
|
|
hass, config, alexa_entity, properties, *, invalidate_access_token=True
|
|
):
|
|
"""Send a ChangeReport message for an Alexa entity.
|
|
|
|
https://developer.amazon.com/docs/smarthome/state-reporting-for-a-smart-home-skill.html#report-state-with-changereport-events
|
|
"""
|
|
token = await config.async_get_access_token()
|
|
|
|
headers = {"Authorization": f"Bearer {token}"}
|
|
|
|
endpoint = alexa_entity.alexa_id()
|
|
|
|
payload = {
|
|
API_CHANGE: {"cause": {"type": Cause.APP_INTERACTION}, "properties": properties}
|
|
}
|
|
|
|
message = AlexaResponse(name="ChangeReport", namespace="Alexa", payload=payload)
|
|
message.set_endpoint_full(token, endpoint)
|
|
|
|
message_serialized = message.serialize()
|
|
session = hass.helpers.aiohttp_client.async_get_clientsession()
|
|
|
|
try:
|
|
with async_timeout.timeout(DEFAULT_TIMEOUT):
|
|
response = await session.post(
|
|
config.endpoint,
|
|
headers=headers,
|
|
json=message_serialized,
|
|
allow_redirects=True,
|
|
)
|
|
|
|
except (asyncio.TimeoutError, aiohttp.ClientError):
|
|
_LOGGER.error("Timeout sending report to Alexa")
|
|
return
|
|
|
|
response_text = await response.text()
|
|
|
|
_LOGGER.debug("Sent: %s", json.dumps(message_serialized))
|
|
_LOGGER.debug("Received (%s): %s", response.status, response_text)
|
|
|
|
if response.status == HTTP_ACCEPTED:
|
|
return
|
|
|
|
response_json = json.loads(response_text)
|
|
|
|
if (
|
|
response_json["payload"]["code"] == "INVALID_ACCESS_TOKEN_EXCEPTION"
|
|
and not invalidate_access_token
|
|
):
|
|
config.async_invalidate_access_token()
|
|
return await async_send_changereport_message(
|
|
hass, config, alexa_entity, properties, invalidate_access_token=False
|
|
)
|
|
|
|
_LOGGER.error(
|
|
"Error when sending ChangeReport to Alexa: %s: %s",
|
|
response_json["payload"]["code"],
|
|
response_json["payload"]["description"],
|
|
)
|
|
|
|
|
|
async def async_send_add_or_update_message(hass, config, entity_ids):
|
|
"""Send an AddOrUpdateReport message for entities.
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-discovery.html#add-or-update-report
|
|
"""
|
|
token = await config.async_get_access_token()
|
|
|
|
headers = {"Authorization": f"Bearer {token}"}
|
|
|
|
endpoints = []
|
|
|
|
for entity_id in entity_ids:
|
|
domain = entity_id.split(".", 1)[0]
|
|
|
|
if domain not in ENTITY_ADAPTERS:
|
|
continue
|
|
|
|
alexa_entity = ENTITY_ADAPTERS[domain](hass, config, hass.states.get(entity_id))
|
|
endpoints.append(alexa_entity.serialize_discovery())
|
|
|
|
payload = {"endpoints": endpoints, "scope": {"type": "BearerToken", "token": token}}
|
|
|
|
message = AlexaResponse(
|
|
name="AddOrUpdateReport", namespace="Alexa.Discovery", payload=payload
|
|
)
|
|
|
|
message_serialized = message.serialize()
|
|
session = hass.helpers.aiohttp_client.async_get_clientsession()
|
|
|
|
return await session.post(
|
|
config.endpoint, headers=headers, json=message_serialized, allow_redirects=True
|
|
)
|
|
|
|
|
|
async def async_send_delete_message(hass, config, entity_ids):
|
|
"""Send an DeleteReport message for entities.
|
|
|
|
https://developer.amazon.com/docs/device-apis/alexa-discovery.html#deletereport-event
|
|
"""
|
|
token = await config.async_get_access_token()
|
|
|
|
headers = {"Authorization": f"Bearer {token}"}
|
|
|
|
endpoints = []
|
|
|
|
for entity_id in entity_ids:
|
|
domain = entity_id.split(".", 1)[0]
|
|
|
|
if domain not in ENTITY_ADAPTERS:
|
|
continue
|
|
|
|
endpoints.append({"endpointId": generate_alexa_id(entity_id)})
|
|
|
|
payload = {"endpoints": endpoints, "scope": {"type": "BearerToken", "token": token}}
|
|
|
|
message = AlexaResponse(
|
|
name="DeleteReport", namespace="Alexa.Discovery", payload=payload
|
|
)
|
|
|
|
message_serialized = message.serialize()
|
|
session = hass.helpers.aiohttp_client.async_get_clientsession()
|
|
|
|
return await session.post(
|
|
config.endpoint, headers=headers, json=message_serialized, allow_redirects=True
|
|
)
|
|
|
|
|
|
async def async_send_doorbell_event_message(hass, config, alexa_entity):
|
|
"""Send a DoorbellPress event message for an Alexa entity.
|
|
|
|
https://developer.amazon.com/docs/smarthome/send-events-to-the-alexa-event-gateway.html
|
|
"""
|
|
token = await config.async_get_access_token()
|
|
|
|
headers = {"Authorization": f"Bearer {token}"}
|
|
|
|
endpoint = alexa_entity.alexa_id()
|
|
|
|
message = AlexaResponse(
|
|
name="DoorbellPress",
|
|
namespace="Alexa.DoorbellEventSource",
|
|
payload={
|
|
"cause": {"type": Cause.PHYSICAL_INTERACTION},
|
|
"timestamp": f"{dt_util.utcnow().replace(tzinfo=None).isoformat()}Z",
|
|
},
|
|
)
|
|
|
|
message.set_endpoint_full(token, endpoint)
|
|
|
|
message_serialized = message.serialize()
|
|
session = hass.helpers.aiohttp_client.async_get_clientsession()
|
|
|
|
try:
|
|
with async_timeout.timeout(DEFAULT_TIMEOUT):
|
|
response = await session.post(
|
|
config.endpoint,
|
|
headers=headers,
|
|
json=message_serialized,
|
|
allow_redirects=True,
|
|
)
|
|
|
|
except (asyncio.TimeoutError, aiohttp.ClientError):
|
|
_LOGGER.error("Timeout sending report to Alexa")
|
|
return
|
|
|
|
response_text = await response.text()
|
|
|
|
_LOGGER.debug("Sent: %s", json.dumps(message_serialized))
|
|
_LOGGER.debug("Received (%s): %s", response.status, response_text)
|
|
|
|
if response.status == HTTP_ACCEPTED:
|
|
return
|
|
|
|
response_json = json.loads(response_text)
|
|
|
|
_LOGGER.error(
|
|
"Error when sending DoorbellPress event to Alexa: %s: %s",
|
|
response_json["payload"]["code"],
|
|
response_json["payload"]["description"],
|
|
)
|