310 lines
10 KiB
Python
310 lines
10 KiB
Python
"""Provides device automations for homekit devices."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from collections.abc import Callable
|
|
from typing import TYPE_CHECKING, Any
|
|
|
|
from aiohomekit.model.characteristics import CharacteristicsTypes
|
|
from aiohomekit.model.characteristics.const import InputEventValues
|
|
from aiohomekit.model.services import Service, ServicesTypes
|
|
from aiohomekit.utils import clamp_enum_to_char
|
|
from typing_extensions import Generator
|
|
import voluptuous as vol
|
|
|
|
from homeassistant.components.device_automation import DEVICE_TRIGGER_BASE_SCHEMA
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.const import CONF_DEVICE_ID, CONF_DOMAIN, CONF_PLATFORM, CONF_TYPE
|
|
from homeassistant.core import CALLBACK_TYPE, HassJob, HomeAssistant, callback
|
|
from homeassistant.helpers.trigger import TriggerActionType, TriggerInfo
|
|
from homeassistant.helpers.typing import ConfigType
|
|
|
|
from .const import DOMAIN, KNOWN_DEVICES, TRIGGERS
|
|
|
|
if TYPE_CHECKING:
|
|
from .connection import HKDevice
|
|
|
|
TRIGGER_TYPES = {
|
|
"doorbell",
|
|
"button1",
|
|
"button2",
|
|
"button3",
|
|
"button4",
|
|
"button5",
|
|
"button6",
|
|
"button7",
|
|
"button8",
|
|
"button9",
|
|
"button10",
|
|
}
|
|
TRIGGER_SUBTYPES = {"single_press", "double_press", "long_press"}
|
|
|
|
CONF_IID = "iid"
|
|
CONF_SUBTYPE = "subtype"
|
|
|
|
TRIGGER_SCHEMA = DEVICE_TRIGGER_BASE_SCHEMA.extend(
|
|
{
|
|
vol.Required(CONF_TYPE): vol.In(TRIGGER_TYPES),
|
|
vol.Required(CONF_SUBTYPE): vol.In(TRIGGER_SUBTYPES),
|
|
}
|
|
)
|
|
|
|
HK_TO_HA_INPUT_EVENT_VALUES = {
|
|
InputEventValues.SINGLE_PRESS: "single_press",
|
|
InputEventValues.DOUBLE_PRESS: "double_press",
|
|
InputEventValues.LONG_PRESS: "long_press",
|
|
}
|
|
|
|
|
|
class TriggerSource:
|
|
"""Represents a stateless source of event data from HomeKit."""
|
|
|
|
def __init__(self, hass: HomeAssistant) -> None:
|
|
"""Initialize a set of triggers for a device."""
|
|
self._hass = hass
|
|
self._triggers: dict[tuple[str, str], dict[str, Any]] = {}
|
|
self._callbacks: dict[tuple[str, str], list[Callable[[Any], None]]] = {}
|
|
self._iid_trigger_keys: dict[int, set[tuple[str, str]]] = {}
|
|
|
|
@callback
|
|
def async_setup(
|
|
self, connection: HKDevice, aid: int, triggers: list[dict[str, Any]]
|
|
) -> None:
|
|
"""Set up a set of triggers for a device.
|
|
|
|
This function must be re-entrant since
|
|
it is called when the device is first added and
|
|
when the config entry is reloaded.
|
|
"""
|
|
for trigger_data in triggers:
|
|
trigger_key = (trigger_data[CONF_TYPE], trigger_data[CONF_SUBTYPE])
|
|
self._triggers[trigger_key] = trigger_data
|
|
iid = trigger_data["characteristic"]
|
|
self._iid_trigger_keys.setdefault(iid, set()).add(trigger_key)
|
|
connection.add_watchable_characteristics([(aid, iid)])
|
|
|
|
def fire(self, iid: int, ev: dict[str, Any]) -> None:
|
|
"""Process events that have been received from a HomeKit accessory."""
|
|
for trigger_key in self._iid_trigger_keys.get(iid, set()):
|
|
for event_handler in self._callbacks.get(trigger_key, []):
|
|
event_handler(ev)
|
|
|
|
def async_get_triggers(self) -> Generator[tuple[str, str]]:
|
|
"""List device triggers for HomeKit devices."""
|
|
yield from self._triggers
|
|
|
|
@callback
|
|
def async_attach_trigger(
|
|
self,
|
|
config: ConfigType,
|
|
action: TriggerActionType,
|
|
trigger_info: TriggerInfo,
|
|
) -> CALLBACK_TYPE:
|
|
"""Attach a trigger."""
|
|
trigger_data = trigger_info["trigger_data"]
|
|
type_: str = config[CONF_TYPE]
|
|
sub_type: str = config[CONF_SUBTYPE]
|
|
trigger_key = (type_, sub_type)
|
|
job = HassJob(action)
|
|
trigger_callbacks = self._callbacks.setdefault(trigger_key, [])
|
|
hass = self._hass
|
|
|
|
@callback
|
|
def event_handler(ev: dict[str, Any]) -> None:
|
|
if sub_type != HK_TO_HA_INPUT_EVENT_VALUES[ev["value"]]:
|
|
return
|
|
hass.async_run_hass_job(job, {"trigger": {**trigger_data, **config}})
|
|
|
|
trigger_callbacks.append(event_handler)
|
|
|
|
def async_remove_handler() -> None:
|
|
trigger_callbacks.remove(event_handler)
|
|
|
|
return async_remove_handler
|
|
|
|
|
|
def enumerate_stateless_switch(service: Service) -> list[dict[str, Any]]:
|
|
"""Enumerate a stateless switch, like a single button."""
|
|
|
|
# A stateless switch that has a SERVICE_LABEL_INDEX is part of a group
|
|
# And is handled separately
|
|
if (
|
|
service.has(CharacteristicsTypes.SERVICE_LABEL_INDEX)
|
|
and len(service.linked) > 0
|
|
):
|
|
return []
|
|
|
|
char = service[CharacteristicsTypes.INPUT_EVENT]
|
|
|
|
# HomeKit itself supports single, double and long presses. But the
|
|
# manufacturer might not - clamp options to what they say.
|
|
all_values = clamp_enum_to_char(InputEventValues, char)
|
|
|
|
return [
|
|
{
|
|
"characteristic": char.iid,
|
|
"value": event_type,
|
|
"type": "button1",
|
|
"subtype": HK_TO_HA_INPUT_EVENT_VALUES[event_type],
|
|
}
|
|
for event_type in all_values
|
|
]
|
|
|
|
|
|
def enumerate_stateless_switch_group(service: Service) -> list[dict[str, Any]]:
|
|
"""Enumerate a group of stateless switches, like a remote control."""
|
|
switches = list(
|
|
service.accessory.services.filter(
|
|
service_type=ServicesTypes.STATELESS_PROGRAMMABLE_SWITCH,
|
|
child_service=service,
|
|
order_by=[CharacteristicsTypes.SERVICE_LABEL_INDEX],
|
|
)
|
|
)
|
|
|
|
results: list[dict[str, Any]] = []
|
|
for idx, switch in enumerate(switches):
|
|
char = switch[CharacteristicsTypes.INPUT_EVENT]
|
|
|
|
# HomeKit itself supports single, double and long presses. But the
|
|
# manufacturer might not - clamp options to what they say.
|
|
all_values = clamp_enum_to_char(InputEventValues, char)
|
|
|
|
results.extend(
|
|
{
|
|
"characteristic": char.iid,
|
|
"value": event_type,
|
|
"type": f"button{idx + 1}",
|
|
"subtype": HK_TO_HA_INPUT_EVENT_VALUES[event_type],
|
|
}
|
|
for event_type in all_values
|
|
)
|
|
return results
|
|
|
|
|
|
def enumerate_doorbell(service: Service) -> list[dict[str, Any]]:
|
|
"""Enumerate doorbell buttons."""
|
|
input_event = service[CharacteristicsTypes.INPUT_EVENT]
|
|
|
|
# HomeKit itself supports single, double and long presses. But the
|
|
# manufacturer might not - clamp options to what they say.
|
|
all_values = clamp_enum_to_char(InputEventValues, input_event)
|
|
|
|
return [
|
|
{
|
|
"characteristic": input_event.iid,
|
|
"value": event_type,
|
|
"type": "doorbell",
|
|
"subtype": HK_TO_HA_INPUT_EVENT_VALUES[event_type],
|
|
}
|
|
for event_type in all_values
|
|
]
|
|
|
|
|
|
TRIGGER_FINDERS = {
|
|
ServicesTypes.SERVICE_LABEL: enumerate_stateless_switch_group,
|
|
ServicesTypes.STATELESS_PROGRAMMABLE_SWITCH: enumerate_stateless_switch,
|
|
ServicesTypes.DOORBELL: enumerate_doorbell,
|
|
}
|
|
|
|
|
|
async def async_setup_triggers_for_entry(
|
|
hass: HomeAssistant, config_entry: ConfigEntry
|
|
) -> None:
|
|
"""Triggers aren't entities as they have no state, but we still need to set them up for a config entry."""
|
|
hkid = config_entry.data["AccessoryPairingID"]
|
|
conn: HKDevice = hass.data[KNOWN_DEVICES][hkid]
|
|
|
|
@callback
|
|
def async_add_characteristic(service: Service) -> bool:
|
|
aid = service.accessory.aid
|
|
service_type = service.type
|
|
|
|
# If not a known service type then we can't handle any stateless events for it
|
|
if service_type not in TRIGGER_FINDERS:
|
|
return False
|
|
|
|
# We can't have multiple trigger sources for the same device id
|
|
# Can't have a doorbell and a remote control in the same accessory
|
|
# They have to be different accessories (they can be on the same bridge)
|
|
# In practice, this is inline with what iOS actually supports AFAWCT.
|
|
device_id = conn.devices[aid]
|
|
if TRIGGERS in hass.data and device_id in hass.data[TRIGGERS]:
|
|
return False
|
|
|
|
# Just because we recognize the service type doesn't mean we can actually
|
|
# extract any triggers - so only proceed if we can
|
|
triggers = TRIGGER_FINDERS[service_type](service)
|
|
if len(triggers) == 0:
|
|
return False
|
|
|
|
trigger = async_get_or_create_trigger_source(conn.hass, device_id)
|
|
trigger.async_setup(conn, aid, triggers)
|
|
|
|
return True
|
|
|
|
conn.add_trigger_factory(async_add_characteristic)
|
|
|
|
|
|
@callback
|
|
def async_get_or_create_trigger_source(
|
|
hass: HomeAssistant, device_id: str
|
|
) -> TriggerSource:
|
|
"""Get or create a trigger source for a device id."""
|
|
trigger_sources: dict[str, TriggerSource] = hass.data.setdefault(TRIGGERS, {})
|
|
if not (source := trigger_sources.get(device_id)):
|
|
source = TriggerSource(hass)
|
|
trigger_sources[device_id] = source
|
|
return source
|
|
|
|
|
|
def async_fire_triggers(
|
|
conn: HKDevice, events: dict[tuple[int, int], dict[str, Any]]
|
|
) -> None:
|
|
"""Process events generated by a HomeKit accessory into automation triggers."""
|
|
trigger_sources: dict[str, TriggerSource] = conn.hass.data.get(TRIGGERS, {})
|
|
if not trigger_sources:
|
|
return
|
|
for (aid, iid), ev in events.items():
|
|
if aid in conn.devices:
|
|
device_id = conn.devices[aid]
|
|
if source := trigger_sources.get(device_id):
|
|
# If the value is None, we received the event via polling
|
|
# and we don't want to trigger on that
|
|
if ev.get("value") is not None:
|
|
source.fire(iid, ev)
|
|
|
|
|
|
async def async_get_triggers(
|
|
hass: HomeAssistant, device_id: str
|
|
) -> list[dict[str, str]]:
|
|
"""List device triggers for homekit devices."""
|
|
|
|
if device_id not in hass.data.get(TRIGGERS, {}):
|
|
return []
|
|
|
|
device: TriggerSource = hass.data[TRIGGERS][device_id]
|
|
|
|
return [
|
|
{
|
|
CONF_PLATFORM: "device",
|
|
CONF_DEVICE_ID: device_id,
|
|
CONF_DOMAIN: DOMAIN,
|
|
CONF_TYPE: trigger,
|
|
CONF_SUBTYPE: subtype,
|
|
}
|
|
for trigger, subtype in device.async_get_triggers()
|
|
]
|
|
|
|
|
|
async def async_attach_trigger(
|
|
hass: HomeAssistant,
|
|
config: ConfigType,
|
|
action: TriggerActionType,
|
|
trigger_info: TriggerInfo,
|
|
) -> CALLBACK_TYPE:
|
|
"""Attach a trigger."""
|
|
device_id = config[CONF_DEVICE_ID]
|
|
return async_get_or_create_trigger_source(hass, device_id).async_attach_trigger(
|
|
config, action, trigger_info
|
|
)
|