mirror of
https://github.com/home-assistant/core.git
synced 2025-11-09 02:49:40 +00:00
191 lines
6.3 KiB
Python
191 lines
6.3 KiB
Python
"""Support for Tuya Alarm."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from base64 import b64decode
|
|
from dataclasses import dataclass
|
|
from enum import StrEnum
|
|
|
|
from tuya_sharing import CustomerDevice, Manager
|
|
|
|
from homeassistant.components.alarm_control_panel import (
|
|
AlarmControlPanelEntity,
|
|
AlarmControlPanelEntityDescription,
|
|
AlarmControlPanelEntityFeature,
|
|
AlarmControlPanelState,
|
|
)
|
|
from homeassistant.core import HomeAssistant, callback
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_connect
|
|
from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
|
|
|
|
from . import TuyaConfigEntry
|
|
from .const import TUYA_DISCOVERY_NEW, DeviceCategory, DPCode, DPType
|
|
from .entity import TuyaEntity
|
|
from .models import EnumTypeData
|
|
from .util import get_dpcode
|
|
|
|
|
|
@dataclass(frozen=True)
|
|
class TuyaAlarmControlPanelEntityDescription(AlarmControlPanelEntityDescription):
|
|
"""Describe a Tuya Alarm Control Panel entity."""
|
|
|
|
master_state: DPCode | None = None
|
|
alarm_msg: DPCode | None = None
|
|
|
|
|
|
class Mode(StrEnum):
|
|
"""Alarm modes."""
|
|
|
|
ARM = "arm"
|
|
DISARMED = "disarmed"
|
|
HOME = "home"
|
|
SOS = "sos"
|
|
|
|
|
|
class State(StrEnum):
|
|
"""Alarm states."""
|
|
|
|
NORMAL = "normal"
|
|
ALARM = "alarm"
|
|
|
|
|
|
STATE_MAPPING: dict[str, AlarmControlPanelState] = {
|
|
Mode.DISARMED: AlarmControlPanelState.DISARMED,
|
|
Mode.ARM: AlarmControlPanelState.ARMED_AWAY,
|
|
Mode.HOME: AlarmControlPanelState.ARMED_HOME,
|
|
Mode.SOS: AlarmControlPanelState.TRIGGERED,
|
|
}
|
|
|
|
|
|
ALARM: dict[DeviceCategory, tuple[TuyaAlarmControlPanelEntityDescription, ...]] = {
|
|
DeviceCategory.MAL: (
|
|
TuyaAlarmControlPanelEntityDescription(
|
|
key=DPCode.MASTER_MODE,
|
|
master_state=DPCode.MASTER_STATE,
|
|
alarm_msg=DPCode.ALARM_MSG,
|
|
name="Alarm",
|
|
),
|
|
)
|
|
}
|
|
|
|
|
|
async def async_setup_entry(
|
|
hass: HomeAssistant,
|
|
entry: TuyaConfigEntry,
|
|
async_add_entities: AddConfigEntryEntitiesCallback,
|
|
) -> None:
|
|
"""Set up Tuya alarm dynamically through Tuya discovery."""
|
|
manager = entry.runtime_data.manager
|
|
|
|
@callback
|
|
def async_discover_device(device_ids: list[str]) -> None:
|
|
"""Discover and add a discovered Tuya siren."""
|
|
entities: list[TuyaAlarmEntity] = []
|
|
for device_id in device_ids:
|
|
device = manager.device_map[device_id]
|
|
if descriptions := ALARM.get(device.category):
|
|
entities.extend(
|
|
TuyaAlarmEntity(device, manager, description)
|
|
for description in descriptions
|
|
if description.key in device.status
|
|
)
|
|
async_add_entities(entities)
|
|
|
|
async_discover_device([*manager.device_map])
|
|
|
|
entry.async_on_unload(
|
|
async_dispatcher_connect(hass, TUYA_DISCOVERY_NEW, async_discover_device)
|
|
)
|
|
|
|
|
|
class TuyaAlarmEntity(TuyaEntity, AlarmControlPanelEntity):
|
|
"""Tuya Alarm Entity."""
|
|
|
|
_attr_name = None
|
|
_attr_code_arm_required = False
|
|
_master_state: EnumTypeData | None = None
|
|
_alarm_msg_dpcode: DPCode | None = None
|
|
|
|
def __init__(
|
|
self,
|
|
device: CustomerDevice,
|
|
device_manager: Manager,
|
|
description: TuyaAlarmControlPanelEntityDescription,
|
|
) -> None:
|
|
"""Init Tuya Alarm."""
|
|
super().__init__(device, device_manager)
|
|
self.entity_description = description
|
|
self._attr_unique_id = f"{super().unique_id}{description.key}"
|
|
|
|
# Determine supported modes
|
|
if supported_modes := self.find_dpcode(
|
|
description.key, dptype=DPType.ENUM, prefer_function=True
|
|
):
|
|
if Mode.HOME in supported_modes.range:
|
|
self._attr_supported_features |= AlarmControlPanelEntityFeature.ARM_HOME
|
|
|
|
if Mode.ARM in supported_modes.range:
|
|
self._attr_supported_features |= AlarmControlPanelEntityFeature.ARM_AWAY
|
|
|
|
if Mode.SOS in supported_modes.range:
|
|
self._attr_supported_features |= AlarmControlPanelEntityFeature.TRIGGER
|
|
|
|
# Determine master state
|
|
if enum_type := self.find_dpcode(
|
|
description.master_state, dptype=DPType.ENUM, prefer_function=True
|
|
):
|
|
self._master_state = enum_type
|
|
|
|
# Determine alarm message
|
|
if dp_code := get_dpcode(self.device, description.alarm_msg):
|
|
self._alarm_msg_dpcode = dp_code
|
|
|
|
@property
|
|
def alarm_state(self) -> AlarmControlPanelState | None:
|
|
"""Return the state of the device."""
|
|
# When the alarm is triggered, only its 'state' is changing. From 'normal' to 'alarm'.
|
|
# The 'mode' doesn't change, and stays as 'arm' or 'home'.
|
|
if (
|
|
self._master_state is not None
|
|
and self.device.status.get(self._master_state.dpcode) == State.ALARM
|
|
):
|
|
# Only report as triggered if NOT a battery warning
|
|
if (
|
|
changed_by := self.changed_by
|
|
) is None or "Sensor Low Battery" not in changed_by:
|
|
return AlarmControlPanelState.TRIGGERED
|
|
|
|
if not (status := self.device.status.get(self.entity_description.key)):
|
|
return None
|
|
return STATE_MAPPING.get(status)
|
|
|
|
@property
|
|
def changed_by(self) -> str | None:
|
|
"""Last change triggered by."""
|
|
if (
|
|
self._master_state is not None
|
|
and self._alarm_msg_dpcode is not None
|
|
and self.device.status.get(self._master_state.dpcode) == State.ALARM
|
|
and (encoded_msg := self.device.status.get(self._alarm_msg_dpcode))
|
|
):
|
|
return b64decode(encoded_msg).decode("utf-16be")
|
|
return None
|
|
|
|
def alarm_disarm(self, code: str | None = None) -> None:
|
|
"""Send Disarm command."""
|
|
self._send_command(
|
|
[{"code": self.entity_description.key, "value": Mode.DISARMED}]
|
|
)
|
|
|
|
def alarm_arm_home(self, code: str | None = None) -> None:
|
|
"""Send Home command."""
|
|
self._send_command([{"code": self.entity_description.key, "value": Mode.HOME}])
|
|
|
|
def alarm_arm_away(self, code: str | None = None) -> None:
|
|
"""Send Arm command."""
|
|
self._send_command([{"code": self.entity_description.key, "value": Mode.ARM}])
|
|
|
|
def alarm_trigger(self, code: str | None = None) -> None:
|
|
"""Send SOS command."""
|
|
self._send_command([{"code": self.entity_description.key, "value": Mode.SOS}])
|