Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Update Vesync light state after on / off commands #135958

Draft
wants to merge 8 commits into
base: dev
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
9 changes: 7 additions & 2 deletions homeassistant/components/vesync/light.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@
)
from homeassistant.config_entries import ConfigEntry
from homeassistant.core import HomeAssistant, callback
from homeassistant.exceptions import HomeAssistantError
from homeassistant.helpers.dispatcher import async_dispatcher_connect
from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
from homeassistant.util import color as color_util
Expand Down Expand Up @@ -137,11 +138,15 @@ def turn_on(self, **kwargs: Any) -> None:
if attribute_adjustment_only:
return
# send turn_on command to pyvesync api
self.device.turn_on()
if not self.device.turn_on():
raise HomeAssistantError("An error occurred while turning on.")
self.schedule_update_ha_state()

def turn_off(self, **kwargs: Any) -> None:
"""Turn the device off."""
self.device.turn_off()
if not self.device.turn_off():
raise HomeAssistantError("An error occurred while turning off.")
self.schedule_update_ha_state()


class VeSyncDimmableLightHA(VeSyncBaseLightHA, LightEntity):
Expand Down
2 changes: 2 additions & 0 deletions tests/components/vesync/common.py
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,8 @@
ENTITY_HUMIDIFIER_HUMIDITY = "sensor.humidifier_200s_humidity"
ENTITY_HUMIDIFIER_300S_NIGHT_LIGHT_SELECT = "select.humidifier_300s_night_light_level"

ENTITY_LIGHT = "light.dimmer_switch"

ALL_DEVICES = load_json_object_fixture("vesync-devices.json", DOMAIN)
ALL_DEVICE_NAMES: list[str] = [
dev["deviceName"] for dev in ALL_DEVICES["result"]["list"]
Expand Down
20 changes: 20 additions & 0 deletions tests/components/vesync/conftest.py
Original file line number Diff line number Diff line change
Expand Up @@ -183,6 +183,26 @@ async def humidifier_config_entry(
return entry


@pytest.fixture(name="light_config_entry")
async def light_config_entry(
hass: HomeAssistant, requests_mock: requests_mock.Mocker, config
) -> MockConfigEntry:
"""Create a mock VeSync config entry for `Dimmable Light`."""
entry = MockConfigEntry(
title="VeSync",
domain=DOMAIN,
data=config[DOMAIN],
)
entry.add_to_hass(hass)

device_name = "Dimmer Switch"
mock_multiple_device_responses(requests_mock, [device_name])
await hass.config_entries.async_setup(entry.entry_id)
await hass.async_block_till_done()

return entry


@pytest.fixture
async def install_humidifier_device(
hass: HomeAssistant,
Expand Down
81 changes: 80 additions & 1 deletion tests/components/vesync/test_light.py
Original file line number Diff line number Diff line change
@@ -1,17 +1,24 @@
"""Tests for the light module."""

from contextlib import nullcontext
from unittest.mock import patch

import pytest
import requests_mock
from syrupy import SnapshotAssertion

from homeassistant.components.light import DOMAIN as LIGHT_DOMAIN
from homeassistant.const import ATTR_ENTITY_ID, SERVICE_TURN_OFF, SERVICE_TURN_ON
from homeassistant.core import HomeAssistant
from homeassistant.exceptions import HomeAssistantError
from homeassistant.helpers import device_registry as dr, entity_registry as er

from .common import ALL_DEVICE_NAMES, mock_devices_response
from .common import ALL_DEVICE_NAMES, ENTITY_LIGHT, mock_devices_response

from tests.common import MockConfigEntry

NoException = nullcontext()


@pytest.mark.parametrize("device_name", ALL_DEVICE_NAMES)
async def test_light_state(
Expand Down Expand Up @@ -49,3 +56,75 @@ async def test_light_state(
# Check states
for entity in entities:
assert hass.states.get(entity.entity_id) == snapshot(name=entity.entity_id)


@pytest.mark.parametrize(
("api_response", "expectation"),
[(False, pytest.raises(HomeAssistantError)), (True, NoException)],
)
async def test_turn_on(
hass: HomeAssistant,
light_config_entry: MockConfigEntry,
api_response: bool,
expectation,
) -> None:
"""Test turn_on method."""

# turn_on returns False indicating failure in which case light.turn_on
# raises HomeAssistantError.
with (
expectation,
patch(
"pyvesync.vesyncswitch.VeSyncDimmerSwitch.turn_on",
return_value=api_response,
) as method_mock,
):
with patch(
"homeassistant.components.vesync.light.VeSyncBaseLightHA.schedule_update_ha_state"
) as update_mock:
await hass.services.async_call(
LIGHT_DOMAIN,
SERVICE_TURN_ON,
{ATTR_ENTITY_ID: ENTITY_LIGHT},
blocking=True,
)

await hass.async_block_till_done()
method_mock.assert_called_once()
update_mock.assert_called_once()


@pytest.mark.parametrize(
("api_response", "expectation"),
[(False, pytest.raises(HomeAssistantError)), (True, NoException)],
)
async def test_turn_off(
hass: HomeAssistant,
light_config_entry: MockConfigEntry,
api_response: bool,
expectation,
) -> None:
"""Test turn_off method."""

# turn_off returns False indicating failure in which case light.turn_off
# raises HomeAssistantError.
with (
expectation,
patch(
"pyvesync.vesyncswitch.VeSyncDimmerSwitch.turn_off",
return_value=api_response,
) as method_mock,
):
with patch(
"homeassistant.components.vesync.light.VeSyncBaseLightHA.schedule_update_ha_state"
) as update_mock:
Comment on lines +61 to +120
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Same comment as with the switch entity

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Planning to make the change here and then update this one it is merged. #137493

await hass.services.async_call(
LIGHT_DOMAIN,
SERVICE_TURN_OFF,
{ATTR_ENTITY_ID: ENTITY_LIGHT},
blocking=True,
)

await hass.async_block_till_done()
method_mock.assert_called_once()
update_mock.assert_called_once()