Fix bool registers in modbus integration (#41506)

When a register is of type bool modbus returns a whole byte, but ONLY the
lowest bit determine true/false.
pull/41523/head
jan iversen 2020-10-08 23:52:41 +02:00 committed by GitHub
parent 8e3258cdfb
commit 4d9ff13384
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 52 additions and 33 deletions

View File

@ -122,5 +122,5 @@ class ModbusBinarySensor(BinarySensorEntity):
self._available = False
return
self._value = result.bits[0]
self._value = result.bits[0] & 1
self._available = True

View File

@ -228,7 +228,7 @@ class ModbusCover(CoverEntity, RestoreEntity):
self._available = False
return
value = bool(result.bits[0])
value = bool(result.bits[0] & 1)
self._available = True
return value

View File

@ -174,7 +174,7 @@ class ModbusCoilSwitch(ToggleEntity, RestoreEntity):
# bits[0] select the lowest bit in result,
# is_on for a binary_sensor is true if the bit are 1
# The other bits are not considered.
return bool(result.bits[0])
return bool(result.bits[0] & 1)
def _write_coil(self, coil, value):
"""Write coil using the Modbus hub slave."""

View File

@ -29,6 +29,13 @@ _LOGGER = logging.getLogger(__name__)
[0xFF],
STATE_ON,
),
(
{
CONF_INPUT_TYPE: CALL_TYPE_COIL,
},
[0x01],
STATE_ON,
),
(
{
CONF_INPUT_TYPE: CALL_TYPE_COIL,
@ -36,6 +43,20 @@ _LOGGER = logging.getLogger(__name__)
[0x00],
STATE_OFF,
),
(
{
CONF_INPUT_TYPE: CALL_TYPE_COIL,
},
[0x80],
STATE_OFF,
),
(
{
CONF_INPUT_TYPE: CALL_TYPE_COIL,
},
[0xFE],
STATE_OFF,
),
(
{
CONF_INPUT_TYPE: CALL_TYPE_DISCRETE,

View File

@ -13,34 +13,6 @@ from .conftest import run_base_read_test, setup_base_test
_LOGGER = logging.getLogger(__name__)
async def run_sensor_test(hass, use_mock_hub, value, expected):
"""Run test for given config."""
switch_name = "modbus_test_switch"
scan_interval = 5
entity_id, now, device = await setup_base_test(
switch_name,
hass,
use_mock_hub,
{
CONF_COILS: [
{CONF_NAME: switch_name, CALL_TYPE_COIL: 1234, CONF_SLAVE: 1},
]
},
SWITCH_DOMAIN,
scan_interval,
)
await run_base_read_test(
entity_id,
hass,
use_mock_hub,
CALL_TYPE_COIL,
value,
expected,
now + timedelta(seconds=scan_interval + 1),
)
@pytest.mark.parametrize(
"regs,expected",
[
@ -48,6 +20,14 @@ async def run_sensor_test(hass, use_mock_hub, value, expected):
[0x00],
STATE_OFF,
),
(
[0x80],
STATE_OFF,
),
(
[0xFE],
STATE_OFF,
),
(
[0xFF],
STATE_ON,
@ -59,10 +39,28 @@ async def run_sensor_test(hass, use_mock_hub, value, expected):
],
)
async def test_coil_switch(hass, mock_hub, regs, expected):
"""Test reading of switch coil."""
await run_sensor_test(
"""Run test for given config."""
switch_name = "modbus_test_switch"
scan_interval = 5
entity_id, now, device = await setup_base_test(
switch_name,
hass,
mock_hub,
{
CONF_COILS: [
{CONF_NAME: switch_name, CALL_TYPE_COIL: 1234, CONF_SLAVE: 1},
]
},
SWITCH_DOMAIN,
scan_interval,
)
await run_base_read_test(
entity_id,
hass,
mock_hub,
CALL_TYPE_COIL,
regs,
expected,
now + timedelta(seconds=scan_interval + 1),
)