Add base class for all modbus platforms (#50878)
* Add base for all platforms. * Please pylint.pull/50090/head
parent
f212049fc2
commit
c650deef98
|
@ -0,0 +1,67 @@
|
||||||
|
"""Base implementation for all modbus platforms."""
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from abc import abstractmethod
|
||||||
|
from datetime import timedelta
|
||||||
|
import logging
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
|
from homeassistant.const import (
|
||||||
|
CONF_ADDRESS,
|
||||||
|
CONF_DEVICE_CLASS,
|
||||||
|
CONF_NAME,
|
||||||
|
CONF_SCAN_INTERVAL,
|
||||||
|
CONF_SLAVE,
|
||||||
|
)
|
||||||
|
from homeassistant.helpers.entity import Entity
|
||||||
|
from homeassistant.helpers.event import async_track_time_interval
|
||||||
|
|
||||||
|
from .const import CONF_INPUT_TYPE
|
||||||
|
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
|
||||||
|
self._name = entry[CONF_NAME]
|
||||||
|
self._slave = entry.get(CONF_SLAVE)
|
||||||
|
self._address = int(entry[CONF_ADDRESS])
|
||||||
|
self._device_class = entry.get(CONF_DEVICE_CLASS)
|
||||||
|
self._input_type = entry[CONF_INPUT_TYPE]
|
||||||
|
self._value = None
|
||||||
|
self._available = True
|
||||||
|
self._scan_interval = timedelta(seconds=entry[CONF_SCAN_INTERVAL])
|
||||||
|
|
||||||
|
@abstractmethod
|
||||||
|
async def async_update(self, now=None):
|
||||||
|
"""Virtual function to be overwritten."""
|
||||||
|
|
||||||
|
async def async_base_added_to_hass(self):
|
||||||
|
"""Handle entity which will be added."""
|
||||||
|
async_track_time_interval(self.hass, self.async_update, self._scan_interval)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def name(self):
|
||||||
|
"""Return the name of the sensor."""
|
||||||
|
return self._name
|
||||||
|
|
||||||
|
@property
|
||||||
|
def should_poll(self):
|
||||||
|
"""Return True if entity has to be polled for state."""
|
||||||
|
return False
|
||||||
|
|
||||||
|
@property
|
||||||
|
def device_class(self) -> str | None:
|
||||||
|
"""Return the device class of the sensor."""
|
||||||
|
return self._device_class
|
||||||
|
|
||||||
|
@property
|
||||||
|
def available(self) -> bool:
|
||||||
|
"""Return True if entity is available."""
|
||||||
|
return self._available
|
|
@ -1,7 +1,6 @@
|
||||||
"""Support for Modbus Coil and Discrete Input sensors."""
|
"""Support for Modbus Coil and Discrete Input sensors."""
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
from datetime import timedelta
|
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
import voluptuous as vol
|
import voluptuous as vol
|
||||||
|
@ -21,9 +20,9 @@ from homeassistant.const import (
|
||||||
)
|
)
|
||||||
from homeassistant.core import HomeAssistant
|
from homeassistant.core import HomeAssistant
|
||||||
from homeassistant.helpers import config_validation as cv
|
from homeassistant.helpers import config_validation as cv
|
||||||
from homeassistant.helpers.event import async_track_time_interval
|
|
||||||
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
||||||
|
|
||||||
|
from .base_platform import BasePlatform
|
||||||
from .const import (
|
from .const import (
|
||||||
CALL_TYPE_COIL,
|
CALL_TYPE_COIL,
|
||||||
CALL_TYPE_DISCRETE,
|
CALL_TYPE_DISCRETE,
|
||||||
|
@ -92,65 +91,25 @@ async def async_setup_platform(
|
||||||
hub = hass.data[MODBUS_DOMAIN][discovery_info[CONF_NAME]]
|
hub = hass.data[MODBUS_DOMAIN][discovery_info[CONF_NAME]]
|
||||||
if CONF_SCAN_INTERVAL not in entry:
|
if CONF_SCAN_INTERVAL not in entry:
|
||||||
entry[CONF_SCAN_INTERVAL] = DEFAULT_SCAN_INTERVAL
|
entry[CONF_SCAN_INTERVAL] = DEFAULT_SCAN_INTERVAL
|
||||||
sensors.append(ModbusBinarySensor(hub, hass, entry))
|
sensors.append(ModbusBinarySensor(hub, entry))
|
||||||
|
|
||||||
async_add_entities(sensors)
|
async_add_entities(sensors)
|
||||||
|
|
||||||
|
|
||||||
class ModbusBinarySensor(BinarySensorEntity):
|
class ModbusBinarySensor(BasePlatform, BinarySensorEntity):
|
||||||
"""Modbus binary sensor."""
|
"""Modbus binary sensor."""
|
||||||
|
|
||||||
def __init__(self, hub, hass, entry):
|
|
||||||
"""Initialize the Modbus binary sensor."""
|
|
||||||
self._hub = hub
|
|
||||||
self._hass = hass
|
|
||||||
self._name = entry[CONF_NAME]
|
|
||||||
self._slave = entry.get(CONF_SLAVE)
|
|
||||||
self._address = int(entry[CONF_ADDRESS])
|
|
||||||
self._device_class = entry.get(CONF_DEVICE_CLASS)
|
|
||||||
self._input_type = entry[CONF_INPUT_TYPE]
|
|
||||||
self._value = None
|
|
||||||
self._available = True
|
|
||||||
self._scan_interval = timedelta(seconds=entry[CONF_SCAN_INTERVAL])
|
|
||||||
|
|
||||||
async def async_added_to_hass(self):
|
async def async_added_to_hass(self):
|
||||||
"""Handle entity which will be added."""
|
"""Handle entity which will be added."""
|
||||||
async_track_time_interval(self._hass, self.async_update, self._scan_interval)
|
await self.async_base_added_to_hass()
|
||||||
|
|
||||||
@property
|
|
||||||
def name(self):
|
|
||||||
"""Return the name of the sensor."""
|
|
||||||
return self._name
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def is_on(self):
|
def is_on(self):
|
||||||
"""Return the state of the sensor."""
|
"""Return the state of the sensor."""
|
||||||
return self._value
|
return self._value
|
||||||
|
|
||||||
@property
|
|
||||||
def device_class(self) -> str | None:
|
|
||||||
"""Return the device class of the sensor."""
|
|
||||||
return self._device_class
|
|
||||||
|
|
||||||
@property
|
|
||||||
def should_poll(self):
|
|
||||||
"""Return True if entity has to be polled for state.
|
|
||||||
|
|
||||||
False if entity pushes its state to HA.
|
|
||||||
"""
|
|
||||||
|
|
||||||
# Handle polling directly in this entity
|
|
||||||
return False
|
|
||||||
|
|
||||||
@property
|
|
||||||
def available(self) -> bool:
|
|
||||||
"""Return True if entity is available."""
|
|
||||||
return self._available
|
|
||||||
|
|
||||||
async def async_update(self, now=None):
|
async def async_update(self, now=None):
|
||||||
"""Update the state of the sensor."""
|
"""Update the state of the sensor."""
|
||||||
# remark "now" is a dummy parameter to avoid problems with
|
|
||||||
# async_track_time_interval
|
|
||||||
result = await self._hub.async_pymodbus_call(
|
result = await self._hub.async_pymodbus_call(
|
||||||
self._slave, self._address, 1, self._input_type
|
self._slave, self._address, 1, self._input_type
|
||||||
)
|
)
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
"""Support for Generic Modbus Thermostats."""
|
"""Support for Generic Modbus Thermostats."""
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
from datetime import timedelta
|
|
||||||
import logging
|
import logging
|
||||||
import struct
|
import struct
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
@ -12,19 +11,18 @@ from homeassistant.components.climate.const import (
|
||||||
SUPPORT_TARGET_TEMPERATURE,
|
SUPPORT_TARGET_TEMPERATURE,
|
||||||
)
|
)
|
||||||
from homeassistant.const import (
|
from homeassistant.const import (
|
||||||
|
CONF_ADDRESS,
|
||||||
CONF_NAME,
|
CONF_NAME,
|
||||||
CONF_OFFSET,
|
CONF_OFFSET,
|
||||||
CONF_SCAN_INTERVAL,
|
|
||||||
CONF_SLAVE,
|
|
||||||
CONF_STRUCTURE,
|
CONF_STRUCTURE,
|
||||||
CONF_TEMPERATURE_UNIT,
|
CONF_TEMPERATURE_UNIT,
|
||||||
TEMP_CELSIUS,
|
TEMP_CELSIUS,
|
||||||
TEMP_FAHRENHEIT,
|
TEMP_FAHRENHEIT,
|
||||||
)
|
)
|
||||||
from homeassistant.core import HomeAssistant
|
from homeassistant.core import HomeAssistant
|
||||||
from homeassistant.helpers.event import async_track_time_interval
|
|
||||||
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
||||||
|
|
||||||
|
from .base_platform import BasePlatform
|
||||||
from .const import (
|
from .const import (
|
||||||
ATTR_TEMPERATURE,
|
ATTR_TEMPERATURE,
|
||||||
CALL_TYPE_REGISTER_HOLDING,
|
CALL_TYPE_REGISTER_HOLDING,
|
||||||
|
@ -34,6 +32,7 @@ from .const import (
|
||||||
CONF_CURRENT_TEMP_REGISTER_TYPE,
|
CONF_CURRENT_TEMP_REGISTER_TYPE,
|
||||||
CONF_DATA_COUNT,
|
CONF_DATA_COUNT,
|
||||||
CONF_DATA_TYPE,
|
CONF_DATA_TYPE,
|
||||||
|
CONF_INPUT_TYPE,
|
||||||
CONF_MAX_TEMP,
|
CONF_MAX_TEMP,
|
||||||
CONF_MIN_TEMP,
|
CONF_MIN_TEMP,
|
||||||
CONF_PRECISION,
|
CONF_PRECISION,
|
||||||
|
@ -99,7 +98,7 @@ async def async_setup_platform(
|
||||||
async_add_entities(entities)
|
async_add_entities(entities)
|
||||||
|
|
||||||
|
|
||||||
class ModbusThermostat(ClimateEntity):
|
class ModbusThermostat(BasePlatform, ClimateEntity):
|
||||||
"""Representation of a Modbus Thermostat."""
|
"""Representation of a Modbus Thermostat."""
|
||||||
|
|
||||||
def __init__(
|
def __init__(
|
||||||
|
@ -108,9 +107,9 @@ class ModbusThermostat(ClimateEntity):
|
||||||
config: dict[str, Any],
|
config: dict[str, Any],
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Initialize the modbus thermostat."""
|
"""Initialize the modbus thermostat."""
|
||||||
self._hub: ModbusHub = hub
|
config[CONF_ADDRESS] = "0"
|
||||||
self._name = config[CONF_NAME]
|
config[CONF_INPUT_TYPE] = ""
|
||||||
self._slave = config.get(CONF_SLAVE)
|
super().__init__(hub, config)
|
||||||
self._target_temperature_register = config[CONF_TARGET_TEMP]
|
self._target_temperature_register = config[CONF_TARGET_TEMP]
|
||||||
self._current_temperature_register = config[CONF_CURRENT_TEMP]
|
self._current_temperature_register = config[CONF_CURRENT_TEMP]
|
||||||
self._current_temperature_register_type = config[
|
self._current_temperature_register_type = config[
|
||||||
|
@ -123,26 +122,15 @@ class ModbusThermostat(ClimateEntity):
|
||||||
self._count = config[CONF_DATA_COUNT]
|
self._count = config[CONF_DATA_COUNT]
|
||||||
self._precision = config[CONF_PRECISION]
|
self._precision = config[CONF_PRECISION]
|
||||||
self._scale = config[CONF_SCALE]
|
self._scale = config[CONF_SCALE]
|
||||||
self._scan_interval = timedelta(seconds=config[CONF_SCAN_INTERVAL])
|
|
||||||
self._offset = config[CONF_OFFSET]
|
self._offset = config[CONF_OFFSET]
|
||||||
self._unit = config[CONF_TEMPERATURE_UNIT]
|
self._unit = config[CONF_TEMPERATURE_UNIT]
|
||||||
self._max_temp = config[CONF_MAX_TEMP]
|
self._max_temp = config[CONF_MAX_TEMP]
|
||||||
self._min_temp = config[CONF_MIN_TEMP]
|
self._min_temp = config[CONF_MIN_TEMP]
|
||||||
self._temp_step = config[CONF_STEP]
|
self._temp_step = config[CONF_STEP]
|
||||||
self._available = True
|
|
||||||
|
|
||||||
async def async_added_to_hass(self):
|
async def async_added_to_hass(self):
|
||||||
"""Handle entity which will be added."""
|
"""Handle entity which will be added."""
|
||||||
async_track_time_interval(self.hass, self.async_update, self._scan_interval)
|
await self.async_base_added_to_hass()
|
||||||
|
|
||||||
@property
|
|
||||||
def should_poll(self):
|
|
||||||
"""Return True if entity has to be polled for state.
|
|
||||||
|
|
||||||
False if entity pushes its state to HA.
|
|
||||||
"""
|
|
||||||
# Handle polling directly in this entity
|
|
||||||
return False
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def supported_features(self):
|
def supported_features(self):
|
||||||
|
@ -164,11 +152,6 @@ class ModbusThermostat(ClimateEntity):
|
||||||
# Home Assistant expects this method.
|
# Home Assistant expects this method.
|
||||||
# We'll keep it here to avoid getting exceptions.
|
# We'll keep it here to avoid getting exceptions.
|
||||||
|
|
||||||
@property
|
|
||||||
def name(self):
|
|
||||||
"""Return the name of the climate device."""
|
|
||||||
return self._name
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def current_temperature(self):
|
def current_temperature(self):
|
||||||
"""Return the current temperature."""
|
"""Return the current temperature."""
|
||||||
|
@ -217,11 +200,6 @@ class ModbusThermostat(ClimateEntity):
|
||||||
self._available = result is not None
|
self._available = result is not None
|
||||||
await self.async_update()
|
await self.async_update()
|
||||||
|
|
||||||
@property
|
|
||||||
def available(self) -> bool:
|
|
||||||
"""Return True if entity is available."""
|
|
||||||
return self._available
|
|
||||||
|
|
||||||
async def async_update(self, now=None):
|
async def async_update(self, now=None):
|
||||||
"""Update Target & Current Temperature."""
|
"""Update Target & Current Temperature."""
|
||||||
# remark "now" is a dummy parameter to avoid problems with
|
# remark "now" is a dummy parameter to avoid problems with
|
||||||
|
|
|
@ -1,17 +1,14 @@
|
||||||
"""Support for Modbus covers."""
|
"""Support for Modbus covers."""
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
from datetime import timedelta
|
|
||||||
import logging
|
import logging
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
from homeassistant.components.cover import SUPPORT_CLOSE, SUPPORT_OPEN, CoverEntity
|
from homeassistant.components.cover import SUPPORT_CLOSE, SUPPORT_OPEN, CoverEntity
|
||||||
from homeassistant.const import (
|
from homeassistant.const import (
|
||||||
|
CONF_ADDRESS,
|
||||||
CONF_COVERS,
|
CONF_COVERS,
|
||||||
CONF_DEVICE_CLASS,
|
|
||||||
CONF_NAME,
|
CONF_NAME,
|
||||||
CONF_SCAN_INTERVAL,
|
|
||||||
CONF_SLAVE,
|
|
||||||
STATE_CLOSED,
|
STATE_CLOSED,
|
||||||
STATE_CLOSING,
|
STATE_CLOSING,
|
||||||
STATE_OPEN,
|
STATE_OPEN,
|
||||||
|
@ -20,15 +17,16 @@ from homeassistant.const import (
|
||||||
STATE_UNKNOWN,
|
STATE_UNKNOWN,
|
||||||
)
|
)
|
||||||
from homeassistant.core import HomeAssistant
|
from homeassistant.core import HomeAssistant
|
||||||
from homeassistant.helpers.event import async_track_time_interval
|
|
||||||
from homeassistant.helpers.restore_state import RestoreEntity
|
from homeassistant.helpers.restore_state import RestoreEntity
|
||||||
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
||||||
|
|
||||||
|
from .base_platform import BasePlatform
|
||||||
from .const import (
|
from .const import (
|
||||||
CALL_TYPE_COIL,
|
CALL_TYPE_COIL,
|
||||||
CALL_TYPE_REGISTER_HOLDING,
|
CALL_TYPE_REGISTER_HOLDING,
|
||||||
CALL_TYPE_WRITE_COIL,
|
CALL_TYPE_WRITE_COIL,
|
||||||
CALL_TYPE_WRITE_REGISTER,
|
CALL_TYPE_WRITE_REGISTER,
|
||||||
|
CONF_INPUT_TYPE,
|
||||||
CONF_REGISTER,
|
CONF_REGISTER,
|
||||||
CONF_STATE_CLOSED,
|
CONF_STATE_CLOSED,
|
||||||
CONF_STATE_CLOSING,
|
CONF_STATE_CLOSING,
|
||||||
|
@ -67,7 +65,7 @@ async def async_setup_platform(
|
||||||
async_add_entities(covers)
|
async_add_entities(covers)
|
||||||
|
|
||||||
|
|
||||||
class ModbusCover(CoverEntity, RestoreEntity):
|
class ModbusCover(BasePlatform, CoverEntity, RestoreEntity):
|
||||||
"""Representation of a Modbus cover."""
|
"""Representation of a Modbus cover."""
|
||||||
|
|
||||||
def __init__(
|
def __init__(
|
||||||
|
@ -76,21 +74,17 @@ class ModbusCover(CoverEntity, RestoreEntity):
|
||||||
config: dict[str, Any],
|
config: dict[str, Any],
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Initialize the modbus cover."""
|
"""Initialize the modbus cover."""
|
||||||
self._hub: ModbusHub = hub
|
config[CONF_ADDRESS] = "0"
|
||||||
|
config[CONF_INPUT_TYPE] = ""
|
||||||
|
super().__init__(hub, config)
|
||||||
self._coil = config.get(CALL_TYPE_COIL)
|
self._coil = config.get(CALL_TYPE_COIL)
|
||||||
self._device_class = config.get(CONF_DEVICE_CLASS)
|
|
||||||
self._name = config[CONF_NAME]
|
|
||||||
self._register = config.get(CONF_REGISTER)
|
self._register = config.get(CONF_REGISTER)
|
||||||
self._slave = config.get(CONF_SLAVE)
|
|
||||||
self._state_closed = config[CONF_STATE_CLOSED]
|
self._state_closed = config[CONF_STATE_CLOSED]
|
||||||
self._state_closing = config[CONF_STATE_CLOSING]
|
self._state_closing = config[CONF_STATE_CLOSING]
|
||||||
self._state_open = config[CONF_STATE_OPEN]
|
self._state_open = config[CONF_STATE_OPEN]
|
||||||
self._state_opening = config[CONF_STATE_OPENING]
|
self._state_opening = config[CONF_STATE_OPENING]
|
||||||
self._status_register = config.get(CONF_STATUS_REGISTER)
|
self._status_register = config.get(CONF_STATUS_REGISTER)
|
||||||
self._status_register_type = config[CONF_STATUS_REGISTER_TYPE]
|
self._status_register_type = config[CONF_STATUS_REGISTER_TYPE]
|
||||||
self._scan_interval = timedelta(seconds=config[CONF_SCAN_INTERVAL])
|
|
||||||
self._value = None
|
|
||||||
self._available = True
|
|
||||||
|
|
||||||
# If we read cover status from coil, and not from optional status register,
|
# If we read cover status from coil, and not from optional status register,
|
||||||
# we interpret boolean value False as closed cover, and value True as open cover.
|
# we interpret boolean value False as closed cover, and value True as open cover.
|
||||||
|
@ -114,6 +108,7 @@ class ModbusCover(CoverEntity, RestoreEntity):
|
||||||
|
|
||||||
async def async_added_to_hass(self):
|
async def async_added_to_hass(self):
|
||||||
"""Handle entity which will be added."""
|
"""Handle entity which will be added."""
|
||||||
|
await self.async_base_added_to_hass()
|
||||||
state = await self.async_get_last_state()
|
state = await self.async_get_last_state()
|
||||||
if state:
|
if state:
|
||||||
convert = {
|
convert = {
|
||||||
|
@ -126,28 +121,11 @@ class ModbusCover(CoverEntity, RestoreEntity):
|
||||||
}
|
}
|
||||||
self._value = convert[state.state]
|
self._value = convert[state.state]
|
||||||
|
|
||||||
async_track_time_interval(self.hass, self.async_update, self._scan_interval)
|
|
||||||
|
|
||||||
@property
|
|
||||||
def device_class(self) -> str | None:
|
|
||||||
"""Return the device class of the sensor."""
|
|
||||||
return self._device_class
|
|
||||||
|
|
||||||
@property
|
|
||||||
def name(self):
|
|
||||||
"""Return the name of the switch."""
|
|
||||||
return self._name
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def supported_features(self):
|
def supported_features(self):
|
||||||
"""Flag supported features."""
|
"""Flag supported features."""
|
||||||
return SUPPORT_OPEN | SUPPORT_CLOSE
|
return SUPPORT_OPEN | SUPPORT_CLOSE
|
||||||
|
|
||||||
@property
|
|
||||||
def available(self) -> bool:
|
|
||||||
"""Return True if entity is available."""
|
|
||||||
return self._available
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def is_opening(self):
|
def is_opening(self):
|
||||||
"""Return if the cover is opening or not."""
|
"""Return if the cover is opening or not."""
|
||||||
|
@ -163,15 +141,6 @@ class ModbusCover(CoverEntity, RestoreEntity):
|
||||||
"""Return if the cover is closed or not."""
|
"""Return if the cover is closed or not."""
|
||||||
return self._value == self._state_closed
|
return self._value == self._state_closed
|
||||||
|
|
||||||
@property
|
|
||||||
def should_poll(self):
|
|
||||||
"""Return True if entity has to be polled for state.
|
|
||||||
|
|
||||||
False if entity pushes its state to HA.
|
|
||||||
"""
|
|
||||||
# Handle polling directly in this entity
|
|
||||||
return False
|
|
||||||
|
|
||||||
async def async_open_cover(self, **kwargs: Any) -> None:
|
async def async_open_cover(self, **kwargs: Any) -> None:
|
||||||
"""Open cover."""
|
"""Open cover."""
|
||||||
result = await self._hub.async_pymodbus_call(
|
result = await self._hub.async_pymodbus_call(
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
"""Support for Modbus Register sensors."""
|
"""Support for Modbus Register sensors."""
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
from datetime import timedelta
|
|
||||||
import logging
|
import logging
|
||||||
import struct
|
import struct
|
||||||
|
|
||||||
|
@ -26,11 +25,11 @@ from homeassistant.const import (
|
||||||
)
|
)
|
||||||
from homeassistant.core import HomeAssistant
|
from homeassistant.core import HomeAssistant
|
||||||
from homeassistant.helpers import config_validation as cv
|
from homeassistant.helpers import config_validation as cv
|
||||||
from homeassistant.helpers.event import async_track_time_interval
|
|
||||||
from homeassistant.helpers.restore_state import RestoreEntity
|
from homeassistant.helpers.restore_state import RestoreEntity
|
||||||
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
||||||
|
|
||||||
from . import number
|
from . import number
|
||||||
|
from .base_platform import BasePlatform
|
||||||
from .const import (
|
from .const import (
|
||||||
CALL_TYPE_REGISTER_HOLDING,
|
CALL_TYPE_REGISTER_HOLDING,
|
||||||
CALL_TYPE_REGISTER_INPUT,
|
CALL_TYPE_REGISTER_INPUT,
|
||||||
|
@ -194,7 +193,7 @@ async def async_setup_platform(
|
||||||
async_add_entities(sensors)
|
async_add_entities(sensors)
|
||||||
|
|
||||||
|
|
||||||
class ModbusRegisterSensor(RestoreEntity, SensorEntity):
|
class ModbusRegisterSensor(BasePlatform, RestoreEntity, SensorEntity):
|
||||||
"""Modbus register sensor."""
|
"""Modbus register sensor."""
|
||||||
|
|
||||||
def __init__(
|
def __init__(
|
||||||
|
@ -204,12 +203,9 @@ class ModbusRegisterSensor(RestoreEntity, SensorEntity):
|
||||||
structure,
|
structure,
|
||||||
):
|
):
|
||||||
"""Initialize the modbus register sensor."""
|
"""Initialize the modbus register sensor."""
|
||||||
self._hub = hub
|
super().__init__(hub, entry)
|
||||||
self._name = entry[CONF_NAME]
|
self._register = self._address
|
||||||
slave = entry.get(CONF_SLAVE)
|
self._register_type = self._input_type
|
||||||
self._slave = int(slave) if slave else None
|
|
||||||
self._register = int(entry[CONF_ADDRESS])
|
|
||||||
self._register_type = entry[CONF_INPUT_TYPE]
|
|
||||||
self._unit_of_measurement = entry.get(CONF_UNIT_OF_MEASUREMENT)
|
self._unit_of_measurement = entry.get(CONF_UNIT_OF_MEASUREMENT)
|
||||||
self._count = int(entry[CONF_COUNT])
|
self._count = int(entry[CONF_COUNT])
|
||||||
self._swap = entry[CONF_SWAP]
|
self._swap = entry[CONF_SWAP]
|
||||||
|
@ -218,54 +214,24 @@ class ModbusRegisterSensor(RestoreEntity, SensorEntity):
|
||||||
self._precision = entry[CONF_PRECISION]
|
self._precision = entry[CONF_PRECISION]
|
||||||
self._structure = structure
|
self._structure = structure
|
||||||
self._data_type = entry[CONF_DATA_TYPE]
|
self._data_type = entry[CONF_DATA_TYPE]
|
||||||
self._device_class = entry.get(CONF_DEVICE_CLASS)
|
|
||||||
self._value = None
|
|
||||||
self._available = True
|
|
||||||
self._scan_interval = timedelta(seconds=entry.get(CONF_SCAN_INTERVAL))
|
|
||||||
|
|
||||||
async def async_added_to_hass(self):
|
async def async_added_to_hass(self):
|
||||||
"""Handle entity which will be added."""
|
"""Handle entity which will be added."""
|
||||||
|
await self.async_base_added_to_hass()
|
||||||
state = await self.async_get_last_state()
|
state = await self.async_get_last_state()
|
||||||
if state:
|
if state:
|
||||||
self._value = state.state
|
self._value = state.state
|
||||||
|
|
||||||
async_track_time_interval(self.hass, self.async_update, self._scan_interval)
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def state(self):
|
def state(self):
|
||||||
"""Return the state of the sensor."""
|
"""Return the state of the sensor."""
|
||||||
return self._value
|
return self._value
|
||||||
|
|
||||||
@property
|
|
||||||
def name(self):
|
|
||||||
"""Return the name of the sensor."""
|
|
||||||
return self._name
|
|
||||||
|
|
||||||
@property
|
|
||||||
def should_poll(self):
|
|
||||||
"""Return True if entity has to be polled for state.
|
|
||||||
|
|
||||||
False if entity pushes its state to HA.
|
|
||||||
"""
|
|
||||||
|
|
||||||
# Handle polling directly in this entity
|
|
||||||
return False
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def unit_of_measurement(self):
|
def unit_of_measurement(self):
|
||||||
"""Return the unit of measurement."""
|
"""Return the unit of measurement."""
|
||||||
return self._unit_of_measurement
|
return self._unit_of_measurement
|
||||||
|
|
||||||
@property
|
|
||||||
def device_class(self) -> str | None:
|
|
||||||
"""Return the device class of the sensor."""
|
|
||||||
return self._device_class
|
|
||||||
|
|
||||||
@property
|
|
||||||
def available(self) -> bool:
|
|
||||||
"""Return True if entity is available."""
|
|
||||||
return self._available
|
|
||||||
|
|
||||||
def _swap_registers(self, registers):
|
def _swap_registers(self, registers):
|
||||||
"""Do swap as needed."""
|
"""Do swap as needed."""
|
||||||
if self._swap in [CONF_SWAP_BYTE, CONF_SWAP_WORD_BYTE]:
|
if self._swap in [CONF_SWAP_BYTE, CONF_SWAP_WORD_BYTE]:
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
"""Support for Modbus switches."""
|
"""Support for Modbus switches."""
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
from datetime import timedelta
|
|
||||||
import logging
|
import logging
|
||||||
|
|
||||||
from homeassistant.components.switch import SwitchEntity
|
from homeassistant.components.switch import SwitchEntity
|
||||||
|
@ -10,16 +9,14 @@ from homeassistant.const import (
|
||||||
CONF_COMMAND_OFF,
|
CONF_COMMAND_OFF,
|
||||||
CONF_COMMAND_ON,
|
CONF_COMMAND_ON,
|
||||||
CONF_NAME,
|
CONF_NAME,
|
||||||
CONF_SCAN_INTERVAL,
|
|
||||||
CONF_SLAVE,
|
|
||||||
CONF_SWITCHES,
|
CONF_SWITCHES,
|
||||||
STATE_ON,
|
STATE_ON,
|
||||||
)
|
)
|
||||||
from homeassistant.core import HomeAssistant
|
from homeassistant.core import HomeAssistant
|
||||||
from homeassistant.helpers.event import async_track_time_interval
|
|
||||||
from homeassistant.helpers.restore_state import RestoreEntity
|
from homeassistant.helpers.restore_state import RestoreEntity
|
||||||
from homeassistant.helpers.typing import ConfigType
|
from homeassistant.helpers.typing import ConfigType
|
||||||
|
|
||||||
|
from .base_platform import BasePlatform
|
||||||
from .const import (
|
from .const import (
|
||||||
CALL_TYPE_COIL,
|
CALL_TYPE_COIL,
|
||||||
CALL_TYPE_WRITE_COIL,
|
CALL_TYPE_WRITE_COIL,
|
||||||
|
@ -49,18 +46,14 @@ async def async_setup_platform(
|
||||||
async_add_entities(switches)
|
async_add_entities(switches)
|
||||||
|
|
||||||
|
|
||||||
class ModbusSwitch(SwitchEntity, RestoreEntity):
|
class ModbusSwitch(BasePlatform, SwitchEntity, RestoreEntity):
|
||||||
"""Base class representing a Modbus switch."""
|
"""Base class representing a Modbus switch."""
|
||||||
|
|
||||||
def __init__(self, hub: ModbusHub, config: dict) -> None:
|
def __init__(self, hub: ModbusHub, config: dict) -> None:
|
||||||
"""Initialize the switch."""
|
"""Initialize the switch."""
|
||||||
self._hub: ModbusHub = hub
|
config[CONF_INPUT_TYPE] = ""
|
||||||
self._name = config[CONF_NAME]
|
super().__init__(hub, config)
|
||||||
self._slave = config.get(CONF_SLAVE)
|
|
||||||
self._is_on = None
|
self._is_on = None
|
||||||
self._available = True
|
|
||||||
self._scan_interval = timedelta(seconds=config[CONF_SCAN_INTERVAL])
|
|
||||||
self._address = config[CONF_ADDRESS]
|
|
||||||
if config[CONF_WRITE_TYPE] == CALL_TYPE_COIL:
|
if config[CONF_WRITE_TYPE] == CALL_TYPE_COIL:
|
||||||
self._write_type = CALL_TYPE_WRITE_COIL
|
self._write_type = CALL_TYPE_WRITE_COIL
|
||||||
else:
|
else:
|
||||||
|
@ -84,32 +77,16 @@ class ModbusSwitch(SwitchEntity, RestoreEntity):
|
||||||
|
|
||||||
async def async_added_to_hass(self):
|
async def async_added_to_hass(self):
|
||||||
"""Handle entity which will be added."""
|
"""Handle entity which will be added."""
|
||||||
|
await self.async_base_added_to_hass()
|
||||||
state = await self.async_get_last_state()
|
state = await self.async_get_last_state()
|
||||||
if state:
|
if state:
|
||||||
self._is_on = state.state == STATE_ON
|
self._is_on = state.state == STATE_ON
|
||||||
|
|
||||||
async_track_time_interval(self.hass, self.async_update, self._scan_interval)
|
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def is_on(self):
|
def is_on(self):
|
||||||
"""Return true if switch is on."""
|
"""Return true if switch is on."""
|
||||||
return self._is_on
|
return self._is_on
|
||||||
|
|
||||||
@property
|
|
||||||
def name(self):
|
|
||||||
"""Return the name of the switch."""
|
|
||||||
return self._name
|
|
||||||
|
|
||||||
@property
|
|
||||||
def should_poll(self):
|
|
||||||
"""Return True if entity has to be polled for state."""
|
|
||||||
return False
|
|
||||||
|
|
||||||
@property
|
|
||||||
def available(self) -> bool:
|
|
||||||
"""Return True if entity is available."""
|
|
||||||
return self._available
|
|
||||||
|
|
||||||
async def async_turn_on(self, **kwargs):
|
async def async_turn_on(self, **kwargs):
|
||||||
"""Set switch on."""
|
"""Set switch on."""
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue