2021-05-20 14:56:11 +00:00
|
|
|
"""Base implementation for all modbus platforms."""
|
|
|
|
from __future__ import annotations
|
|
|
|
|
|
|
|
from abc import abstractmethod
|
2021-09-29 14:19:06 +00:00
|
|
|
from collections.abc import Callable
|
2021-09-20 12:59:30 +00:00
|
|
|
from datetime import datetime, timedelta
|
2021-05-20 14:56:11 +00:00
|
|
|
import logging
|
2021-07-20 04:52:58 +00:00
|
|
|
import struct
|
2021-09-29 14:19:06 +00:00
|
|
|
from typing import Any, cast
|
2021-05-20 14:56:11 +00:00
|
|
|
|
|
|
|
from homeassistant.const import (
|
|
|
|
CONF_ADDRESS,
|
2021-05-24 18:13:25 +00:00
|
|
|
CONF_COMMAND_OFF,
|
|
|
|
CONF_COMMAND_ON,
|
2021-07-20 04:52:58 +00:00
|
|
|
CONF_COUNT,
|
2021-05-24 18:13:25 +00:00
|
|
|
CONF_DELAY,
|
2021-05-20 14:56:11 +00:00
|
|
|
CONF_DEVICE_CLASS,
|
|
|
|
CONF_NAME,
|
2021-07-20 04:52:58 +00:00
|
|
|
CONF_OFFSET,
|
2021-05-20 14:56:11 +00:00
|
|
|
CONF_SCAN_INTERVAL,
|
|
|
|
CONF_SLAVE,
|
2021-07-20 04:52:58 +00:00
|
|
|
CONF_STRUCTURE,
|
2022-01-26 09:37:15 +00:00
|
|
|
CONF_UNIQUE_ID,
|
2023-09-03 15:48:25 +00:00
|
|
|
STATE_OFF,
|
2021-05-24 18:13:25 +00:00
|
|
|
STATE_ON,
|
2021-05-20 14:56:11 +00:00
|
|
|
)
|
2021-09-14 07:42:50 +00:00
|
|
|
from homeassistant.core import callback
|
|
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_connect
|
2021-08-08 21:23:21 +00:00
|
|
|
from homeassistant.helpers.entity import Entity, ToggleEntity
|
2021-05-24 18:13:25 +00:00
|
|
|
from homeassistant.helpers.event import async_call_later, async_track_time_interval
|
|
|
|
from homeassistant.helpers.restore_state import RestoreEntity
|
2021-05-20 14:56:11 +00:00
|
|
|
|
2021-05-24 18:13:25 +00:00
|
|
|
from .const import (
|
|
|
|
CALL_TYPE_COIL,
|
2021-08-25 19:02:06 +00:00
|
|
|
CALL_TYPE_DISCRETE,
|
2021-07-12 18:22:53 +00:00
|
|
|
CALL_TYPE_REGISTER_HOLDING,
|
2021-08-25 19:02:06 +00:00
|
|
|
CALL_TYPE_REGISTER_INPUT,
|
2021-05-24 18:13:25 +00:00
|
|
|
CALL_TYPE_WRITE_COIL,
|
2021-07-12 18:22:53 +00:00
|
|
|
CALL_TYPE_WRITE_COILS,
|
2021-05-24 18:13:25 +00:00
|
|
|
CALL_TYPE_WRITE_REGISTER,
|
2021-07-12 18:22:53 +00:00
|
|
|
CALL_TYPE_WRITE_REGISTERS,
|
|
|
|
CALL_TYPE_X_COILS,
|
|
|
|
CALL_TYPE_X_REGISTER_HOLDINGS,
|
2021-07-20 04:52:58 +00:00
|
|
|
CONF_DATA_TYPE,
|
2023-09-15 11:49:33 +00:00
|
|
|
CONF_DEVICE_ADDRESS,
|
2021-05-24 18:13:25 +00:00
|
|
|
CONF_INPUT_TYPE,
|
2021-08-21 13:49:50 +00:00
|
|
|
CONF_LAZY_ERROR,
|
2023-02-02 21:29:03 +00:00
|
|
|
CONF_MAX_VALUE,
|
|
|
|
CONF_MIN_VALUE,
|
2023-08-06 11:47:54 +00:00
|
|
|
CONF_NAN_VALUE,
|
2021-07-20 04:52:58 +00:00
|
|
|
CONF_PRECISION,
|
|
|
|
CONF_SCALE,
|
2023-08-18 11:23:04 +00:00
|
|
|
CONF_SLAVE_COUNT,
|
2021-05-24 18:13:25 +00:00
|
|
|
CONF_STATE_OFF,
|
|
|
|
CONF_STATE_ON,
|
2021-07-20 04:52:58 +00:00
|
|
|
CONF_SWAP,
|
|
|
|
CONF_SWAP_BYTE,
|
2023-08-18 11:23:04 +00:00
|
|
|
CONF_SWAP_NONE,
|
2021-07-20 04:52:58 +00:00
|
|
|
CONF_SWAP_WORD,
|
|
|
|
CONF_SWAP_WORD_BYTE,
|
2021-05-24 18:13:25 +00:00
|
|
|
CONF_VERIFY,
|
2023-09-15 12:00:02 +00:00
|
|
|
CONF_VIRTUAL_COUNT,
|
2021-05-24 18:13:25 +00:00
|
|
|
CONF_WRITE_TYPE,
|
2023-02-02 21:29:03 +00:00
|
|
|
CONF_ZERO_SUPPRESS,
|
2021-09-14 07:42:50 +00:00
|
|
|
SIGNAL_START_ENTITY,
|
|
|
|
SIGNAL_STOP_ENTITY,
|
2021-10-15 05:09:59 +00:00
|
|
|
DataType,
|
2021-05-24 18:13:25 +00:00
|
|
|
)
|
2021-05-20 14:56:11 +00:00
|
|
|
from .modbus import ModbusHub
|
|
|
|
|
|
|
|
PARALLEL_UPDATES = 1
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
class BasePlatform(Entity):
|
|
|
|
"""Base for readonly platforms."""
|
|
|
|
|
|
|
|
def __init__(self, hub: ModbusHub, entry: dict[str, Any]) -> None:
|
|
|
|
"""Initialize the Modbus binary sensor."""
|
|
|
|
self._hub = hub
|
2023-09-15 11:49:33 +00:00
|
|
|
self._slave = entry.get(CONF_SLAVE, None) or entry.get(CONF_DEVICE_ADDRESS, 0)
|
2021-05-20 14:56:11 +00:00
|
|
|
self._address = int(entry[CONF_ADDRESS])
|
|
|
|
self._input_type = entry[CONF_INPUT_TYPE]
|
2021-09-20 12:59:30 +00:00
|
|
|
self._value: str | None = None
|
2021-05-24 10:59:55 +00:00
|
|
|
self._scan_interval = int(entry[CONF_SCAN_INTERVAL])
|
2021-07-13 19:45:42 +00:00
|
|
|
self._call_active = False
|
2021-09-14 07:42:50 +00:00
|
|
|
self._cancel_timer: Callable[[], None] | None = None
|
2021-09-18 06:57:27 +00:00
|
|
|
self._cancel_call: Callable[[], None] | None = None
|
2021-05-20 14:56:11 +00:00
|
|
|
|
2022-01-26 09:37:15 +00:00
|
|
|
self._attr_unique_id = entry.get(CONF_UNIQUE_ID)
|
2021-07-26 19:20:34 +00:00
|
|
|
self._attr_name = entry[CONF_NAME]
|
|
|
|
self._attr_should_poll = False
|
|
|
|
self._attr_device_class = entry.get(CONF_DEVICE_CLASS)
|
|
|
|
self._attr_available = True
|
|
|
|
self._attr_unit_of_measurement = None
|
2021-08-21 13:49:50 +00:00
|
|
|
self._lazy_error_count = entry[CONF_LAZY_ERROR]
|
|
|
|
self._lazy_errors = self._lazy_error_count
|
2021-07-26 19:20:34 +00:00
|
|
|
|
2023-02-02 21:29:03 +00:00
|
|
|
def get_optional_numeric_config(config_name: str) -> int | float | None:
|
|
|
|
if (val := entry.get(config_name)) is None:
|
|
|
|
return None
|
|
|
|
assert isinstance(
|
|
|
|
val, (float, int)
|
|
|
|
), f"Expected float or int but {config_name} was {type(val)}"
|
|
|
|
return val
|
|
|
|
|
|
|
|
self._min_value = get_optional_numeric_config(CONF_MIN_VALUE)
|
|
|
|
self._max_value = get_optional_numeric_config(CONF_MAX_VALUE)
|
2023-08-06 11:47:54 +00:00
|
|
|
self._nan_value = entry.get(CONF_NAN_VALUE, None)
|
2023-02-02 21:29:03 +00:00
|
|
|
self._zero_suppress = get_optional_numeric_config(CONF_ZERO_SUPPRESS)
|
|
|
|
|
2021-05-20 14:56:11 +00:00
|
|
|
@abstractmethod
|
2021-09-20 12:59:30 +00:00
|
|
|
async def async_update(self, now: datetime | None = None) -> None:
|
2021-05-20 14:56:11 +00:00
|
|
|
"""Virtual function to be overwritten."""
|
|
|
|
|
2021-09-14 07:42:50 +00:00
|
|
|
@callback
|
2021-09-18 06:57:27 +00:00
|
|
|
def async_run(self) -> None:
|
2021-09-14 07:42:50 +00:00
|
|
|
"""Remote start entity."""
|
2021-09-18 06:57:27 +00:00
|
|
|
self.async_hold(update=False)
|
2023-09-12 14:01:15 +00:00
|
|
|
self._cancel_call = async_call_later(
|
|
|
|
self.hass, timedelta(milliseconds=100), self.async_update
|
|
|
|
)
|
2021-05-24 10:59:55 +00:00
|
|
|
if self._scan_interval > 0:
|
2021-09-14 07:42:50 +00:00
|
|
|
self._cancel_timer = async_track_time_interval(
|
2021-05-24 10:59:55 +00:00
|
|
|
self.hass, self.async_update, timedelta(seconds=self._scan_interval)
|
|
|
|
)
|
2021-09-14 07:42:50 +00:00
|
|
|
self._attr_available = True
|
|
|
|
self.async_write_ha_state()
|
|
|
|
|
|
|
|
@callback
|
2021-09-20 12:59:30 +00:00
|
|
|
def async_hold(self, update: bool = True) -> None:
|
2021-09-14 07:42:50 +00:00
|
|
|
"""Remote stop entity."""
|
2021-09-18 06:57:27 +00:00
|
|
|
if self._cancel_call:
|
|
|
|
self._cancel_call()
|
|
|
|
self._cancel_call = None
|
2021-09-14 07:42:50 +00:00
|
|
|
if self._cancel_timer:
|
|
|
|
self._cancel_timer()
|
|
|
|
self._cancel_timer = None
|
2021-09-18 06:57:27 +00:00
|
|
|
if update:
|
|
|
|
self._attr_available = False
|
|
|
|
self.async_write_ha_state()
|
2021-09-14 07:42:50 +00:00
|
|
|
|
2021-09-20 12:59:30 +00:00
|
|
|
async def async_base_added_to_hass(self) -> None:
|
2021-09-14 07:42:50 +00:00
|
|
|
"""Handle entity which will be added."""
|
2021-09-18 06:57:27 +00:00
|
|
|
self.async_run()
|
|
|
|
self.async_on_remove(
|
|
|
|
async_dispatcher_connect(self.hass, SIGNAL_STOP_ENTITY, self.async_hold)
|
|
|
|
)
|
|
|
|
self.async_on_remove(
|
|
|
|
async_dispatcher_connect(self.hass, SIGNAL_START_ENTITY, self.async_run)
|
2021-09-14 07:42:50 +00:00
|
|
|
)
|
2021-05-20 14:56:11 +00:00
|
|
|
|
2021-05-24 18:13:25 +00:00
|
|
|
|
2021-07-20 04:52:58 +00:00
|
|
|
class BaseStructPlatform(BasePlatform, RestoreEntity):
|
|
|
|
"""Base class representing a sensor/climate."""
|
|
|
|
|
|
|
|
def __init__(self, hub: ModbusHub, config: dict) -> None:
|
|
|
|
"""Initialize the switch."""
|
|
|
|
super().__init__(hub, config)
|
|
|
|
self._swap = config[CONF_SWAP]
|
2023-08-18 11:23:04 +00:00
|
|
|
if self._swap == CONF_SWAP_NONE:
|
|
|
|
self._swap = None
|
2021-07-20 04:52:58 +00:00
|
|
|
self._data_type = config[CONF_DATA_TYPE]
|
2021-09-20 12:59:30 +00:00
|
|
|
self._structure: str = config[CONF_STRUCTURE]
|
2021-07-20 04:52:58 +00:00
|
|
|
self._precision = config[CONF_PRECISION]
|
|
|
|
self._scale = config[CONF_SCALE]
|
2023-09-07 18:56:00 +00:00
|
|
|
if self._scale < 1 and not self._precision:
|
|
|
|
self._precision = 2
|
2021-07-20 04:52:58 +00:00
|
|
|
self._offset = config[CONF_OFFSET]
|
2023-09-15 12:00:02 +00:00
|
|
|
self._slave_count = config.get(CONF_SLAVE_COUNT, None) or config.get(
|
|
|
|
CONF_VIRTUAL_COUNT, 0
|
|
|
|
)
|
2023-08-18 11:23:04 +00:00
|
|
|
self._slave_size = self._count = config[CONF_COUNT]
|
2021-07-20 04:52:58 +00:00
|
|
|
|
2023-08-18 11:23:04 +00:00
|
|
|
def _swap_registers(self, registers: list[int], slave_count: int) -> list[int]:
|
2021-07-20 04:52:58 +00:00
|
|
|
"""Do swap as needed."""
|
2023-08-18 11:23:04 +00:00
|
|
|
if slave_count:
|
|
|
|
swapped = []
|
|
|
|
for i in range(0, self._slave_count + 1):
|
|
|
|
inx = i * self._slave_size
|
|
|
|
inx2 = inx + self._slave_size
|
|
|
|
swapped.extend(self._swap_registers(registers[inx:inx2], 0))
|
|
|
|
return swapped
|
2021-07-29 23:20:03 +00:00
|
|
|
if self._swap in (CONF_SWAP_BYTE, CONF_SWAP_WORD_BYTE):
|
2021-07-20 04:52:58 +00:00
|
|
|
# convert [12][34] --> [21][43]
|
|
|
|
for i, register in enumerate(registers):
|
|
|
|
registers[i] = int.from_bytes(
|
|
|
|
register.to_bytes(2, byteorder="little"),
|
|
|
|
byteorder="big",
|
|
|
|
signed=False,
|
|
|
|
)
|
2021-07-29 23:20:03 +00:00
|
|
|
if self._swap in (CONF_SWAP_WORD, CONF_SWAP_WORD_BYTE):
|
2021-07-20 04:52:58 +00:00
|
|
|
# convert [12][34] ==> [34][12]
|
|
|
|
registers.reverse()
|
|
|
|
return registers
|
|
|
|
|
2023-09-12 14:05:59 +00:00
|
|
|
def __process_raw_value(
|
|
|
|
self, entry: float | int | str | bytes
|
|
|
|
) -> float | int | str | bytes | None:
|
2023-08-06 11:47:54 +00:00
|
|
|
"""Process value from sensor with NaN handling, scaling, offset, min/max etc."""
|
|
|
|
if self._nan_value and entry in (self._nan_value, -self._nan_value):
|
2023-08-28 07:07:31 +00:00
|
|
|
return None
|
2023-09-12 14:05:59 +00:00
|
|
|
if isinstance(entry, bytes):
|
|
|
|
return entry
|
2023-02-02 21:29:03 +00:00
|
|
|
val: float | int = self._scale * entry + self._offset
|
|
|
|
if self._min_value is not None and val < self._min_value:
|
|
|
|
return self._min_value
|
|
|
|
if self._max_value is not None and val > self._max_value:
|
|
|
|
return self._max_value
|
|
|
|
if self._zero_suppress is not None and abs(val) <= self._zero_suppress:
|
|
|
|
return 0
|
|
|
|
return val
|
|
|
|
|
2021-10-20 22:22:01 +00:00
|
|
|
def unpack_structure_result(self, registers: list[int]) -> str | None:
|
2021-07-20 04:52:58 +00:00
|
|
|
"""Convert registers to proper result."""
|
|
|
|
|
2023-08-18 11:23:04 +00:00
|
|
|
if self._swap:
|
|
|
|
registers = self._swap_registers(registers, self._slave_count)
|
2021-07-20 04:52:58 +00:00
|
|
|
byte_string = b"".join([x.to_bytes(2, byteorder="big") for x in registers])
|
2021-10-15 05:09:59 +00:00
|
|
|
if self._data_type == DataType.STRING:
|
2021-08-08 21:23:21 +00:00
|
|
|
return byte_string.decode()
|
2021-07-20 04:52:58 +00:00
|
|
|
|
2021-10-20 22:22:01 +00:00
|
|
|
try:
|
|
|
|
val = struct.unpack(self._structure, byte_string)
|
|
|
|
except struct.error as err:
|
|
|
|
recv_size = len(registers) * 2
|
|
|
|
msg = f"Received {recv_size} bytes, unpack error {err}"
|
|
|
|
_LOGGER.error(msg)
|
|
|
|
return None
|
2021-08-08 21:23:21 +00:00
|
|
|
# Issue: https://github.com/home-assistant/core/issues/41944
|
|
|
|
# If unpack() returns a tuple greater than 1, don't try to process the value.
|
|
|
|
# Instead, return the values of unpack(...) separated by commas.
|
|
|
|
if len(val) > 1:
|
2023-02-02 21:29:03 +00:00
|
|
|
# Apply scale, precision, limits to floats and ints
|
2021-08-08 21:23:21 +00:00
|
|
|
v_result = []
|
|
|
|
for entry in val:
|
2023-02-02 21:29:03 +00:00
|
|
|
v_temp = self.__process_raw_value(entry)
|
2021-07-20 04:52:58 +00:00
|
|
|
|
|
|
|
# We could convert int to float, and the code would still work; however
|
|
|
|
# we lose some precision, and unit tests will fail. Therefore, we do
|
|
|
|
# the conversion only when it's absolutely necessary.
|
2021-08-08 21:23:21 +00:00
|
|
|
if isinstance(v_temp, int) and self._precision == 0:
|
|
|
|
v_result.append(str(v_temp))
|
2023-08-28 07:07:31 +00:00
|
|
|
elif v_temp is None:
|
2023-09-12 14:05:59 +00:00
|
|
|
v_result.append("0")
|
2023-08-06 12:26:56 +00:00
|
|
|
elif v_temp != v_temp: # noqa: PLR0124
|
|
|
|
# NaN float detection replace with None
|
2023-09-12 14:05:59 +00:00
|
|
|
v_result.append("0")
|
2021-07-20 04:52:58 +00:00
|
|
|
else:
|
2021-08-08 21:23:21 +00:00
|
|
|
v_result.append(f"{float(v_temp):.{self._precision}f}")
|
|
|
|
return ",".join(map(str, v_result))
|
|
|
|
|
2023-09-12 14:05:59 +00:00
|
|
|
# NaN float detection replace with None
|
|
|
|
if val[0] != val[0]: # noqa: PLR0124
|
|
|
|
return None
|
|
|
|
if byte_string == b"nan\x00":
|
|
|
|
return None
|
|
|
|
|
2023-02-02 21:29:03 +00:00
|
|
|
# Apply scale, precision, limits to floats and ints
|
|
|
|
val_result = self.__process_raw_value(val[0])
|
2021-07-20 04:52:58 +00:00
|
|
|
|
2021-08-08 21:23:21 +00:00
|
|
|
# We could convert int to float, and the code would still work; however
|
|
|
|
# we lose some precision, and unit tests will fail. Therefore, we do
|
|
|
|
# the conversion only when it's absolutely necessary.
|
2023-08-06 12:26:56 +00:00
|
|
|
|
2023-08-28 07:07:31 +00:00
|
|
|
if val_result is None:
|
|
|
|
return None
|
2021-09-20 12:59:30 +00:00
|
|
|
if isinstance(val_result, int) and self._precision == 0:
|
|
|
|
return str(val_result)
|
2023-09-12 14:05:59 +00:00
|
|
|
if isinstance(val_result, bytes):
|
|
|
|
return val_result.decode()
|
2021-09-20 12:59:30 +00:00
|
|
|
return f"{float(val_result):.{self._precision}f}"
|
2021-07-20 04:52:58 +00:00
|
|
|
|
2021-08-08 21:23:21 +00:00
|
|
|
|
|
|
|
class BaseSwitch(BasePlatform, ToggleEntity, RestoreEntity):
|
2021-05-24 18:13:25 +00:00
|
|
|
"""Base class representing a Modbus switch."""
|
|
|
|
|
|
|
|
def __init__(self, hub: ModbusHub, config: dict) -> None:
|
|
|
|
"""Initialize the switch."""
|
|
|
|
config[CONF_INPUT_TYPE] = ""
|
|
|
|
super().__init__(hub, config)
|
2021-08-08 21:23:21 +00:00
|
|
|
self._attr_is_on = False
|
2021-07-12 18:22:53 +00:00
|
|
|
convert = {
|
|
|
|
CALL_TYPE_REGISTER_HOLDING: (
|
|
|
|
CALL_TYPE_REGISTER_HOLDING,
|
|
|
|
CALL_TYPE_WRITE_REGISTER,
|
|
|
|
),
|
2021-08-25 19:02:06 +00:00
|
|
|
CALL_TYPE_DISCRETE: (
|
|
|
|
CALL_TYPE_DISCRETE,
|
|
|
|
None,
|
|
|
|
),
|
|
|
|
CALL_TYPE_REGISTER_INPUT: (
|
|
|
|
CALL_TYPE_REGISTER_INPUT,
|
|
|
|
None,
|
|
|
|
),
|
2021-07-12 18:22:53 +00:00
|
|
|
CALL_TYPE_COIL: (CALL_TYPE_COIL, CALL_TYPE_WRITE_COIL),
|
|
|
|
CALL_TYPE_X_COILS: (CALL_TYPE_COIL, CALL_TYPE_WRITE_COILS),
|
|
|
|
CALL_TYPE_X_REGISTER_HOLDINGS: (
|
|
|
|
CALL_TYPE_REGISTER_HOLDING,
|
|
|
|
CALL_TYPE_WRITE_REGISTERS,
|
|
|
|
),
|
|
|
|
}
|
2021-09-20 12:59:30 +00:00
|
|
|
self._write_type = cast(str, convert[config[CONF_WRITE_TYPE]][1])
|
2021-05-24 18:13:25 +00:00
|
|
|
self.command_on = config[CONF_COMMAND_ON]
|
|
|
|
self._command_off = config[CONF_COMMAND_OFF]
|
|
|
|
if CONF_VERIFY in config:
|
|
|
|
if config[CONF_VERIFY] is None:
|
|
|
|
config[CONF_VERIFY] = {}
|
|
|
|
self._verify_active = True
|
|
|
|
self._verify_delay = config[CONF_VERIFY].get(CONF_DELAY, 0)
|
|
|
|
self._verify_address = config[CONF_VERIFY].get(
|
|
|
|
CONF_ADDRESS, config[CONF_ADDRESS]
|
|
|
|
)
|
2021-08-16 02:57:37 +00:00
|
|
|
self._verify_type = convert[
|
|
|
|
config[CONF_VERIFY].get(CONF_INPUT_TYPE, config[CONF_WRITE_TYPE])
|
|
|
|
][0]
|
2021-05-24 18:13:25 +00:00
|
|
|
self._state_on = config[CONF_VERIFY].get(CONF_STATE_ON, self.command_on)
|
|
|
|
self._state_off = config[CONF_VERIFY].get(CONF_STATE_OFF, self._command_off)
|
|
|
|
else:
|
|
|
|
self._verify_active = False
|
|
|
|
|
2021-09-20 12:59:30 +00:00
|
|
|
async def async_added_to_hass(self) -> None:
|
2021-05-24 18:13:25 +00:00
|
|
|
"""Handle entity which will be added."""
|
|
|
|
await self.async_base_added_to_hass()
|
2021-10-30 14:30:13 +00:00
|
|
|
if state := await self.async_get_last_state():
|
2023-09-03 15:48:25 +00:00
|
|
|
if state.state == STATE_ON:
|
|
|
|
self._attr_is_on = True
|
|
|
|
elif state.state == STATE_OFF:
|
|
|
|
self._attr_is_on = False
|
2021-05-24 18:13:25 +00:00
|
|
|
|
2021-09-20 12:59:30 +00:00
|
|
|
async def async_turn(self, command: int) -> None:
|
2021-05-24 18:13:25 +00:00
|
|
|
"""Evaluate switch result."""
|
2023-08-04 18:14:32 +00:00
|
|
|
result = await self._hub.async_pb_call(
|
2021-05-24 18:13:25 +00:00
|
|
|
self._slave, self._address, command, self._write_type
|
|
|
|
)
|
|
|
|
if result is None:
|
2021-07-26 19:20:34 +00:00
|
|
|
self._attr_available = False
|
2021-05-24 18:13:25 +00:00
|
|
|
self.async_write_ha_state()
|
|
|
|
return
|
|
|
|
|
2021-07-26 19:20:34 +00:00
|
|
|
self._attr_available = True
|
2021-05-24 18:13:25 +00:00
|
|
|
if not self._verify_active:
|
2021-08-08 21:23:21 +00:00
|
|
|
self._attr_is_on = command == self.command_on
|
2021-05-24 18:13:25 +00:00
|
|
|
self.async_write_ha_state()
|
|
|
|
return
|
|
|
|
|
|
|
|
if self._verify_delay:
|
|
|
|
async_call_later(self.hass, self._verify_delay, self.async_update)
|
|
|
|
else:
|
|
|
|
await self.async_update()
|
|
|
|
|
2021-09-20 12:59:30 +00:00
|
|
|
async def async_turn_off(self, **kwargs: Any) -> None:
|
2021-05-24 18:13:25 +00:00
|
|
|
"""Set switch off."""
|
|
|
|
await self.async_turn(self._command_off)
|
|
|
|
|
2021-09-20 12:59:30 +00:00
|
|
|
async def async_update(self, now: datetime | None = None) -> None:
|
2021-05-24 18:13:25 +00:00
|
|
|
"""Update the entity state."""
|
|
|
|
# remark "now" is a dummy parameter to avoid problems with
|
|
|
|
# async_track_time_interval
|
|
|
|
if not self._verify_active:
|
2021-07-26 19:20:34 +00:00
|
|
|
self._attr_available = True
|
2021-05-24 18:13:25 +00:00
|
|
|
self.async_write_ha_state()
|
|
|
|
return
|
|
|
|
|
2021-07-13 19:45:42 +00:00
|
|
|
# do not allow multiple active calls to the same platform
|
|
|
|
if self._call_active:
|
|
|
|
return
|
|
|
|
self._call_active = True
|
2023-08-04 18:14:32 +00:00
|
|
|
result = await self._hub.async_pb_call(
|
2021-05-24 18:13:25 +00:00
|
|
|
self._slave, self._verify_address, 1, self._verify_type
|
|
|
|
)
|
2021-07-13 19:45:42 +00:00
|
|
|
self._call_active = False
|
2021-05-24 18:13:25 +00:00
|
|
|
if result is None:
|
2021-08-21 13:49:50 +00:00
|
|
|
if self._lazy_errors:
|
|
|
|
self._lazy_errors -= 1
|
|
|
|
return
|
|
|
|
self._lazy_errors = self._lazy_error_count
|
2021-07-26 19:20:34 +00:00
|
|
|
self._attr_available = False
|
2021-05-24 18:13:25 +00:00
|
|
|
self.async_write_ha_state()
|
|
|
|
return
|
|
|
|
|
2021-08-21 13:49:50 +00:00
|
|
|
self._lazy_errors = self._lazy_error_count
|
2021-07-26 19:20:34 +00:00
|
|
|
self._attr_available = True
|
2022-02-21 18:15:03 +00:00
|
|
|
if self._verify_type in (CALL_TYPE_COIL, CALL_TYPE_DISCRETE):
|
2021-08-08 21:23:21 +00:00
|
|
|
self._attr_is_on = bool(result.bits[0] & 1)
|
2021-05-24 18:13:25 +00:00
|
|
|
else:
|
|
|
|
value = int(result.registers[0])
|
|
|
|
if value == self._state_on:
|
2021-08-08 21:23:21 +00:00
|
|
|
self._attr_is_on = True
|
2021-05-24 18:13:25 +00:00
|
|
|
elif value == self._state_off:
|
2021-08-08 21:23:21 +00:00
|
|
|
self._attr_is_on = False
|
2021-05-24 18:13:25 +00:00
|
|
|
elif value is not None:
|
|
|
|
_LOGGER.error(
|
2022-12-22 12:35:47 +00:00
|
|
|
(
|
|
|
|
"Unexpected response from modbus device slave %s register %s,"
|
|
|
|
" got 0x%2x"
|
|
|
|
),
|
2021-05-24 18:13:25 +00:00
|
|
|
self._slave,
|
|
|
|
self._verify_address,
|
|
|
|
value,
|
|
|
|
)
|
|
|
|
self.async_write_ha_state()
|