From 20f7af25e937d471a60b06585fde8c6644c9ddb1 Mon Sep 17 00:00:00 2001 From: IceBotYT <34712694+IceBotYT@users.noreply.github.com> Date: Mon, 19 Aug 2024 15:25:11 -0400 Subject: [PATCH] Add switch platform to Nice G.O. (#124237) * Add switch platform to Nice G.O. * Replace cover with switch in switch.py * Use icon translations * Fix tests * Use constants in test_switch.py * Use ATTR_ENTITY_ID --- homeassistant/components/nice_go/__init__.py | 2 +- .../components/nice_go/coordinator.py | 3 ++ homeassistant/components/nice_go/icons.json | 9 ++++ homeassistant/components/nice_go/strings.json | 5 ++ homeassistant/components/nice_go/switch.py | 49 +++++++++++++++++++ .../nice_go/fixtures/get_all_barriers.json | 2 +- .../nice_go/snapshots/test_diagnostics.ambr | 2 + tests/components/nice_go/test_init.py | 2 +- tests/components/nice_go/test_switch.py | 43 ++++++++++++++++ 9 files changed, 114 insertions(+), 3 deletions(-) create mode 100644 homeassistant/components/nice_go/icons.json create mode 100644 homeassistant/components/nice_go/switch.py create mode 100644 tests/components/nice_go/test_switch.py diff --git a/homeassistant/components/nice_go/__init__.py b/homeassistant/components/nice_go/__init__.py index f0bfea58b89170..bf8dbfcaecdc0e 100644 --- a/homeassistant/components/nice_go/__init__.py +++ b/homeassistant/components/nice_go/__init__.py @@ -11,7 +11,7 @@ from .coordinator import NiceGOUpdateCoordinator _LOGGER = logging.getLogger(__name__) -PLATFORMS: list[Platform] = [Platform.COVER, Platform.LIGHT] +PLATFORMS: list[Platform] = [Platform.COVER, Platform.LIGHT, Platform.SWITCH] type NiceGOConfigEntry = ConfigEntry[NiceGOUpdateCoordinator] diff --git a/homeassistant/components/nice_go/coordinator.py b/homeassistant/components/nice_go/coordinator.py index 196ed0a211c256..323e0a08fe84ee 100644 --- a/homeassistant/components/nice_go/coordinator.py +++ b/homeassistant/components/nice_go/coordinator.py @@ -46,6 +46,7 @@ class NiceGODevice: light_status: bool fw_version: str connected: bool + vacation_mode: bool class NiceGOUpdateCoordinator(DataUpdateCoordinator[dict[str, NiceGODevice]]): @@ -105,6 +106,7 @@ async def _parse_barrier(self, barrier_state: BarrierState) -> NiceGODevice | No connected = barrier_state.connectionState.connected else: connected = False + vacation_mode = barrier_state.reported["vcnMode"] return NiceGODevice( id=device_id, @@ -113,6 +115,7 @@ async def _parse_barrier(self, barrier_state: BarrierState) -> NiceGODevice | No light_status=light_status, fw_version=fw_version, connected=connected, + vacation_mode=vacation_mode, ) async def _async_update_data(self) -> dict[str, NiceGODevice]: diff --git a/homeassistant/components/nice_go/icons.json b/homeassistant/components/nice_go/icons.json new file mode 100644 index 00000000000000..fdce7c43e242c1 --- /dev/null +++ b/homeassistant/components/nice_go/icons.json @@ -0,0 +1,9 @@ +{ + "entity": { + "switch": { + "vacation_mode": { + "default": "mdi:beach" + } + } + } +} diff --git a/homeassistant/components/nice_go/strings.json b/homeassistant/components/nice_go/strings.json index 261f71ebcbef2f..7f2ea1512ddc32 100644 --- a/homeassistant/components/nice_go/strings.json +++ b/homeassistant/components/nice_go/strings.json @@ -22,6 +22,11 @@ "light": { "name": "[%key:component::light::title%]" } + }, + "switch": { + "vacation_mode": { + "name": "Vacation mode" + } } }, "issues": { diff --git a/homeassistant/components/nice_go/switch.py b/homeassistant/components/nice_go/switch.py new file mode 100644 index 00000000000000..e7290aabdd6a3e --- /dev/null +++ b/homeassistant/components/nice_go/switch.py @@ -0,0 +1,49 @@ +"""Nice G.O. switch platform.""" + +from __future__ import annotations + +import logging +from typing import Any + +from homeassistant.components.switch import SwitchDeviceClass, SwitchEntity +from homeassistant.core import HomeAssistant +from homeassistant.helpers.entity_platform import AddEntitiesCallback + +from . import NiceGOConfigEntry +from .entity import NiceGOEntity + +_LOGGER = logging.getLogger(__name__) + + +async def async_setup_entry( + hass: HomeAssistant, + config_entry: NiceGOConfigEntry, + async_add_entities: AddEntitiesCallback, +) -> None: + """Set up Nice G.O. switch.""" + coordinator = config_entry.runtime_data + + async_add_entities( + NiceGOSwitchEntity(coordinator, device_id, device_data.name, "switch") + for device_id, device_data in coordinator.data.items() + ) + + +class NiceGOSwitchEntity(NiceGOEntity, SwitchEntity): + """Representation of a Nice G.O. switch.""" + + _attr_device_class = SwitchDeviceClass.SWITCH + _attr_translation_key = "vacation_mode" + + @property + def is_on(self) -> bool: + """Return if switch is on.""" + return self.data.vacation_mode + + async def async_turn_on(self, **kwargs: Any) -> None: + """Turn the switch on.""" + await self.coordinator.api.vacation_mode_on(self.data.id) + + async def async_turn_off(self, **kwargs: Any) -> None: + """Turn the switch off.""" + await self.coordinator.api.vacation_mode_off(self.data.id) diff --git a/tests/components/nice_go/fixtures/get_all_barriers.json b/tests/components/nice_go/fixtures/get_all_barriers.json index 481c73d91a8bf7..adb0fb4bacde7b 100644 --- a/tests/components/nice_go/fixtures/get_all_barriers.json +++ b/tests/components/nice_go/fixtures/get_all_barriers.json @@ -49,7 +49,7 @@ "migrationStatus": "DONE", "deviceId": "2", "lightStatus": "0,100", - "vcnMode": false, + "vcnMode": true, "deviceFwVersion": "1.2.3.4.5.6", "barrierStatus": "1,100,0,0,-1,0,3,0" }, diff --git a/tests/components/nice_go/snapshots/test_diagnostics.ambr b/tests/components/nice_go/snapshots/test_diagnostics.ambr index f79b0607ce2bfa..abd3b3103d1d71 100644 --- a/tests/components/nice_go/snapshots/test_diagnostics.ambr +++ b/tests/components/nice_go/snapshots/test_diagnostics.ambr @@ -9,6 +9,7 @@ 'id': '1', 'light_status': True, 'name': 'Test Garage 1', + 'vacation_mode': False, }), '2': dict({ 'barrier_status': 'open', @@ -17,6 +18,7 @@ 'id': '2', 'light_status': False, 'name': 'Test Garage 2', + 'vacation_mode': True, }), }), 'entry': dict({ diff --git a/tests/components/nice_go/test_init.py b/tests/components/nice_go/test_init.py index d6877d72724273..249622d23b0ecb 100644 --- a/tests/components/nice_go/test_init.py +++ b/tests/components/nice_go/test_init.py @@ -275,7 +275,7 @@ async def test_no_connection_state( "item": { "deviceId": "1", "desired": '{"key": "value"}', - "reported": '{"displayName":"Test Garage 1", "migrationStatus":"DONE", "barrierStatus": "1,100,0", "deviceFwVersion": "1.0.0", "lightStatus": "1,100"}', + "reported": '{"displayName":"Test Garage 1", "migrationStatus":"DONE", "barrierStatus": "1,100,0", "deviceFwVersion": "1.0.0", "lightStatus": "1,100", "vcnMode": false}', "connectionState": None, "version": None, "timestamp": None, diff --git a/tests/components/nice_go/test_switch.py b/tests/components/nice_go/test_switch.py new file mode 100644 index 00000000000000..f34cba495c92cc --- /dev/null +++ b/tests/components/nice_go/test_switch.py @@ -0,0 +1,43 @@ +"""Nice G.O. switch tests.""" + +from unittest.mock import AsyncMock + +from homeassistant.components.switch import ( + DOMAIN as SWITCH_DOMAIN, + SERVICE_TURN_OFF, + SERVICE_TURN_ON, +) +from homeassistant.const import ATTR_ENTITY_ID, Platform +from homeassistant.core import HomeAssistant + +from . import setup_integration + +from tests.common import MockConfigEntry + + +async def test_turn_on( + hass: HomeAssistant, mock_nice_go: AsyncMock, mock_config_entry: MockConfigEntry +) -> None: + """Test turn on switch.""" + await setup_integration(hass, mock_config_entry, [Platform.SWITCH]) + await hass.services.async_call( + SWITCH_DOMAIN, + SERVICE_TURN_ON, + {ATTR_ENTITY_ID: "switch.test_garage_1_vacation_mode"}, + blocking=True, + ) + mock_nice_go.vacation_mode_on.assert_called_once_with("1") + + +async def test_turn_off( + hass: HomeAssistant, mock_nice_go: AsyncMock, mock_config_entry: MockConfigEntry +) -> None: + """Test turn off switch.""" + await setup_integration(hass, mock_config_entry, [Platform.SWITCH]) + await hass.services.async_call( + SWITCH_DOMAIN, + SERVICE_TURN_OFF, + {ATTR_ENTITY_ID: "switch.test_garage_2_vacation_mode"}, + blocking=True, + ) + mock_nice_go.vacation_mode_off.assert_called_once_with("2")