"""Generic Z-Wave Entity Class.""" from __future__ import annotations from collections.abc import Sequence from typing import Any from zwave_js_server.const import NodeStatus from zwave_js_server.exceptions import BaseZwaveJSServerError from zwave_js_server.model.driver import Driver from zwave_js_server.model.value import Value as ZwaveValue, get_value_id_str from homeassistant.config_entries import ConfigEntry from homeassistant.core import callback from homeassistant.exceptions import HomeAssistantError from homeassistant.helpers.dispatcher import async_dispatcher_connect from homeassistant.helpers.entity import DeviceInfo, Entity from homeassistant.helpers.typing import UNDEFINED from .const import DOMAIN, LOGGER from .discovery import ZwaveDiscoveryInfo from .helpers import get_device_id, get_unique_id, get_valueless_base_unique_id EVENT_VALUE_UPDATED = "value updated" EVENT_VALUE_REMOVED = "value removed" EVENT_DEAD = "dead" EVENT_ALIVE = "alive" class ZWaveBaseEntity(Entity): """Generic Entity Class for a Z-Wave Device.""" _attr_should_poll = False _attr_has_entity_name = True def __init__( self, config_entry: ConfigEntry, driver: Driver, info: ZwaveDiscoveryInfo ) -> None: """Initialize a generic Z-Wave device entity.""" self.config_entry = config_entry self.driver = driver self.info = info # entities requiring additional values, can add extra ids to this list self.watched_value_ids = {self.info.primary_value.value_id} if self.info.additional_value_ids_to_watch: self.watched_value_ids = self.watched_value_ids.union( self.info.additional_value_ids_to_watch ) # Entity class attributes self._attr_name = self.generate_name() self._attr_unique_id = get_unique_id(driver, self.info.primary_value.value_id) if self.info.entity_registry_enabled_default is False: self._attr_entity_registry_enabled_default = False if self.info.entity_category is not None: self._attr_entity_category = self.info.entity_category if self.info.assumed_state: self._attr_assumed_state = True # device is precreated in main handler self._attr_device_info = DeviceInfo( identifiers={get_device_id(driver, self.info.node)}, ) @callback def on_value_update(self) -> None: """Call when one of the watched values change. To be overridden by platforms needing this event. """ async def _async_poll_value(self, value_or_id: str | ZwaveValue) -> None: """Poll a value.""" # We log an error instead of raising an exception because this service call occurs # in a separate task and we don't want to raise the exception in that separate task # because it is confusing to the user. try: await self.info.node.async_poll_value(value_or_id) except BaseZwaveJSServerError as err: LOGGER.error("Error while refreshing value %s: %s", value_or_id, err) async def async_poll_value(self, refresh_all_values: bool) -> None: """Poll a value.""" if not refresh_all_values: await self._async_poll_value(self.info.primary_value) LOGGER.info( ( "Refreshing primary value %s for %s, " "state update may be delayed for devices on battery" ), self.info.primary_value, self.entity_id, ) return for value_id in self.watched_value_ids: await self._async_poll_value(value_id) LOGGER.info( ( "Refreshing values %s for %s, state update may be delayed for " "devices on battery" ), ", ".join(self.watched_value_ids), self.entity_id, ) async def async_added_to_hass(self) -> None: """Call when entity is added.""" # Add value_changed callbacks. self.async_on_remove( self.info.node.on(EVENT_VALUE_UPDATED, self._value_changed) ) self.async_on_remove( self.info.node.on(EVENT_VALUE_REMOVED, self._value_removed) ) self.async_on_remove( async_dispatcher_connect( self.hass, ( f"{DOMAIN}_" f"{get_valueless_base_unique_id(self.driver, self.info.node)}_" "remove_entity" ), self.async_remove, ) ) self.async_on_remove( async_dispatcher_connect( self.hass, ( f"{DOMAIN}_" f"{get_valueless_base_unique_id(self.driver, self.info.node)}_" "remove_entity_on_interview_started" ), self.async_remove, ) ) for status_event in (EVENT_ALIVE, EVENT_DEAD): self.async_on_remove( self.info.node.on(status_event, self._node_status_alive_or_dead) ) self.async_on_remove( async_dispatcher_connect( self.hass, f"{DOMAIN}_{self.unique_id}_poll_value", self.async_poll_value, ) ) def generate_name( self, include_value_name: bool = False, alternate_value_name: str | None = None, additional_info: Sequence[str | None] | None = None, name_prefix: str | None = None, ) -> str: """Generate entity name.""" name = "" if ( hasattr(self, "entity_description") and self.entity_description and self.entity_description.name and self.entity_description.name is not UNDEFINED ): name = self.entity_description.name if name_prefix: name = f"{name_prefix} {name}".strip() value_name = "" if alternate_value_name: value_name = alternate_value_name elif include_value_name: value_name = ( self.info.primary_value.metadata.label or self.info.primary_value.property_key_name or self.info.primary_value.property_name or "" ) name = f"{name} {value_name}".strip() # Only include non empty additional info if additional_info := [item for item in (additional_info or []) if item]: name = f"{name} {' '.join(additional_info)}" # append endpoint if > 1 if ( self.info.primary_value.endpoint is not None and self.info.primary_value.endpoint > 1 ): name += f" ({self.info.primary_value.endpoint})" return name @property def available(self) -> bool: """Return entity availability.""" return ( self.driver.client.connected and bool(self.info.node.ready) and self.info.node.status != NodeStatus.DEAD ) @callback def _node_status_alive_or_dead(self, event_data: dict) -> None: """Call when node status changes to alive or dead. Should not be overridden by subclasses. """ self.async_write_ha_state() @callback def _value_changed(self, event_data: dict) -> None: """Call when a value associated with our node changes. Should not be overridden by subclasses. """ value_id = event_data["value"].value_id if value_id not in self.watched_value_ids: return value = self.info.node.values[value_id] LOGGER.debug( "[%s] Value %s/%s changed to: %s", self.entity_id, value.property_, value.property_key_name, value.value, ) self.on_value_update() self.async_write_ha_state() @callback def _value_removed(self, event_data: dict) -> None: """Call when a value associated with our node is removed. Should not be overridden by subclasses. """ value_id = event_data["value"].value_id if value_id != self.info.primary_value.value_id: return LOGGER.debug( "[%s] Primary value %s is being removed", self.entity_id, value_id, ) self.hass.async_create_task(self.async_remove()) @callback def get_zwave_value( self, value_property: str | int, command_class: int | None = None, endpoint: int | None = None, value_property_key: int | str | None = None, add_to_watched_value_ids: bool = True, check_all_endpoints: bool = False, ) -> ZwaveValue | None: """Return specific ZwaveValue on this ZwaveNode.""" # use commandclass and endpoint from primary value if omitted return_value = None if command_class is None: command_class = self.info.primary_value.command_class if endpoint is None: endpoint = self.info.primary_value.endpoint # lookup value by value_id value_id = get_value_id_str( self.info.node, command_class, value_property, endpoint=endpoint, property_key=value_property_key, ) return_value = self.info.node.values.get(value_id) # If we haven't found a value and check_all_endpoints is True, we should # return the first value we can find on any other endpoint if return_value is None and check_all_endpoints: for endpoint_idx in self.info.node.endpoints: if endpoint_idx != self.info.primary_value.endpoint: value_id = get_value_id_str( self.info.node, command_class, value_property, endpoint=endpoint_idx, property_key=value_property_key, ) return_value = self.info.node.values.get(value_id) if return_value: break # add to watched_ids list so we will be triggered when the value updates if ( return_value and return_value.value_id not in self.watched_value_ids and add_to_watched_value_ids ): self.watched_value_ids.add(return_value.value_id) return return_value async def _async_set_value( self, value: ZwaveValue, new_value: Any, options: dict | None = None, wait_for_result: bool | None = None, ) -> bool | None: """Set value on node.""" try: return await self.info.node.async_set_value( value, new_value, options=options, wait_for_result=wait_for_result ) except BaseZwaveJSServerError as err: LOGGER.error("Unable to set value %s: %s", value.value_id, err) raise HomeAssistantError from err