2019-02-14 04:35:12 +00:00
|
|
|
"""Support for Modbus switches."""
|
2015-04-15 14:47:42 +00:00
|
|
|
import logging
|
2020-02-12 17:37:16 +00:00
|
|
|
from typing import Optional
|
2019-02-24 09:22:17 +00:00
|
|
|
|
2020-02-12 17:37:16 +00:00
|
|
|
from pymodbus.exceptions import ConnectionException, ModbusException
|
|
|
|
from pymodbus.pdu import ExceptionResponse
|
2016-09-13 20:47:44 +00:00
|
|
|
import voluptuous as vol
|
2015-04-15 14:47:42 +00:00
|
|
|
|
2019-12-19 13:00:22 +00:00
|
|
|
from homeassistant.components.switch import PLATFORM_SCHEMA
|
2017-11-16 06:17:10 +00:00
|
|
|
from homeassistant.const import (
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_COMMAND_OFF,
|
|
|
|
CONF_COMMAND_ON,
|
|
|
|
CONF_NAME,
|
|
|
|
CONF_SLAVE,
|
|
|
|
STATE_ON,
|
|
|
|
)
|
2019-02-24 09:22:17 +00:00
|
|
|
from homeassistant.helpers import config_validation as cv
|
2015-04-15 14:47:42 +00:00
|
|
|
from homeassistant.helpers.entity import ToggleEntity
|
2019-02-11 19:00:37 +00:00
|
|
|
from homeassistant.helpers.restore_state import RestoreEntity
|
2015-04-15 14:47:42 +00:00
|
|
|
|
2019-03-21 05:56:46 +00:00
|
|
|
from . import CONF_HUB, DEFAULT_HUB, DOMAIN as MODBUS_DOMAIN
|
|
|
|
|
2015-04-15 14:47:42 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2019-02-14 04:35:12 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_COIL = "coil"
|
|
|
|
CONF_COILS = "coils"
|
|
|
|
CONF_REGISTER = "register"
|
|
|
|
CONF_REGISTER_TYPE = "register_type"
|
|
|
|
CONF_REGISTERS = "registers"
|
|
|
|
CONF_STATE_OFF = "state_off"
|
|
|
|
CONF_STATE_ON = "state_on"
|
|
|
|
CONF_VERIFY_REGISTER = "verify_register"
|
|
|
|
CONF_VERIFY_STATE = "verify_state"
|
|
|
|
|
|
|
|
REGISTER_TYPE_HOLDING = "holding"
|
|
|
|
REGISTER_TYPE_INPUT = "input"
|
|
|
|
|
|
|
|
REGISTERS_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_COMMAND_OFF): cv.positive_int,
|
|
|
|
vol.Required(CONF_COMMAND_ON): cv.positive_int,
|
|
|
|
vol.Required(CONF_NAME): cv.string,
|
|
|
|
vol.Required(CONF_REGISTER): cv.positive_int,
|
|
|
|
vol.Optional(CONF_HUB, default=DEFAULT_HUB): cv.string,
|
|
|
|
vol.Optional(CONF_REGISTER_TYPE, default=REGISTER_TYPE_HOLDING): vol.In(
|
|
|
|
[REGISTER_TYPE_HOLDING, REGISTER_TYPE_INPUT]
|
|
|
|
),
|
|
|
|
vol.Optional(CONF_SLAVE): cv.positive_int,
|
|
|
|
vol.Optional(CONF_STATE_OFF): cv.positive_int,
|
|
|
|
vol.Optional(CONF_STATE_ON): cv.positive_int,
|
|
|
|
vol.Optional(CONF_VERIFY_REGISTER): cv.positive_int,
|
|
|
|
vol.Optional(CONF_VERIFY_STATE, default=True): cv.boolean,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
|
|
|
COILS_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Required(CONF_COIL): cv.positive_int,
|
|
|
|
vol.Required(CONF_NAME): cv.string,
|
|
|
|
vol.Required(CONF_SLAVE): cv.positive_int,
|
|
|
|
vol.Optional(CONF_HUB, default=DEFAULT_HUB): cv.string,
|
|
|
|
}
|
|
|
|
)
|
2016-09-13 20:47:44 +00:00
|
|
|
|
2017-11-16 06:17:10 +00:00
|
|
|
PLATFORM_SCHEMA = vol.All(
|
|
|
|
cv.has_at_least_one_key(CONF_COILS, CONF_REGISTERS),
|
2019-07-31 19:25:30 +00:00
|
|
|
PLATFORM_SCHEMA.extend(
|
|
|
|
{
|
|
|
|
vol.Optional(CONF_COILS): [COILS_SCHEMA],
|
|
|
|
vol.Optional(CONF_REGISTERS): [REGISTERS_SCHEMA],
|
|
|
|
}
|
|
|
|
),
|
|
|
|
)
|
2017-11-16 06:17:10 +00:00
|
|
|
|
2015-04-21 14:40:13 +00:00
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Read configuration and create Modbus devices."""
|
2015-04-15 14:47:42 +00:00
|
|
|
switches = []
|
2017-11-16 06:17:10 +00:00
|
|
|
if CONF_COILS in config:
|
|
|
|
for coil in config.get(CONF_COILS):
|
2019-02-11 19:00:37 +00:00
|
|
|
hub_name = coil.get(CONF_HUB)
|
|
|
|
hub = hass.data[MODBUS_DOMAIN][hub_name]
|
2019-07-31 19:25:30 +00:00
|
|
|
switches.append(
|
|
|
|
ModbusCoilSwitch(
|
|
|
|
hub, coil.get(CONF_NAME), coil.get(CONF_SLAVE), coil.get(CONF_COIL)
|
|
|
|
)
|
|
|
|
)
|
2017-11-16 06:17:10 +00:00
|
|
|
if CONF_REGISTERS in config:
|
|
|
|
for register in config.get(CONF_REGISTERS):
|
2019-02-11 19:00:37 +00:00
|
|
|
hub_name = register.get(CONF_HUB)
|
|
|
|
hub = hass.data[MODBUS_DOMAIN][hub_name]
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
switches.append(
|
|
|
|
ModbusRegisterSwitch(
|
|
|
|
hub,
|
|
|
|
register.get(CONF_NAME),
|
|
|
|
register.get(CONF_SLAVE),
|
|
|
|
register.get(CONF_REGISTER),
|
|
|
|
register.get(CONF_COMMAND_ON),
|
|
|
|
register.get(CONF_COMMAND_OFF),
|
|
|
|
register.get(CONF_VERIFY_STATE),
|
|
|
|
register.get(CONF_VERIFY_REGISTER),
|
|
|
|
register.get(CONF_REGISTER_TYPE),
|
|
|
|
register.get(CONF_STATE_ON),
|
|
|
|
register.get(CONF_STATE_OFF),
|
|
|
|
)
|
|
|
|
)
|
2019-02-14 04:35:12 +00:00
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
add_entities(switches)
|
2015-04-15 14:47:42 +00:00
|
|
|
|
2015-04-21 14:40:13 +00:00
|
|
|
|
2019-02-11 19:00:37 +00:00
|
|
|
class ModbusCoilSwitch(ToggleEntity, RestoreEntity):
|
2017-11-16 06:17:10 +00:00
|
|
|
"""Representation of a Modbus coil switch."""
|
2015-04-15 14:47:42 +00:00
|
|
|
|
2019-02-11 19:00:37 +00:00
|
|
|
def __init__(self, hub, name, slave, coil):
|
2017-11-16 06:17:10 +00:00
|
|
|
"""Initialize the coil switch."""
|
2019-02-11 19:00:37 +00:00
|
|
|
self._hub = hub
|
2015-04-15 14:47:42 +00:00
|
|
|
self._name = name
|
2016-09-13 20:47:44 +00:00
|
|
|
self._slave = int(slave) if slave else None
|
|
|
|
self._coil = int(coil)
|
2015-04-15 14:47:42 +00:00
|
|
|
self._is_on = None
|
2020-02-12 17:37:16 +00:00
|
|
|
self._available = True
|
2015-04-15 14:47:42 +00:00
|
|
|
|
2019-02-11 19:00:37 +00:00
|
|
|
async def async_added_to_hass(self):
|
|
|
|
"""Handle entity which will be added."""
|
|
|
|
state = await self.async_get_last_state()
|
|
|
|
if not state:
|
|
|
|
return
|
|
|
|
self._is_on = state.state == STATE_ON
|
|
|
|
|
2015-04-15 14:47:42 +00:00
|
|
|
@property
|
|
|
|
def is_on(self):
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Return true if switch is on."""
|
2015-04-15 14:47:42 +00:00
|
|
|
return self._is_on
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Return the name of the switch."""
|
2015-04-15 14:47:42 +00:00
|
|
|
return self._name
|
|
|
|
|
2020-02-12 17:37:16 +00:00
|
|
|
@property
|
|
|
|
def available(self) -> bool:
|
|
|
|
"""Return True if entity is available."""
|
|
|
|
return self._available
|
|
|
|
|
2015-04-15 14:47:42 +00:00
|
|
|
def turn_on(self, **kwargs):
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Set switch on."""
|
2020-02-12 17:37:16 +00:00
|
|
|
self._write_coil(self._coil, True)
|
2015-04-15 14:47:42 +00:00
|
|
|
|
|
|
|
def turn_off(self, **kwargs):
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Set switch off."""
|
2020-02-12 17:37:16 +00:00
|
|
|
self._write_coil(self._coil, False)
|
2015-04-15 14:47:42 +00:00
|
|
|
|
|
|
|
def update(self):
|
2016-03-08 12:35:39 +00:00
|
|
|
"""Update the state of the switch."""
|
2020-02-12 17:37:16 +00:00
|
|
|
self._is_on = self._read_coil(self._coil)
|
|
|
|
|
|
|
|
def _read_coil(self, coil) -> Optional[bool]:
|
|
|
|
"""Read coil using the Modbus hub slave."""
|
2017-07-07 05:30:23 +00:00
|
|
|
try:
|
2020-02-12 17:37:16 +00:00
|
|
|
result = self._hub.read_coils(self._slave, coil, 1)
|
|
|
|
except ConnectionException:
|
|
|
|
self._set_unavailable()
|
|
|
|
return
|
|
|
|
|
|
|
|
if isinstance(result, (ModbusException, ExceptionResponse)):
|
|
|
|
self._set_unavailable()
|
|
|
|
return
|
|
|
|
|
|
|
|
value = bool(result.bits[0])
|
|
|
|
self._available = True
|
|
|
|
|
|
|
|
return value
|
|
|
|
|
|
|
|
def _write_coil(self, coil, value):
|
|
|
|
"""Write coil using the Modbus hub slave."""
|
|
|
|
try:
|
|
|
|
self._hub.write_coil(self._slave, coil, value)
|
|
|
|
except ConnectionException:
|
|
|
|
self._set_unavailable()
|
|
|
|
return
|
|
|
|
|
|
|
|
self._available = True
|
|
|
|
|
|
|
|
def _set_unavailable(self):
|
|
|
|
"""Set unavailable state and log it as an error."""
|
|
|
|
if not self._available:
|
|
|
|
return
|
|
|
|
|
|
|
|
_LOGGER.error(
|
|
|
|
"No response from hub %s, slave %s, coil %s",
|
|
|
|
self._hub.name,
|
|
|
|
self._slave,
|
|
|
|
self._coil,
|
|
|
|
)
|
|
|
|
self._available = False
|
2017-11-16 06:17:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ModbusRegisterSwitch(ModbusCoilSwitch):
|
|
|
|
"""Representation of a Modbus register switch."""
|
|
|
|
|
|
|
|
# pylint: disable=super-init-not-called
|
2019-07-31 19:25:30 +00:00
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
hub,
|
|
|
|
name,
|
|
|
|
slave,
|
|
|
|
register,
|
|
|
|
command_on,
|
|
|
|
command_off,
|
|
|
|
verify_state,
|
|
|
|
verify_register,
|
|
|
|
register_type,
|
|
|
|
state_on,
|
|
|
|
state_off,
|
|
|
|
):
|
2017-11-16 06:17:10 +00:00
|
|
|
"""Initialize the register switch."""
|
2019-02-11 19:00:37 +00:00
|
|
|
self._hub = hub
|
2017-11-16 06:17:10 +00:00
|
|
|
self._name = name
|
|
|
|
self._slave = slave
|
|
|
|
self._register = register
|
|
|
|
self._command_on = command_on
|
|
|
|
self._command_off = command_off
|
|
|
|
self._verify_state = verify_state
|
2019-07-31 19:25:30 +00:00
|
|
|
self._verify_register = verify_register if verify_register else self._register
|
2017-11-16 06:17:10 +00:00
|
|
|
self._register_type = register_type
|
2020-02-12 17:37:16 +00:00
|
|
|
self._available = True
|
2017-12-21 13:24:19 +00:00
|
|
|
|
|
|
|
if state_on is not None:
|
|
|
|
self._state_on = state_on
|
|
|
|
else:
|
|
|
|
self._state_on = self._command_on
|
|
|
|
|
|
|
|
if state_off is not None:
|
|
|
|
self._state_off = state_off
|
|
|
|
else:
|
|
|
|
self._state_off = self._command_off
|
|
|
|
|
2017-11-16 06:17:10 +00:00
|
|
|
self._is_on = None
|
|
|
|
|
|
|
|
def turn_on(self, **kwargs):
|
|
|
|
"""Set switch on."""
|
2020-02-12 17:37:16 +00:00
|
|
|
|
|
|
|
# Only holding register is writable
|
|
|
|
if self._register_type == REGISTER_TYPE_HOLDING:
|
|
|
|
self._write_register(self._command_on)
|
|
|
|
if not self._verify_state:
|
|
|
|
self._is_on = True
|
2017-11-16 06:17:10 +00:00
|
|
|
|
|
|
|
def turn_off(self, **kwargs):
|
|
|
|
"""Set switch off."""
|
2020-02-12 17:37:16 +00:00
|
|
|
|
|
|
|
# Only holding register is writable
|
|
|
|
if self._register_type == REGISTER_TYPE_HOLDING:
|
|
|
|
self._write_register(self._command_off)
|
|
|
|
if not self._verify_state:
|
|
|
|
self._is_on = False
|
|
|
|
|
|
|
|
@property
|
|
|
|
def available(self) -> bool:
|
|
|
|
"""Return True if entity is available."""
|
|
|
|
return self._available
|
2017-11-16 06:17:10 +00:00
|
|
|
|
|
|
|
def update(self):
|
|
|
|
"""Update the state of the switch."""
|
|
|
|
if not self._verify_state:
|
|
|
|
return
|
|
|
|
|
2020-02-12 17:37:16 +00:00
|
|
|
value = self._read_register()
|
2017-11-16 06:17:10 +00:00
|
|
|
if value == self._state_on:
|
|
|
|
self._is_on = True
|
|
|
|
elif value == self._state_off:
|
|
|
|
self._is_on = False
|
2020-02-12 17:37:16 +00:00
|
|
|
elif value is not None:
|
2017-11-16 06:17:10 +00:00
|
|
|
_LOGGER.error(
|
2020-01-02 19:17:10 +00:00
|
|
|
"Unexpected response from hub %s, slave %s register %s, got 0x%2x",
|
2019-07-31 19:25:30 +00:00
|
|
|
self._hub.name,
|
|
|
|
self._slave,
|
2020-02-12 17:37:16 +00:00
|
|
|
self._register,
|
2019-07-31 19:25:30 +00:00
|
|
|
value,
|
|
|
|
)
|
2020-02-12 17:37:16 +00:00
|
|
|
|
|
|
|
def _read_register(self) -> Optional[int]:
|
|
|
|
try:
|
|
|
|
if self._register_type == REGISTER_TYPE_INPUT:
|
|
|
|
result = self._hub.read_input_registers(self._slave, self._register, 1)
|
|
|
|
else:
|
|
|
|
result = self._hub.read_holding_registers(
|
|
|
|
self._slave, self._register, 1
|
|
|
|
)
|
|
|
|
except ConnectionException:
|
|
|
|
self._set_unavailable()
|
|
|
|
return
|
|
|
|
|
|
|
|
if isinstance(result, (ModbusException, ExceptionResponse)):
|
|
|
|
self._set_unavailable()
|
|
|
|
return
|
|
|
|
|
|
|
|
value = int(result.registers[0])
|
|
|
|
self._available = True
|
|
|
|
|
|
|
|
return value
|
|
|
|
|
|
|
|
def _write_register(self, value):
|
|
|
|
"""Write holding register using the Modbus hub slave."""
|
|
|
|
try:
|
|
|
|
self._hub.write_register(self._slave, self._register, value)
|
|
|
|
except ConnectionException:
|
|
|
|
self._set_unavailable()
|
|
|
|
return
|
|
|
|
|
|
|
|
self._available = True
|
|
|
|
|
|
|
|
def _set_unavailable(self):
|
|
|
|
"""Set unavailable state and log it as an error."""
|
|
|
|
if not self._available:
|
|
|
|
return
|
|
|
|
|
|
|
|
_LOGGER.error(
|
|
|
|
"No response from hub %s, slave %s, register %s",
|
|
|
|
self._hub.name,
|
|
|
|
self._slave,
|
|
|
|
self._register,
|
|
|
|
)
|
|
|
|
self._available = False
|