mirror of
https://github.com/home-assistant/core.git
synced 2025-07-19 19:27:45 +00:00
Add switcher s12 support (#127277)
Co-authored-by: Joostlek <joostlek@outlook.com> Co-authored-by: Shay Levy <levyshay1@gmail.com>
This commit is contained in:
parent
21256c4529
commit
c24579bfb2
@ -40,21 +40,27 @@ async def async_setup_entry(
|
|||||||
@callback
|
@callback
|
||||||
def async_add_cover(coordinator: SwitcherDataUpdateCoordinator) -> None:
|
def async_add_cover(coordinator: SwitcherDataUpdateCoordinator) -> None:
|
||||||
"""Add cover from Switcher device."""
|
"""Add cover from Switcher device."""
|
||||||
|
entities: list[CoverEntity] = []
|
||||||
if coordinator.data.device_type.category in (
|
if coordinator.data.device_type.category in (
|
||||||
DeviceCategory.SHUTTER,
|
DeviceCategory.SHUTTER,
|
||||||
DeviceCategory.SINGLE_SHUTTER_DUAL_LIGHT,
|
DeviceCategory.SINGLE_SHUTTER_DUAL_LIGHT,
|
||||||
):
|
):
|
||||||
async_add_entities([SwitcherCoverEntity(coordinator, 0)])
|
entities.append(SwitcherSingleCoverEntity(coordinator, 0))
|
||||||
|
if (
|
||||||
|
coordinator.data.device_type.category
|
||||||
|
== DeviceCategory.DUAL_SHUTTER_SINGLE_LIGHT
|
||||||
|
):
|
||||||
|
entities.extend(SwitcherDualCoverEntity(coordinator, i) for i in range(2))
|
||||||
|
async_add_entities(entities)
|
||||||
|
|
||||||
config_entry.async_on_unload(
|
config_entry.async_on_unload(
|
||||||
async_dispatcher_connect(hass, SIGNAL_DEVICE_ADD, async_add_cover)
|
async_dispatcher_connect(hass, SIGNAL_DEVICE_ADD, async_add_cover)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
class SwitcherCoverEntity(SwitcherEntity, CoverEntity):
|
class SwitcherBaseCoverEntity(SwitcherEntity, CoverEntity):
|
||||||
"""Representation of a Switcher cover entity."""
|
"""Representation of a Switcher cover entity."""
|
||||||
|
|
||||||
_attr_name = None
|
|
||||||
_attr_device_class = CoverDeviceClass.SHUTTER
|
_attr_device_class = CoverDeviceClass.SHUTTER
|
||||||
_attr_supported_features = (
|
_attr_supported_features = (
|
||||||
CoverEntityFeature.OPEN
|
CoverEntityFeature.OPEN
|
||||||
@ -62,19 +68,7 @@ class SwitcherCoverEntity(SwitcherEntity, CoverEntity):
|
|||||||
| CoverEntityFeature.SET_POSITION
|
| CoverEntityFeature.SET_POSITION
|
||||||
| CoverEntityFeature.STOP
|
| CoverEntityFeature.STOP
|
||||||
)
|
)
|
||||||
|
_cover_id: int
|
||||||
def __init__(
|
|
||||||
self,
|
|
||||||
coordinator: SwitcherDataUpdateCoordinator,
|
|
||||||
cover_id: int,
|
|
||||||
) -> None:
|
|
||||||
"""Initialize the entity."""
|
|
||||||
super().__init__(coordinator)
|
|
||||||
self._cover_id = cover_id
|
|
||||||
|
|
||||||
self._attr_unique_id = f"{coordinator.device_id}-{coordinator.mac_address}"
|
|
||||||
|
|
||||||
self._update_data()
|
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def _handle_coordinator_update(self) -> None:
|
def _handle_coordinator_update(self) -> None:
|
||||||
@ -137,3 +131,44 @@ class SwitcherCoverEntity(SwitcherEntity, CoverEntity):
|
|||||||
async def async_stop_cover(self, **kwargs: Any) -> None:
|
async def async_stop_cover(self, **kwargs: Any) -> None:
|
||||||
"""Stop the cover."""
|
"""Stop the cover."""
|
||||||
await self._async_call_api(API_STOP, self._cover_id)
|
await self._async_call_api(API_STOP, self._cover_id)
|
||||||
|
|
||||||
|
|
||||||
|
class SwitcherSingleCoverEntity(SwitcherBaseCoverEntity):
|
||||||
|
"""Representation of a Switcher single cover entity."""
|
||||||
|
|
||||||
|
_attr_name = None
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
coordinator: SwitcherDataUpdateCoordinator,
|
||||||
|
cover_id: int,
|
||||||
|
) -> None:
|
||||||
|
"""Initialize the entity."""
|
||||||
|
super().__init__(coordinator)
|
||||||
|
self._cover_id = cover_id
|
||||||
|
|
||||||
|
self._attr_unique_id = f"{coordinator.device_id}-{coordinator.mac_address}"
|
||||||
|
|
||||||
|
self._update_data()
|
||||||
|
|
||||||
|
|
||||||
|
class SwitcherDualCoverEntity(SwitcherBaseCoverEntity):
|
||||||
|
"""Representation of a Switcher dual cover entity."""
|
||||||
|
|
||||||
|
_attr_translation_key = "cover"
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
coordinator: SwitcherDataUpdateCoordinator,
|
||||||
|
cover_id: int,
|
||||||
|
) -> None:
|
||||||
|
"""Initialize the entity."""
|
||||||
|
super().__init__(coordinator)
|
||||||
|
self._cover_id = cover_id
|
||||||
|
|
||||||
|
self._attr_translation_placeholders = {"cover_id": str(cover_id + 1)}
|
||||||
|
self._attr_unique_id = (
|
||||||
|
f"{coordinator.device_id}-{coordinator.mac_address}-{cover_id}"
|
||||||
|
)
|
||||||
|
|
||||||
|
self._update_data()
|
||||||
|
@ -34,42 +34,31 @@ async def async_setup_entry(
|
|||||||
@callback
|
@callback
|
||||||
def async_add_light(coordinator: SwitcherDataUpdateCoordinator) -> None:
|
def async_add_light(coordinator: SwitcherDataUpdateCoordinator) -> None:
|
||||||
"""Add light from Switcher device."""
|
"""Add light from Switcher device."""
|
||||||
|
entities: list[LightEntity] = []
|
||||||
if (
|
if (
|
||||||
coordinator.data.device_type.category
|
coordinator.data.device_type.category
|
||||||
== DeviceCategory.SINGLE_SHUTTER_DUAL_LIGHT
|
== DeviceCategory.SINGLE_SHUTTER_DUAL_LIGHT
|
||||||
):
|
):
|
||||||
async_add_entities(
|
entities.extend(SwitcherDualLightEntity(coordinator, i) for i in range(2))
|
||||||
[
|
if (
|
||||||
SwitcherLightEntity(coordinator, 0),
|
coordinator.data.device_type.category
|
||||||
SwitcherLightEntity(coordinator, 1),
|
== DeviceCategory.DUAL_SHUTTER_SINGLE_LIGHT
|
||||||
]
|
):
|
||||||
)
|
entities.append(SwitcherSingleLightEntity(coordinator, 0))
|
||||||
|
async_add_entities(entities)
|
||||||
|
|
||||||
config_entry.async_on_unload(
|
config_entry.async_on_unload(
|
||||||
async_dispatcher_connect(hass, SIGNAL_DEVICE_ADD, async_add_light)
|
async_dispatcher_connect(hass, SIGNAL_DEVICE_ADD, async_add_light)
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
class SwitcherLightEntity(SwitcherEntity, LightEntity):
|
class SwitcherBaseLightEntity(SwitcherEntity, LightEntity):
|
||||||
"""Representation of a Switcher light entity."""
|
"""Representation of a Switcher light entity."""
|
||||||
|
|
||||||
_attr_color_mode = ColorMode.ONOFF
|
_attr_color_mode = ColorMode.ONOFF
|
||||||
_attr_supported_color_modes = {ColorMode.ONOFF}
|
_attr_supported_color_modes = {ColorMode.ONOFF}
|
||||||
_attr_translation_key = "light"
|
control_result: bool | None = None
|
||||||
|
_light_id: int
|
||||||
def __init__(
|
|
||||||
self, coordinator: SwitcherDataUpdateCoordinator, light_id: int
|
|
||||||
) -> None:
|
|
||||||
"""Initialize the entity."""
|
|
||||||
super().__init__(coordinator)
|
|
||||||
self._light_id = light_id
|
|
||||||
self.control_result: bool | None = None
|
|
||||||
|
|
||||||
# Entity class attributes
|
|
||||||
self._attr_translation_placeholders = {"light_id": str(light_id + 1)}
|
|
||||||
self._attr_unique_id = (
|
|
||||||
f"{coordinator.device_id}-{coordinator.mac_address}-{light_id}"
|
|
||||||
)
|
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def _handle_coordinator_update(self) -> None:
|
def _handle_coordinator_update(self) -> None:
|
||||||
@ -123,3 +112,44 @@ class SwitcherLightEntity(SwitcherEntity, LightEntity):
|
|||||||
await self._async_call_api(API_SET_LIGHT, DeviceState.OFF, self._light_id)
|
await self._async_call_api(API_SET_LIGHT, DeviceState.OFF, self._light_id)
|
||||||
self.control_result = False
|
self.control_result = False
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
|
||||||
|
|
||||||
|
class SwitcherSingleLightEntity(SwitcherBaseLightEntity):
|
||||||
|
"""Representation of a Switcher single light entity."""
|
||||||
|
|
||||||
|
_attr_name = None
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
coordinator: SwitcherDataUpdateCoordinator,
|
||||||
|
light_id: int,
|
||||||
|
) -> None:
|
||||||
|
"""Initialize the entity."""
|
||||||
|
super().__init__(coordinator)
|
||||||
|
self._light_id = light_id
|
||||||
|
self.control_result: bool | None = None
|
||||||
|
|
||||||
|
# Entity class attributes
|
||||||
|
self._attr_unique_id = f"{coordinator.device_id}-{coordinator.mac_address}"
|
||||||
|
|
||||||
|
|
||||||
|
class SwitcherDualLightEntity(SwitcherBaseLightEntity):
|
||||||
|
"""Representation of a Switcher dual light entity."""
|
||||||
|
|
||||||
|
_attr_translation_key = "light"
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
coordinator: SwitcherDataUpdateCoordinator,
|
||||||
|
light_id: int,
|
||||||
|
) -> None:
|
||||||
|
"""Initialize the entity."""
|
||||||
|
super().__init__(coordinator)
|
||||||
|
self._light_id = light_id
|
||||||
|
self.control_result: bool | None = None
|
||||||
|
|
||||||
|
# Entity class attributes
|
||||||
|
self._attr_translation_placeholders = {"light_id": str(light_id + 1)}
|
||||||
|
self._attr_unique_id = (
|
||||||
|
f"{coordinator.device_id}-{coordinator.mac_address}-{light_id}"
|
||||||
|
)
|
||||||
|
@ -43,6 +43,11 @@
|
|||||||
"name": "Vertical swing off"
|
"name": "Vertical swing off"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"cover": {
|
||||||
|
"cover": {
|
||||||
|
"name": "Cover {cover_id}"
|
||||||
|
}
|
||||||
|
},
|
||||||
"light": {
|
"light": {
|
||||||
"light": {
|
"light": {
|
||||||
"name": "Light {light_id}"
|
"name": "Light {light_id}"
|
||||||
|
@ -4,6 +4,7 @@ from aioswitcher.device import (
|
|||||||
DeviceState,
|
DeviceState,
|
||||||
DeviceType,
|
DeviceType,
|
||||||
ShutterDirection,
|
ShutterDirection,
|
||||||
|
SwitcherDualShutterSingleLight,
|
||||||
SwitcherPowerPlug,
|
SwitcherPowerPlug,
|
||||||
SwitcherShutter,
|
SwitcherShutter,
|
||||||
SwitcherSingleShutterDualLight,
|
SwitcherSingleShutterDualLight,
|
||||||
@ -21,16 +22,19 @@ DUMMY_DEVICE_ID2 = "cafe12"
|
|||||||
DUMMY_DEVICE_ID3 = "bada77"
|
DUMMY_DEVICE_ID3 = "bada77"
|
||||||
DUMMY_DEVICE_ID4 = "bbd164"
|
DUMMY_DEVICE_ID4 = "bbd164"
|
||||||
DUMMY_DEVICE_ID5 = "bcdb64"
|
DUMMY_DEVICE_ID5 = "bcdb64"
|
||||||
|
DUMMY_DEVICE_ID6 = "bcdc64"
|
||||||
DUMMY_DEVICE_KEY1 = "18"
|
DUMMY_DEVICE_KEY1 = "18"
|
||||||
DUMMY_DEVICE_KEY2 = "01"
|
DUMMY_DEVICE_KEY2 = "01"
|
||||||
DUMMY_DEVICE_KEY3 = "12"
|
DUMMY_DEVICE_KEY3 = "12"
|
||||||
DUMMY_DEVICE_KEY4 = "07"
|
DUMMY_DEVICE_KEY4 = "07"
|
||||||
DUMMY_DEVICE_KEY5 = "15"
|
DUMMY_DEVICE_KEY5 = "15"
|
||||||
|
DUMMY_DEVICE_KEY6 = "16"
|
||||||
DUMMY_DEVICE_NAME1 = "Plug 23BC"
|
DUMMY_DEVICE_NAME1 = "Plug 23BC"
|
||||||
DUMMY_DEVICE_NAME2 = "Heater FE12"
|
DUMMY_DEVICE_NAME2 = "Heater FE12"
|
||||||
DUMMY_DEVICE_NAME3 = "Breeze AB39"
|
DUMMY_DEVICE_NAME3 = "Breeze AB39"
|
||||||
DUMMY_DEVICE_NAME4 = "Runner DD77"
|
DUMMY_DEVICE_NAME4 = "Runner DD77"
|
||||||
DUMMY_DEVICE_NAME5 = "RunnerS11 6CF5"
|
DUMMY_DEVICE_NAME5 = "RunnerS11 6CF5"
|
||||||
|
DUMMY_DEVICE_NAME6 = "RunnerS12 A9BE"
|
||||||
DUMMY_DEVICE_PASSWORD = "12345678"
|
DUMMY_DEVICE_PASSWORD = "12345678"
|
||||||
DUMMY_ELECTRIC_CURRENT1 = 0.5
|
DUMMY_ELECTRIC_CURRENT1 = 0.5
|
||||||
DUMMY_ELECTRIC_CURRENT2 = 12.8
|
DUMMY_ELECTRIC_CURRENT2 = 12.8
|
||||||
@ -39,16 +43,19 @@ DUMMY_IP_ADDRESS2 = "192.168.100.158"
|
|||||||
DUMMY_IP_ADDRESS3 = "192.168.100.159"
|
DUMMY_IP_ADDRESS3 = "192.168.100.159"
|
||||||
DUMMY_IP_ADDRESS4 = "192.168.100.160"
|
DUMMY_IP_ADDRESS4 = "192.168.100.160"
|
||||||
DUMMY_IP_ADDRESS5 = "192.168.100.161"
|
DUMMY_IP_ADDRESS5 = "192.168.100.161"
|
||||||
|
DUMMY_IP_ADDRESS6 = "192.168.100.162"
|
||||||
DUMMY_MAC_ADDRESS1 = "A1:B2:C3:45:67:D8"
|
DUMMY_MAC_ADDRESS1 = "A1:B2:C3:45:67:D8"
|
||||||
DUMMY_MAC_ADDRESS2 = "A1:B2:C3:45:67:D9"
|
DUMMY_MAC_ADDRESS2 = "A1:B2:C3:45:67:D9"
|
||||||
DUMMY_MAC_ADDRESS3 = "A1:B2:C3:45:67:DA"
|
DUMMY_MAC_ADDRESS3 = "A1:B2:C3:45:67:DA"
|
||||||
DUMMY_MAC_ADDRESS4 = "A1:B2:C3:45:67:DB"
|
DUMMY_MAC_ADDRESS4 = "A1:B2:C3:45:67:DB"
|
||||||
DUMMY_MAC_ADDRESS5 = "A1:B2:C3:45:67:DC"
|
DUMMY_MAC_ADDRESS5 = "A1:B2:C3:45:67:DC"
|
||||||
|
DUMMY_MAC_ADDRESS6 = "A1:B2:C3:45:67:DD"
|
||||||
DUMMY_TOKEN_NEEDED1 = False
|
DUMMY_TOKEN_NEEDED1 = False
|
||||||
DUMMY_TOKEN_NEEDED2 = False
|
DUMMY_TOKEN_NEEDED2 = False
|
||||||
DUMMY_TOKEN_NEEDED3 = False
|
DUMMY_TOKEN_NEEDED3 = False
|
||||||
DUMMY_TOKEN_NEEDED4 = False
|
DUMMY_TOKEN_NEEDED4 = False
|
||||||
DUMMY_TOKEN_NEEDED5 = True
|
DUMMY_TOKEN_NEEDED5 = True
|
||||||
|
DUMMY_TOKEN_NEEDED6 = True
|
||||||
DUMMY_PHONE_ID = "1234"
|
DUMMY_PHONE_ID = "1234"
|
||||||
DUMMY_POWER_CONSUMPTION1 = 100
|
DUMMY_POWER_CONSUMPTION1 = 100
|
||||||
DUMMY_POWER_CONSUMPTION2 = 2780
|
DUMMY_POWER_CONSUMPTION2 = 2780
|
||||||
@ -61,9 +68,12 @@ DUMMY_FAN_LEVEL = ThermostatFanLevel.LOW
|
|||||||
DUMMY_SWING = ThermostatSwing.OFF
|
DUMMY_SWING = ThermostatSwing.OFF
|
||||||
DUMMY_REMOTE_ID = "ELEC7001"
|
DUMMY_REMOTE_ID = "ELEC7001"
|
||||||
DUMMY_POSITION = [54]
|
DUMMY_POSITION = [54]
|
||||||
|
DUMMY_POSITION_2 = [54, 54]
|
||||||
DUMMY_DIRECTION = [ShutterDirection.SHUTTER_STOP]
|
DUMMY_DIRECTION = [ShutterDirection.SHUTTER_STOP]
|
||||||
|
DUMMY_DIRECTION_2 = [ShutterDirection.SHUTTER_STOP, ShutterDirection.SHUTTER_STOP]
|
||||||
DUMMY_USERNAME = "email"
|
DUMMY_USERNAME = "email"
|
||||||
DUMMY_TOKEN = "zvVvd7JxtN7CgvkD1Psujw=="
|
DUMMY_TOKEN = "zvVvd7JxtN7CgvkD1Psujw=="
|
||||||
|
DUMMY_LIGHT = [DeviceState.ON]
|
||||||
DUMMY_LIGHT_2 = [DeviceState.ON, DeviceState.ON]
|
DUMMY_LIGHT_2 = [DeviceState.ON, DeviceState.ON]
|
||||||
|
|
||||||
DUMMY_PLUG_DEVICE = SwitcherPowerPlug(
|
DUMMY_PLUG_DEVICE = SwitcherPowerPlug(
|
||||||
@ -121,6 +131,20 @@ DUMMY_SINGLE_SHUTTER_DUAL_LIGHT_DEVICE = SwitcherSingleShutterDualLight(
|
|||||||
DUMMY_LIGHT_2,
|
DUMMY_LIGHT_2,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
DUMMY_DUAL_SHUTTER_SINGLE_LIGHT_DEVICE = SwitcherDualShutterSingleLight(
|
||||||
|
DeviceType.RUNNER_S12,
|
||||||
|
DeviceState.ON,
|
||||||
|
DUMMY_DEVICE_ID6,
|
||||||
|
DUMMY_DEVICE_KEY6,
|
||||||
|
DUMMY_IP_ADDRESS6,
|
||||||
|
DUMMY_MAC_ADDRESS6,
|
||||||
|
DUMMY_DEVICE_NAME6,
|
||||||
|
DUMMY_TOKEN_NEEDED6,
|
||||||
|
DUMMY_POSITION_2,
|
||||||
|
DUMMY_DIRECTION_2,
|
||||||
|
DUMMY_LIGHT,
|
||||||
|
)
|
||||||
|
|
||||||
DUMMY_THERMOSTAT_DEVICE = SwitcherThermostat(
|
DUMMY_THERMOSTAT_DEVICE = SwitcherThermostat(
|
||||||
DeviceType.BREEZE,
|
DeviceType.BREEZE,
|
||||||
DeviceState.ON,
|
DeviceState.ON,
|
||||||
|
@ -11,6 +11,7 @@ from homeassistant.core import HomeAssistant
|
|||||||
from homeassistant.data_entry_flow import FlowResultType
|
from homeassistant.data_entry_flow import FlowResultType
|
||||||
|
|
||||||
from .consts import (
|
from .consts import (
|
||||||
|
DUMMY_DUAL_SHUTTER_SINGLE_LIGHT_DEVICE,
|
||||||
DUMMY_PLUG_DEVICE,
|
DUMMY_PLUG_DEVICE,
|
||||||
DUMMY_SINGLE_SHUTTER_DUAL_LIGHT_DEVICE,
|
DUMMY_SINGLE_SHUTTER_DUAL_LIGHT_DEVICE,
|
||||||
DUMMY_TOKEN,
|
DUMMY_TOKEN,
|
||||||
@ -62,6 +63,7 @@ async def test_user_setup(
|
|||||||
[
|
[
|
||||||
[
|
[
|
||||||
DUMMY_SINGLE_SHUTTER_DUAL_LIGHT_DEVICE,
|
DUMMY_SINGLE_SHUTTER_DUAL_LIGHT_DEVICE,
|
||||||
|
DUMMY_DUAL_SHUTTER_SINGLE_LIGHT_DEVICE,
|
||||||
]
|
]
|
||||||
],
|
],
|
||||||
indirect=True,
|
indirect=True,
|
||||||
@ -106,6 +108,7 @@ async def test_user_setup_found_token_device_valid_token(
|
|||||||
[
|
[
|
||||||
[
|
[
|
||||||
DUMMY_SINGLE_SHUTTER_DUAL_LIGHT_DEVICE,
|
DUMMY_SINGLE_SHUTTER_DUAL_LIGHT_DEVICE,
|
||||||
|
DUMMY_DUAL_SHUTTER_SINGLE_LIGHT_DEVICE,
|
||||||
]
|
]
|
||||||
],
|
],
|
||||||
indirect=True,
|
indirect=True,
|
||||||
|
@ -23,6 +23,7 @@ from homeassistant.util import slugify
|
|||||||
|
|
||||||
from . import init_integration
|
from . import init_integration
|
||||||
from .consts import (
|
from .consts import (
|
||||||
|
DUMMY_DUAL_SHUTTER_SINGLE_LIGHT_DEVICE as DEVICE3,
|
||||||
DUMMY_SHUTTER_DEVICE as DEVICE,
|
DUMMY_SHUTTER_DEVICE as DEVICE,
|
||||||
DUMMY_SINGLE_SHUTTER_DUAL_LIGHT_DEVICE as DEVICE2,
|
DUMMY_SINGLE_SHUTTER_DUAL_LIGHT_DEVICE as DEVICE2,
|
||||||
DUMMY_TOKEN as TOKEN,
|
DUMMY_TOKEN as TOKEN,
|
||||||
@ -31,16 +32,65 @@ from .consts import (
|
|||||||
|
|
||||||
ENTITY_ID = f"{COVER_DOMAIN}.{slugify(DEVICE.name)}"
|
ENTITY_ID = f"{COVER_DOMAIN}.{slugify(DEVICE.name)}"
|
||||||
ENTITY_ID2 = f"{COVER_DOMAIN}.{slugify(DEVICE2.name)}"
|
ENTITY_ID2 = f"{COVER_DOMAIN}.{slugify(DEVICE2.name)}"
|
||||||
|
ENTITY_ID3 = f"{COVER_DOMAIN}.{slugify(DEVICE3.name)}_cover_1"
|
||||||
|
ENTITY_ID3_2 = f"{COVER_DOMAIN}.{slugify(DEVICE3.name)}_cover_2"
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize(
|
@pytest.mark.parametrize(
|
||||||
("device", "entity_id"),
|
(
|
||||||
|
"device",
|
||||||
|
"entity_id",
|
||||||
|
"cover_id",
|
||||||
|
"position_open",
|
||||||
|
"position_close",
|
||||||
|
"direction_open",
|
||||||
|
"direction_close",
|
||||||
|
"direction_stop",
|
||||||
|
),
|
||||||
[
|
[
|
||||||
(DEVICE, ENTITY_ID),
|
(
|
||||||
(DEVICE2, ENTITY_ID2),
|
DEVICE,
|
||||||
|
ENTITY_ID,
|
||||||
|
0,
|
||||||
|
[77],
|
||||||
|
[0],
|
||||||
|
[ShutterDirection.SHUTTER_UP],
|
||||||
|
[ShutterDirection.SHUTTER_DOWN],
|
||||||
|
[ShutterDirection.SHUTTER_STOP],
|
||||||
|
),
|
||||||
|
(
|
||||||
|
DEVICE2,
|
||||||
|
ENTITY_ID2,
|
||||||
|
0,
|
||||||
|
[77],
|
||||||
|
[0],
|
||||||
|
[ShutterDirection.SHUTTER_UP],
|
||||||
|
[ShutterDirection.SHUTTER_DOWN],
|
||||||
|
[ShutterDirection.SHUTTER_STOP],
|
||||||
|
),
|
||||||
|
(
|
||||||
|
DEVICE3,
|
||||||
|
ENTITY_ID3,
|
||||||
|
0,
|
||||||
|
[77, 0],
|
||||||
|
[0, 0],
|
||||||
|
[ShutterDirection.SHUTTER_UP, ShutterDirection.SHUTTER_STOP],
|
||||||
|
[ShutterDirection.SHUTTER_DOWN, ShutterDirection.SHUTTER_STOP],
|
||||||
|
[ShutterDirection.SHUTTER_STOP, ShutterDirection.SHUTTER_STOP],
|
||||||
|
),
|
||||||
|
(
|
||||||
|
DEVICE3,
|
||||||
|
ENTITY_ID3_2,
|
||||||
|
1,
|
||||||
|
[0, 77],
|
||||||
|
[0, 0],
|
||||||
|
[ShutterDirection.SHUTTER_STOP, ShutterDirection.SHUTTER_UP],
|
||||||
|
[ShutterDirection.SHUTTER_STOP, ShutterDirection.SHUTTER_DOWN],
|
||||||
|
[ShutterDirection.SHUTTER_STOP, ShutterDirection.SHUTTER_STOP],
|
||||||
|
),
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
@pytest.mark.parametrize("mock_bridge", [[DEVICE, DEVICE2]], indirect=True)
|
@pytest.mark.parametrize("mock_bridge", [[DEVICE, DEVICE2, DEVICE3]], indirect=True)
|
||||||
async def test_cover(
|
async def test_cover(
|
||||||
hass: HomeAssistant,
|
hass: HomeAssistant,
|
||||||
mock_bridge,
|
mock_bridge,
|
||||||
@ -48,6 +98,12 @@ async def test_cover(
|
|||||||
monkeypatch: pytest.MonkeyPatch,
|
monkeypatch: pytest.MonkeyPatch,
|
||||||
device,
|
device,
|
||||||
entity_id: str,
|
entity_id: str,
|
||||||
|
cover_id: int,
|
||||||
|
position_open: list[int],
|
||||||
|
position_close: list[int],
|
||||||
|
direction_open: list[ShutterDirection],
|
||||||
|
direction_close: list[ShutterDirection],
|
||||||
|
direction_stop: list[ShutterDirection],
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Test cover services."""
|
"""Test cover services."""
|
||||||
await init_integration(hass, USERNAME, TOKEN)
|
await init_integration(hass, USERNAME, TOKEN)
|
||||||
@ -68,12 +124,12 @@ async def test_cover(
|
|||||||
blocking=True,
|
blocking=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
monkeypatch.setattr(device, "position", [77])
|
monkeypatch.setattr(device, "position", position_open)
|
||||||
mock_bridge.mock_callbacks([device])
|
mock_bridge.mock_callbacks([device])
|
||||||
await hass.async_block_till_done()
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
assert mock_api.call_count == 2
|
assert mock_api.call_count == 2
|
||||||
mock_control_device.assert_called_once_with(77, 0)
|
mock_control_device.assert_called_once_with(77, cover_id)
|
||||||
state = hass.states.get(entity_id)
|
state = hass.states.get(entity_id)
|
||||||
assert state.state == CoverState.OPEN
|
assert state.state == CoverState.OPEN
|
||||||
assert state.attributes[ATTR_CURRENT_POSITION] == 77
|
assert state.attributes[ATTR_CURRENT_POSITION] == 77
|
||||||
@ -89,12 +145,12 @@ async def test_cover(
|
|||||||
blocking=True,
|
blocking=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
monkeypatch.setattr(device, "direction", [ShutterDirection.SHUTTER_UP])
|
monkeypatch.setattr(device, "direction", direction_open)
|
||||||
mock_bridge.mock_callbacks([device])
|
mock_bridge.mock_callbacks([device])
|
||||||
await hass.async_block_till_done()
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
assert mock_api.call_count == 4
|
assert mock_api.call_count == 4
|
||||||
mock_control_device.assert_called_once_with(100, 0)
|
mock_control_device.assert_called_once_with(100, cover_id)
|
||||||
state = hass.states.get(entity_id)
|
state = hass.states.get(entity_id)
|
||||||
assert state.state == CoverState.OPENING
|
assert state.state == CoverState.OPENING
|
||||||
|
|
||||||
@ -109,12 +165,12 @@ async def test_cover(
|
|||||||
blocking=True,
|
blocking=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
monkeypatch.setattr(device, "direction", [ShutterDirection.SHUTTER_DOWN])
|
monkeypatch.setattr(device, "direction", direction_close)
|
||||||
mock_bridge.mock_callbacks([device])
|
mock_bridge.mock_callbacks([device])
|
||||||
await hass.async_block_till_done()
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
assert mock_api.call_count == 6
|
assert mock_api.call_count == 6
|
||||||
mock_control_device.assert_called_once_with(0, 0)
|
mock_control_device.assert_called_once_with(0, cover_id)
|
||||||
state = hass.states.get(entity_id)
|
state = hass.states.get(entity_id)
|
||||||
assert state.state == CoverState.CLOSING
|
assert state.state == CoverState.CLOSING
|
||||||
|
|
||||||
@ -129,17 +185,17 @@ async def test_cover(
|
|||||||
blocking=True,
|
blocking=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
monkeypatch.setattr(device, "direction", [ShutterDirection.SHUTTER_STOP])
|
monkeypatch.setattr(device, "direction", direction_stop)
|
||||||
mock_bridge.mock_callbacks([device])
|
mock_bridge.mock_callbacks([device])
|
||||||
await hass.async_block_till_done()
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
assert mock_api.call_count == 8
|
assert mock_api.call_count == 8
|
||||||
mock_control_device.assert_called_once_with(0)
|
mock_control_device.assert_called_once_with(cover_id)
|
||||||
state = hass.states.get(entity_id)
|
state = hass.states.get(entity_id)
|
||||||
assert state.state == CoverState.OPEN
|
assert state.state == CoverState.OPEN
|
||||||
|
|
||||||
# Test closed on position == 0
|
# Test closed on position == 0
|
||||||
monkeypatch.setattr(device, "position", [0])
|
monkeypatch.setattr(device, "position", position_close)
|
||||||
mock_bridge.mock_callbacks([device])
|
mock_bridge.mock_callbacks([device])
|
||||||
await hass.async_block_till_done()
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
@ -149,19 +205,22 @@ async def test_cover(
|
|||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize(
|
@pytest.mark.parametrize(
|
||||||
("device", "entity_id"),
|
("device", "entity_id", "cover_id"),
|
||||||
[
|
[
|
||||||
(DEVICE, ENTITY_ID),
|
(DEVICE, ENTITY_ID, 0),
|
||||||
(DEVICE2, ENTITY_ID2),
|
(DEVICE2, ENTITY_ID2, 0),
|
||||||
|
(DEVICE3, ENTITY_ID3, 0),
|
||||||
|
(DEVICE3, ENTITY_ID3_2, 1),
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
@pytest.mark.parametrize("mock_bridge", [[DEVICE, DEVICE2]], indirect=True)
|
@pytest.mark.parametrize("mock_bridge", [[DEVICE, DEVICE2, DEVICE3]], indirect=True)
|
||||||
async def test_cover_control_fail(
|
async def test_cover_control_fail(
|
||||||
hass: HomeAssistant,
|
hass: HomeAssistant,
|
||||||
mock_bridge,
|
mock_bridge,
|
||||||
mock_api,
|
mock_api,
|
||||||
device,
|
device,
|
||||||
entity_id: str,
|
entity_id: str,
|
||||||
|
cover_id: int,
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Test cover control fail."""
|
"""Test cover control fail."""
|
||||||
await init_integration(hass, USERNAME, TOKEN)
|
await init_integration(hass, USERNAME, TOKEN)
|
||||||
@ -185,7 +244,7 @@ async def test_cover_control_fail(
|
|||||||
)
|
)
|
||||||
|
|
||||||
assert mock_api.call_count == 2
|
assert mock_api.call_count == 2
|
||||||
mock_control_device.assert_called_once_with(44, 0)
|
mock_control_device.assert_called_once_with(44, cover_id)
|
||||||
state = hass.states.get(entity_id)
|
state = hass.states.get(entity_id)
|
||||||
assert state.state == STATE_UNAVAILABLE
|
assert state.state == STATE_UNAVAILABLE
|
||||||
|
|
||||||
@ -210,16 +269,16 @@ async def test_cover_control_fail(
|
|||||||
)
|
)
|
||||||
|
|
||||||
assert mock_api.call_count == 4
|
assert mock_api.call_count == 4
|
||||||
mock_control_device.assert_called_once_with(27, 0)
|
mock_control_device.assert_called_once_with(27, cover_id)
|
||||||
state = hass.states.get(entity_id)
|
state = hass.states.get(entity_id)
|
||||||
assert state.state == STATE_UNAVAILABLE
|
assert state.state == STATE_UNAVAILABLE
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize("mock_bridge", [[DEVICE2]], indirect=True)
|
@pytest.mark.parametrize("mock_bridge", [[DEVICE2, DEVICE3]], indirect=True)
|
||||||
async def test_cover2_no_token(
|
async def test_cover2_no_token(
|
||||||
hass: HomeAssistant, mock_bridge, mock_api, monkeypatch: pytest.MonkeyPatch
|
hass: HomeAssistant, mock_bridge, mock_api, monkeypatch: pytest.MonkeyPatch
|
||||||
) -> None:
|
) -> None:
|
||||||
"""Test single cover dual light without token services."""
|
"""Test cover with token needed without token specified."""
|
||||||
await init_integration(hass)
|
await init_integration(hass)
|
||||||
assert mock_bridge
|
assert mock_bridge
|
||||||
|
|
||||||
|
@ -21,6 +21,7 @@ from homeassistant.util import slugify
|
|||||||
|
|
||||||
from . import init_integration
|
from . import init_integration
|
||||||
from .consts import (
|
from .consts import (
|
||||||
|
DUMMY_DUAL_SHUTTER_SINGLE_LIGHT_DEVICE as DEVICE2,
|
||||||
DUMMY_SINGLE_SHUTTER_DUAL_LIGHT_DEVICE as DEVICE,
|
DUMMY_SINGLE_SHUTTER_DUAL_LIGHT_DEVICE as DEVICE,
|
||||||
DUMMY_TOKEN as TOKEN,
|
DUMMY_TOKEN as TOKEN,
|
||||||
DUMMY_USERNAME as USERNAME,
|
DUMMY_USERNAME as USERNAME,
|
||||||
@ -28,21 +29,24 @@ from .consts import (
|
|||||||
|
|
||||||
ENTITY_ID = f"{LIGHT_DOMAIN}.{slugify(DEVICE.name)}_light_1"
|
ENTITY_ID = f"{LIGHT_DOMAIN}.{slugify(DEVICE.name)}_light_1"
|
||||||
ENTITY_ID2 = f"{LIGHT_DOMAIN}.{slugify(DEVICE.name)}_light_2"
|
ENTITY_ID2 = f"{LIGHT_DOMAIN}.{slugify(DEVICE.name)}_light_2"
|
||||||
|
ENTITY_ID3 = f"{LIGHT_DOMAIN}.{slugify(DEVICE2.name)}"
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize(
|
@pytest.mark.parametrize(
|
||||||
("entity_id", "light_id", "device_state"),
|
("device", "entity_id", "light_id", "device_state"),
|
||||||
[
|
[
|
||||||
(ENTITY_ID, 0, [DeviceState.OFF, DeviceState.ON]),
|
(DEVICE, ENTITY_ID, 0, [DeviceState.OFF, DeviceState.ON]),
|
||||||
(ENTITY_ID2, 1, [DeviceState.ON, DeviceState.OFF]),
|
(DEVICE, ENTITY_ID2, 1, [DeviceState.ON, DeviceState.OFF]),
|
||||||
|
(DEVICE2, ENTITY_ID3, 0, [DeviceState.OFF]),
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
@pytest.mark.parametrize("mock_bridge", [[DEVICE]], indirect=True)
|
@pytest.mark.parametrize("mock_bridge", [[DEVICE, DEVICE2]], indirect=True)
|
||||||
async def test_light(
|
async def test_light(
|
||||||
hass: HomeAssistant,
|
hass: HomeAssistant,
|
||||||
mock_bridge,
|
mock_bridge,
|
||||||
mock_api,
|
mock_api,
|
||||||
monkeypatch: pytest.MonkeyPatch,
|
monkeypatch: pytest.MonkeyPatch,
|
||||||
|
device,
|
||||||
entity_id: str,
|
entity_id: str,
|
||||||
light_id: int,
|
light_id: int,
|
||||||
device_state: list[DeviceState],
|
device_state: list[DeviceState],
|
||||||
@ -56,8 +60,8 @@ async def test_light(
|
|||||||
assert state.state == STATE_ON
|
assert state.state == STATE_ON
|
||||||
|
|
||||||
# Test state change on --> off for light
|
# Test state change on --> off for light
|
||||||
monkeypatch.setattr(DEVICE, "light", device_state)
|
monkeypatch.setattr(device, "light", device_state)
|
||||||
mock_bridge.mock_callbacks([DEVICE])
|
mock_bridge.mock_callbacks([device])
|
||||||
await hass.async_block_till_done()
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
state = hass.states.get(entity_id)
|
state = hass.states.get(entity_id)
|
||||||
@ -91,10 +95,11 @@ async def test_light(
|
|||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize(
|
@pytest.mark.parametrize(
|
||||||
("entity_id", "light_id", "device_state"),
|
("device", "entity_id", "light_id", "device_state"),
|
||||||
[
|
[
|
||||||
(ENTITY_ID, 0, [DeviceState.OFF, DeviceState.ON]),
|
(DEVICE, ENTITY_ID, 0, [DeviceState.OFF, DeviceState.ON]),
|
||||||
(ENTITY_ID2, 1, [DeviceState.ON, DeviceState.OFF]),
|
(DEVICE, ENTITY_ID2, 1, [DeviceState.ON, DeviceState.OFF]),
|
||||||
|
(DEVICE2, ENTITY_ID3, 0, [DeviceState.OFF]),
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
@pytest.mark.parametrize("mock_bridge", [[DEVICE]], indirect=True)
|
@pytest.mark.parametrize("mock_bridge", [[DEVICE]], indirect=True)
|
||||||
@ -104,6 +109,7 @@ async def test_light_control_fail(
|
|||||||
mock_api,
|
mock_api,
|
||||||
monkeypatch: pytest.MonkeyPatch,
|
monkeypatch: pytest.MonkeyPatch,
|
||||||
caplog: pytest.LogCaptureFixture,
|
caplog: pytest.LogCaptureFixture,
|
||||||
|
device,
|
||||||
entity_id: str,
|
entity_id: str,
|
||||||
light_id: int,
|
light_id: int,
|
||||||
device_state: list[DeviceState],
|
device_state: list[DeviceState],
|
||||||
@ -113,8 +119,8 @@ async def test_light_control_fail(
|
|||||||
assert mock_bridge
|
assert mock_bridge
|
||||||
|
|
||||||
# Test initial state - light off
|
# Test initial state - light off
|
||||||
monkeypatch.setattr(DEVICE, "light", device_state)
|
monkeypatch.setattr(device, "light", device_state)
|
||||||
mock_bridge.mock_callbacks([DEVICE])
|
mock_bridge.mock_callbacks([device])
|
||||||
await hass.async_block_till_done()
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
state = hass.states.get(entity_id)
|
state = hass.states.get(entity_id)
|
||||||
@ -139,7 +145,7 @@ async def test_light_control_fail(
|
|||||||
assert state.state == STATE_UNAVAILABLE
|
assert state.state == STATE_UNAVAILABLE
|
||||||
|
|
||||||
# Make device available again
|
# Make device available again
|
||||||
mock_bridge.mock_callbacks([DEVICE])
|
mock_bridge.mock_callbacks([device])
|
||||||
await hass.async_block_till_done()
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
state = hass.states.get(entity_id)
|
state = hass.states.get(entity_id)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user