2020-05-05 04:03:12 +00:00
|
|
|
"""Representation of ISYEntity Types."""
|
2021-04-12 16:43:14 +00:00
|
|
|
from __future__ import annotations
|
2020-05-05 04:03:12 +00:00
|
|
|
|
2022-02-03 16:02:05 +00:00
|
|
|
from typing import Any, cast
|
|
|
|
|
2020-05-08 04:15:42 +00:00
|
|
|
from pyisy.constants import (
|
|
|
|
COMMAND_FRIENDLY_NAME,
|
|
|
|
EMPTY_TIME,
|
|
|
|
EVENT_PROPS_IGNORED,
|
2020-05-10 00:48:51 +00:00
|
|
|
PROTO_GROUP,
|
2022-05-03 16:49:52 +00:00
|
|
|
PROTO_INSTEON,
|
2020-05-10 00:48:51 +00:00
|
|
|
PROTO_ZWAVE,
|
2020-05-08 04:15:42 +00:00
|
|
|
)
|
2022-02-03 16:02:05 +00:00
|
|
|
from pyisy.helpers import EventListener, NodeProperty
|
|
|
|
from pyisy.nodes import Node
|
|
|
|
from pyisy.programs import Program
|
2020-05-08 04:15:42 +00:00
|
|
|
|
2021-10-23 19:01:34 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
ATTR_IDENTIFIERS,
|
|
|
|
ATTR_MANUFACTURER,
|
|
|
|
ATTR_MODEL,
|
|
|
|
ATTR_NAME,
|
|
|
|
ATTR_SUGGESTED_AREA,
|
|
|
|
STATE_OFF,
|
|
|
|
STATE_ON,
|
|
|
|
)
|
2021-05-18 19:15:47 +00:00
|
|
|
from homeassistant.core import callback
|
|
|
|
from homeassistant.exceptions import HomeAssistantError
|
2021-10-23 19:01:34 +00:00
|
|
|
from homeassistant.helpers.entity import DeviceInfo, Entity
|
2020-05-05 04:03:12 +00:00
|
|
|
|
2021-10-25 07:37:37 +00:00
|
|
|
from . import _async_isy_to_configuration_url
|
2021-05-18 19:15:47 +00:00
|
|
|
from .const import DOMAIN
|
2020-05-10 00:48:51 +00:00
|
|
|
|
2020-05-05 04:03:12 +00:00
|
|
|
|
|
|
|
class ISYEntity(Entity):
|
|
|
|
"""Representation of an ISY994 device."""
|
|
|
|
|
2022-02-03 16:02:05 +00:00
|
|
|
_name: str | None = None
|
2022-05-03 16:49:52 +00:00
|
|
|
_attr_should_poll = False
|
2020-05-05 04:03:12 +00:00
|
|
|
|
2022-02-03 16:02:05 +00:00
|
|
|
def __init__(self, node: Node) -> None:
|
2020-05-05 04:03:12 +00:00
|
|
|
"""Initialize the insteon device."""
|
|
|
|
self._node = node
|
2022-02-03 16:02:05 +00:00
|
|
|
self._attrs: dict[str, Any] = {}
|
|
|
|
self._change_handler: EventListener | None = None
|
|
|
|
self._control_handler: EventListener | None = None
|
2020-05-05 04:03:12 +00:00
|
|
|
|
|
|
|
async def async_added_to_hass(self) -> None:
|
|
|
|
"""Subscribe to the node change events."""
|
2021-05-18 19:15:47 +00:00
|
|
|
self._change_handler = self._node.status_events.subscribe(self.async_on_update)
|
2020-05-05 04:03:12 +00:00
|
|
|
|
2020-05-08 04:15:42 +00:00
|
|
|
if hasattr(self._node, "control_events"):
|
2021-05-18 19:15:47 +00:00
|
|
|
self._control_handler = self._node.control_events.subscribe(
|
|
|
|
self.async_on_control
|
|
|
|
)
|
2020-05-05 04:03:12 +00:00
|
|
|
|
2021-05-18 19:15:47 +00:00
|
|
|
@callback
|
2022-02-03 16:02:05 +00:00
|
|
|
def async_on_update(self, event: NodeProperty) -> None:
|
2020-05-05 04:03:12 +00:00
|
|
|
"""Handle the update event from the ISY994 Node."""
|
2021-05-18 19:15:47 +00:00
|
|
|
self.async_write_ha_state()
|
2020-05-05 04:03:12 +00:00
|
|
|
|
2021-05-18 19:15:47 +00:00
|
|
|
@callback
|
|
|
|
def async_on_control(self, event: NodeProperty) -> None:
|
2020-05-05 04:03:12 +00:00
|
|
|
"""Handle a control event from the ISY994 Node."""
|
2020-05-08 04:15:42 +00:00
|
|
|
event_data = {
|
|
|
|
"entity_id": self.entity_id,
|
|
|
|
"control": event.control,
|
|
|
|
"value": event.value,
|
|
|
|
"formatted": event.formatted,
|
|
|
|
"uom": event.uom,
|
|
|
|
"precision": event.prec,
|
|
|
|
}
|
|
|
|
|
2020-05-12 02:32:19 +00:00
|
|
|
if event.control not in EVENT_PROPS_IGNORED:
|
2020-05-08 04:15:42 +00:00
|
|
|
# New state attributes may be available, update the state.
|
2021-05-18 19:15:47 +00:00
|
|
|
self.async_write_ha_state()
|
2020-05-08 04:15:42 +00:00
|
|
|
|
2022-09-05 06:45:35 +00:00
|
|
|
self.hass.bus.async_fire("isy994_control", event_data)
|
2020-05-05 04:03:12 +00:00
|
|
|
|
2020-05-10 00:48:51 +00:00
|
|
|
@property
|
2022-02-03 16:02:05 +00:00
|
|
|
def device_info(self) -> DeviceInfo | None:
|
2020-05-10 00:48:51 +00:00
|
|
|
"""Return the device_info of the device."""
|
|
|
|
if hasattr(self._node, "protocol") and self._node.protocol == PROTO_GROUP:
|
|
|
|
# not a device
|
|
|
|
return None
|
2021-10-25 07:37:37 +00:00
|
|
|
isy = self._node.isy
|
|
|
|
uuid = isy.configuration["uuid"]
|
2020-05-10 00:48:51 +00:00
|
|
|
node = self._node
|
2021-10-25 07:37:37 +00:00
|
|
|
url = _async_isy_to_configuration_url(isy)
|
|
|
|
|
2022-05-03 16:49:52 +00:00
|
|
|
basename = self._name or str(self._node.name)
|
2020-05-10 00:48:51 +00:00
|
|
|
|
|
|
|
if hasattr(self._node, "parent_node") and self._node.parent_node is not None:
|
|
|
|
# This is not the parent node, get the parent node.
|
|
|
|
node = self._node.parent_node
|
|
|
|
basename = node.name
|
|
|
|
|
2021-10-23 19:01:34 +00:00
|
|
|
device_info = DeviceInfo(
|
|
|
|
manufacturer="Unknown",
|
|
|
|
model="Unknown",
|
|
|
|
name=basename,
|
|
|
|
via_device=(DOMAIN, uuid),
|
2021-10-25 07:37:37 +00:00
|
|
|
configuration_url=url,
|
2021-10-23 19:01:34 +00:00
|
|
|
)
|
2020-05-10 00:48:51 +00:00
|
|
|
|
|
|
|
if hasattr(node, "address"):
|
2022-02-03 16:02:05 +00:00
|
|
|
assert isinstance(node.address, str)
|
|
|
|
device_info[ATTR_NAME] = f"{basename} ({node.address})"
|
2020-05-10 00:48:51 +00:00
|
|
|
if hasattr(node, "primary_node"):
|
2021-10-23 19:01:34 +00:00
|
|
|
device_info[ATTR_IDENTIFIERS] = {(DOMAIN, f"{uuid}_{node.address}")}
|
2020-05-10 00:48:51 +00:00
|
|
|
# ISYv5 Device Types
|
|
|
|
if hasattr(node, "node_def_id") and node.node_def_id is not None:
|
2022-02-03 16:02:05 +00:00
|
|
|
model: str = str(node.node_def_id)
|
2020-05-10 00:48:51 +00:00
|
|
|
# Numerical Device Type
|
|
|
|
if hasattr(node, "type") and node.type is not None:
|
2022-02-03 16:02:05 +00:00
|
|
|
model += f" {node.type}"
|
|
|
|
device_info[ATTR_MODEL] = model
|
2020-05-10 00:48:51 +00:00
|
|
|
if hasattr(node, "protocol"):
|
2022-02-03 16:02:05 +00:00
|
|
|
model = str(device_info[ATTR_MODEL])
|
|
|
|
manufacturer = str(node.protocol)
|
2020-05-10 00:48:51 +00:00
|
|
|
if node.protocol == PROTO_ZWAVE:
|
|
|
|
# Get extra information for Z-Wave Devices
|
2022-02-03 16:02:05 +00:00
|
|
|
manufacturer += f" MfrID:{node.zwave_props.mfr_id}"
|
|
|
|
model += (
|
2020-05-10 00:48:51 +00:00
|
|
|
f" Type:{node.zwave_props.devtype_gen} "
|
|
|
|
f"ProductTypeID:{node.zwave_props.prod_type_id} "
|
|
|
|
f"ProductID:{node.zwave_props.product_id}"
|
|
|
|
)
|
2022-02-03 16:02:05 +00:00
|
|
|
device_info[ATTR_MANUFACTURER] = manufacturer
|
|
|
|
device_info[ATTR_MODEL] = model
|
2021-02-23 03:27:33 +00:00
|
|
|
if hasattr(node, "folder") and node.folder is not None:
|
2021-10-23 19:01:34 +00:00
|
|
|
device_info[ATTR_SUGGESTED_AREA] = node.folder
|
2021-05-18 19:15:47 +00:00
|
|
|
# Note: sw_version is not exposed by the ISY for the individual devices.
|
2020-05-10 00:48:51 +00:00
|
|
|
|
|
|
|
return device_info
|
|
|
|
|
2020-05-05 04:03:12 +00:00
|
|
|
@property
|
2022-02-03 16:02:05 +00:00
|
|
|
def unique_id(self) -> str | None:
|
2020-05-05 04:03:12 +00:00
|
|
|
"""Get the unique identifier of the device."""
|
2020-05-09 19:49:00 +00:00
|
|
|
if hasattr(self._node, "address"):
|
|
|
|
return f"{self._node.isy.configuration['uuid']}_{self._node.address}"
|
|
|
|
return None
|
|
|
|
|
|
|
|
@property
|
2022-02-03 16:02:05 +00:00
|
|
|
def old_unique_id(self) -> str | None:
|
2020-05-09 19:49:00 +00:00
|
|
|
"""Get the old unique identifier of the device."""
|
2020-05-08 04:15:42 +00:00
|
|
|
if hasattr(self._node, "address"):
|
2022-02-03 16:02:05 +00:00
|
|
|
return cast(str, self._node.address)
|
2020-05-05 04:03:12 +00:00
|
|
|
return None
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self) -> str:
|
|
|
|
"""Get the name of the device."""
|
|
|
|
return self._name or str(self._node.name)
|
|
|
|
|
|
|
|
|
|
|
|
class ISYNodeEntity(ISYEntity):
|
|
|
|
"""Representation of a ISY Nodebase (Node/Group) entity."""
|
|
|
|
|
|
|
|
@property
|
2021-04-12 16:43:14 +00:00
|
|
|
def extra_state_attributes(self) -> dict:
|
2020-05-08 04:15:42 +00:00
|
|
|
"""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.
|
|
|
|
"""
|
2020-05-05 04:03:12 +00:00
|
|
|
attr = {}
|
2022-05-03 16:49:52 +00:00
|
|
|
node = self._node
|
|
|
|
# Insteon aux_properties are now their own sensors
|
|
|
|
if (
|
|
|
|
hasattr(self._node, "aux_properties")
|
|
|
|
and getattr(node, "protocol", None) != PROTO_INSTEON
|
|
|
|
):
|
|
|
|
for name, value in self._node.aux_properties.items():
|
2020-05-08 04:15:42 +00:00
|
|
|
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
|
2020-05-05 04:03:12 +00:00
|
|
|
|
2022-02-03 16:02:05 +00:00
|
|
|
async def async_send_node_command(self, command: str) -> None:
|
2020-05-11 15:58:58 +00:00
|
|
|
"""Respond to an entity service command call."""
|
|
|
|
if not hasattr(self._node, command):
|
2021-05-18 19:15:47 +00:00
|
|
|
raise HomeAssistantError(
|
|
|
|
f"Invalid service call: {command} for device {self.entity_id}"
|
2020-05-11 15:58:58 +00:00
|
|
|
)
|
2021-05-18 19:15:47 +00:00
|
|
|
await getattr(self._node, command)()
|
2020-05-11 15:58:58 +00:00
|
|
|
|
2021-05-18 19:15:47 +00:00
|
|
|
async def async_send_raw_node_command(
|
2022-02-03 16:02:05 +00:00
|
|
|
self,
|
|
|
|
command: str,
|
|
|
|
value: Any | None = None,
|
|
|
|
unit_of_measurement: str | None = None,
|
|
|
|
parameters: Any | None = None,
|
|
|
|
) -> None:
|
2020-05-11 15:58:58 +00:00
|
|
|
"""Respond to an entity service raw command call."""
|
|
|
|
if not hasattr(self._node, "send_cmd"):
|
2021-05-18 19:15:47 +00:00
|
|
|
raise HomeAssistantError(
|
|
|
|
f"Invalid service call: {command} for device {self.entity_id}"
|
2020-05-11 15:58:58 +00:00
|
|
|
)
|
2021-05-18 19:15:47 +00:00
|
|
|
await self._node.send_cmd(command, value, unit_of_measurement, parameters)
|
2020-05-11 15:58:58 +00:00
|
|
|
|
2022-02-03 16:02:05 +00:00
|
|
|
async def async_get_zwave_parameter(self, parameter: Any) -> None:
|
2021-06-11 11:35:03 +00:00
|
|
|
"""Respond to an entity service command to request a Z-Wave device parameter from the ISY."""
|
2021-05-20 00:08:35 +00:00
|
|
|
if not hasattr(self._node, "protocol") or self._node.protocol != PROTO_ZWAVE:
|
|
|
|
raise HomeAssistantError(
|
|
|
|
f"Invalid service call: cannot request Z-Wave Parameter for non-Z-Wave device {self.entity_id}"
|
|
|
|
)
|
|
|
|
await self._node.get_zwave_parameter(parameter)
|
|
|
|
|
2022-02-03 16:02:05 +00:00
|
|
|
async def async_set_zwave_parameter(
|
|
|
|
self, parameter: Any, value: Any | None, size: int | None
|
|
|
|
) -> None:
|
2021-06-11 11:35:03 +00:00
|
|
|
"""Respond to an entity service command to set a Z-Wave device parameter via the ISY."""
|
2021-05-20 00:08:35 +00:00
|
|
|
if not hasattr(self._node, "protocol") or self._node.protocol != PROTO_ZWAVE:
|
|
|
|
raise HomeAssistantError(
|
|
|
|
f"Invalid service call: cannot set Z-Wave Parameter for non-Z-Wave device {self.entity_id}"
|
|
|
|
)
|
|
|
|
await self._node.set_zwave_parameter(parameter, value, size)
|
|
|
|
await self._node.get_zwave_parameter(parameter)
|
|
|
|
|
2022-02-03 16:02:05 +00:00
|
|
|
async def async_rename_node(self, name: str) -> None:
|
2021-06-11 11:35:03 +00:00
|
|
|
"""Respond to an entity service command to rename a node on the ISY."""
|
2021-05-20 00:08:35 +00:00
|
|
|
await self._node.rename(name)
|
|
|
|
|
2020-05-05 04:03:12 +00:00
|
|
|
|
|
|
|
class ISYProgramEntity(ISYEntity):
|
|
|
|
"""Representation of an ISY994 program base."""
|
|
|
|
|
2022-02-03 16:02:05 +00:00
|
|
|
def __init__(self, name: str, status: Any | None, actions: Program = None) -> None:
|
2020-05-05 04:03:12 +00:00
|
|
|
"""Initialize the ISY994 program-based entity."""
|
|
|
|
super().__init__(status)
|
|
|
|
self._name = name
|
|
|
|
self._actions = actions
|
2020-05-08 04:15:42 +00:00
|
|
|
|
|
|
|
@property
|
2021-04-12 16:43:14 +00:00
|
|
|
def extra_state_attributes(self) -> dict:
|
2020-05-08 04:15:42 +00:00
|
|
|
"""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
|