267 lines
9.7 KiB
Python
267 lines
9.7 KiB
Python
"""Representation of ISYEntity Types."""
|
|
from __future__ import annotations
|
|
|
|
from typing import Any, cast
|
|
|
|
from pyisy.constants import (
|
|
ATTR_ACTION,
|
|
ATTR_CONTROL,
|
|
COMMAND_FRIENDLY_NAME,
|
|
EMPTY_TIME,
|
|
EVENT_PROPS_IGNORED,
|
|
NC_NODE_ENABLED,
|
|
PROTO_INSTEON,
|
|
PROTO_ZWAVE,
|
|
TAG_ADDRESS,
|
|
TAG_ENABLED,
|
|
)
|
|
from pyisy.helpers import EventListener, NodeProperty
|
|
from pyisy.nodes import Group, Node, NodeChangedEvent
|
|
from pyisy.programs import Program
|
|
from pyisy.variables import Variable
|
|
|
|
from homeassistant.const import STATE_OFF, STATE_ON
|
|
from homeassistant.core import callback
|
|
from homeassistant.exceptions import HomeAssistantError
|
|
from homeassistant.helpers.entity import DeviceInfo, Entity, EntityDescription
|
|
|
|
from .const import DOMAIN
|
|
|
|
|
|
class ISYEntity(Entity):
|
|
"""Representation of an ISY device."""
|
|
|
|
_attr_has_entity_name = False
|
|
_attr_should_poll = False
|
|
_node: Node | Program | Variable
|
|
|
|
def __init__(
|
|
self,
|
|
node: Node | Group | Variable | Program,
|
|
device_info: DeviceInfo | None = None,
|
|
) -> None:
|
|
"""Initialize the ISY/IoX entity."""
|
|
self._node = node
|
|
self._attr_name = node.name
|
|
if device_info is None:
|
|
device_info = DeviceInfo(identifiers={(DOMAIN, node.isy.uuid)})
|
|
self._attr_device_info = device_info
|
|
self._attr_unique_id = f"{node.isy.uuid}_{node.address}"
|
|
self._attrs: dict[str, Any] = {}
|
|
self._change_handler: EventListener | None = None
|
|
self._control_handler: EventListener | None = None
|
|
|
|
async def async_added_to_hass(self) -> None:
|
|
"""Subscribe to the node change events."""
|
|
self._change_handler = self._node.status_events.subscribe(self.async_on_update)
|
|
|
|
if hasattr(self._node, "control_events"):
|
|
self._control_handler = self._node.control_events.subscribe(
|
|
self.async_on_control
|
|
)
|
|
|
|
@callback
|
|
def async_on_update(self, event: NodeProperty) -> None:
|
|
"""Handle the update event from the ISY Node."""
|
|
self.async_write_ha_state()
|
|
|
|
@callback
|
|
def async_on_control(self, event: NodeProperty) -> None:
|
|
"""Handle a control event from the ISY Node."""
|
|
event_data = {
|
|
"entity_id": self.entity_id,
|
|
"control": event.control,
|
|
"value": event.value,
|
|
"formatted": event.formatted,
|
|
"uom": event.uom,
|
|
"precision": event.prec,
|
|
}
|
|
|
|
if event.control not in EVENT_PROPS_IGNORED:
|
|
# New state attributes may be available, update the state.
|
|
self.async_write_ha_state()
|
|
|
|
self.hass.bus.async_fire("isy994_control", event_data)
|
|
|
|
|
|
class ISYNodeEntity(ISYEntity):
|
|
"""Representation of a ISY Nodebase (Node/Group) entity."""
|
|
|
|
def __init__(
|
|
self,
|
|
node: Node | Group | Variable | Program,
|
|
device_info: DeviceInfo | None = None,
|
|
) -> None:
|
|
"""Initialize the ISY/IoX node entity."""
|
|
super().__init__(node, device_info=device_info)
|
|
if hasattr(node, "parent_node") and node.parent_node is None:
|
|
self._attr_has_entity_name = True
|
|
self._attr_name = None
|
|
|
|
@property
|
|
def available(self) -> bool:
|
|
"""Return entity availability."""
|
|
return getattr(self._node, TAG_ENABLED, True)
|
|
|
|
@property
|
|
def extra_state_attributes(self) -> dict:
|
|
"""Get the state attributes for the device.
|
|
|
|
The 'aux_properties' in the pyisy Node class are combined with the
|
|
other attributes which have been picked up from the event stream and
|
|
the combined result are returned as the device state attributes.
|
|
"""
|
|
attr = {}
|
|
node = self._node
|
|
# Insteon aux_properties are now their own sensors
|
|
if hasattr(self._node, "aux_properties") and node.protocol != PROTO_INSTEON:
|
|
for name, value in self._node.aux_properties.items():
|
|
attr_name = COMMAND_FRIENDLY_NAME.get(name, name)
|
|
attr[attr_name] = str(value.formatted).lower()
|
|
|
|
# If a Group/Scene, set a property if the entire scene is on/off
|
|
if hasattr(self._node, "group_all_on"):
|
|
attr["group_all_on"] = STATE_ON if self._node.group_all_on else STATE_OFF
|
|
|
|
self._attrs.update(attr)
|
|
return self._attrs
|
|
|
|
async def async_send_node_command(self, command: str) -> None:
|
|
"""Respond to an entity service command call."""
|
|
if not hasattr(self._node, command):
|
|
raise HomeAssistantError(
|
|
f"Invalid service call: {command} for device {self.entity_id}"
|
|
)
|
|
await getattr(self._node, command)()
|
|
|
|
async def async_send_raw_node_command(
|
|
self,
|
|
command: str,
|
|
value: Any | None = None,
|
|
unit_of_measurement: str | None = None,
|
|
parameters: Any | None = None,
|
|
) -> None:
|
|
"""Respond to an entity service raw command call."""
|
|
if not hasattr(self._node, "send_cmd"):
|
|
raise HomeAssistantError(
|
|
f"Invalid service call: {command} for device {self.entity_id}"
|
|
)
|
|
await self._node.send_cmd(command, value, unit_of_measurement, parameters)
|
|
|
|
async def async_get_zwave_parameter(self, parameter: Any) -> None:
|
|
"""Respond to an entity service command to request a Z-Wave device parameter from the ISY."""
|
|
if self._node.protocol != PROTO_ZWAVE:
|
|
raise HomeAssistantError(
|
|
"Invalid service call: cannot request Z-Wave Parameter for non-Z-Wave"
|
|
f" device {self.entity_id}"
|
|
)
|
|
await self._node.get_zwave_parameter(parameter)
|
|
|
|
async def async_set_zwave_parameter(
|
|
self, parameter: Any, value: Any | None, size: int | None
|
|
) -> None:
|
|
"""Respond to an entity service command to set a Z-Wave device parameter via the ISY."""
|
|
if self._node.protocol != PROTO_ZWAVE:
|
|
raise HomeAssistantError(
|
|
"Invalid service call: cannot set Z-Wave Parameter for non-Z-Wave"
|
|
f" device {self.entity_id}"
|
|
)
|
|
await self._node.set_zwave_parameter(parameter, value, size)
|
|
await self._node.get_zwave_parameter(parameter)
|
|
|
|
async def async_rename_node(self, name: str) -> None:
|
|
"""Respond to an entity service command to rename a node on the ISY."""
|
|
await self._node.rename(name)
|
|
|
|
|
|
class ISYProgramEntity(ISYEntity):
|
|
"""Representation of an ISY program base."""
|
|
|
|
_actions: Program
|
|
_status: Program
|
|
|
|
def __init__(self, name: str, status: Program, actions: Program = None) -> None:
|
|
"""Initialize the ISY program-based entity."""
|
|
super().__init__(status)
|
|
self._attr_name = name
|
|
self._actions = actions
|
|
|
|
@property
|
|
def extra_state_attributes(self) -> dict:
|
|
"""Get the state attributes for the device."""
|
|
attr = {}
|
|
if self._actions:
|
|
attr["actions_enabled"] = self._actions.enabled
|
|
if self._actions.last_finished != EMPTY_TIME:
|
|
attr["actions_last_finished"] = self._actions.last_finished
|
|
if self._actions.last_run != EMPTY_TIME:
|
|
attr["actions_last_run"] = self._actions.last_run
|
|
if self._actions.last_update != EMPTY_TIME:
|
|
attr["actions_last_update"] = self._actions.last_update
|
|
attr["ran_else"] = self._actions.ran_else
|
|
attr["ran_then"] = self._actions.ran_then
|
|
attr["run_at_startup"] = self._actions.run_at_startup
|
|
attr["running"] = self._actions.running
|
|
attr["status_enabled"] = self._node.enabled
|
|
if self._node.last_finished != EMPTY_TIME:
|
|
attr["status_last_finished"] = self._node.last_finished
|
|
if self._node.last_run != EMPTY_TIME:
|
|
attr["status_last_run"] = self._node.last_run
|
|
if self._node.last_update != EMPTY_TIME:
|
|
attr["status_last_update"] = self._node.last_update
|
|
return attr
|
|
|
|
|
|
class ISYAuxControlEntity(Entity):
|
|
"""Representation of a ISY/IoX Aux Control base entity."""
|
|
|
|
_attr_should_poll = False
|
|
|
|
def __init__(
|
|
self,
|
|
node: Node,
|
|
control: str,
|
|
unique_id: str,
|
|
description: EntityDescription,
|
|
device_info: DeviceInfo | None,
|
|
) -> None:
|
|
"""Initialize the ISY Aux Control Number entity."""
|
|
self._node = node
|
|
self._control = control
|
|
name = COMMAND_FRIENDLY_NAME.get(control, control).replace("_", " ").title()
|
|
if node.address != node.primary_node:
|
|
name = f"{node.name} {name}"
|
|
self._attr_name = name
|
|
self.entity_description = description
|
|
self._attr_has_entity_name = node.address == node.primary_node
|
|
self._attr_unique_id = unique_id
|
|
self._attr_device_info = device_info
|
|
self._change_handler: EventListener = None
|
|
self._availability_handler: EventListener = None
|
|
|
|
async def async_added_to_hass(self) -> None:
|
|
"""Subscribe to the node control change events."""
|
|
self._change_handler = self._node.control_events.subscribe(
|
|
self.async_on_update,
|
|
event_filter={ATTR_CONTROL: self._control},
|
|
key=self.unique_id,
|
|
)
|
|
self._availability_handler = self._node.isy.nodes.status_events.subscribe(
|
|
self.async_on_update,
|
|
event_filter={
|
|
TAG_ADDRESS: self._node.address,
|
|
ATTR_ACTION: NC_NODE_ENABLED,
|
|
},
|
|
key=self.unique_id,
|
|
)
|
|
|
|
@callback
|
|
def async_on_update(self, event: NodeProperty | NodeChangedEvent, key: str) -> None:
|
|
"""Handle a control event from the ISY Node."""
|
|
self.async_write_ha_state()
|
|
|
|
@property
|
|
def available(self) -> bool:
|
|
"""Return entity availability."""
|
|
return cast(bool, self._node.enabled)
|