mirror of
https://github.com/home-assistant/core.git
synced 2026-01-25 09:02:38 +01:00
221 lines
7.5 KiB
Python
221 lines
7.5 KiB
Python
"""Test Velux light entities."""
|
|
|
|
from unittest.mock import AsyncMock
|
|
|
|
import pytest
|
|
|
|
from homeassistant.components.light import (
|
|
ATTR_BRIGHTNESS,
|
|
DOMAIN as LIGHT_DOMAIN,
|
|
SERVICE_TURN_OFF,
|
|
SERVICE_TURN_ON,
|
|
)
|
|
from homeassistant.const import Platform
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.helpers import device_registry as dr, entity_registry as er
|
|
|
|
from . import update_callback_entity
|
|
|
|
from tests.common import MockConfigEntry
|
|
|
|
# Apply setup_integration fixture to all tests in this module
|
|
pytestmark = pytest.mark.usefixtures("setup_integration")
|
|
|
|
|
|
@pytest.fixture
|
|
def platform() -> Platform:
|
|
"""Fixture to specify platform to test."""
|
|
return Platform.LIGHT
|
|
|
|
|
|
async def test_light_setup(
|
|
hass: HomeAssistant,
|
|
entity_registry: er.EntityRegistry,
|
|
device_registry: dr.DeviceRegistry,
|
|
mock_light: AsyncMock,
|
|
) -> None:
|
|
"""Test light entity setup and device association."""
|
|
|
|
test_entity_id = f"light.{mock_light.name.lower().replace(' ', '_')}"
|
|
|
|
# Check that the entity exists and its name matches the node name (the light is the main feature).
|
|
state = hass.states.get(test_entity_id)
|
|
assert state is not None
|
|
assert state.attributes.get("friendly_name") == mock_light.name
|
|
|
|
# Get entity + device entry
|
|
entity_entry = entity_registry.async_get(test_entity_id)
|
|
assert entity_entry is not None
|
|
assert entity_entry.device_id is not None
|
|
device_entry = device_registry.async_get(entity_entry.device_id)
|
|
assert device_entry is not None
|
|
|
|
# Verify device has correct identifiers + name
|
|
assert ("velux", mock_light.serial_number) in device_entry.identifiers
|
|
assert device_entry.name == mock_light.name
|
|
|
|
|
|
# This test is not light specific, it just uses the light platform to test the base entity class.
|
|
async def test_entity_callbacks(
|
|
hass: HomeAssistant,
|
|
mock_config_entry: MockConfigEntry,
|
|
mock_light: AsyncMock,
|
|
) -> None:
|
|
"""Ensure the entity unregisters its device-updated callback when unloaded."""
|
|
# Entity is created by setup_integration; callback should be registered
|
|
test_entity_id = f"light.{mock_light.name.lower().replace(' ', '_')}"
|
|
state = hass.states.get(test_entity_id)
|
|
assert state is not None
|
|
|
|
# Callback is registered exactly once with a callable
|
|
assert mock_light.register_device_updated_cb.call_count == 1
|
|
cb = mock_light.register_device_updated_cb.call_args[0][0]
|
|
assert callable(cb)
|
|
|
|
# Unload the config entry to trigger async_will_remove_from_hass
|
|
assert await hass.config_entries.async_unload(mock_config_entry.entry_id)
|
|
await hass.async_block_till_done()
|
|
|
|
# Callback must be unregistered with the same callable
|
|
assert mock_light.unregister_device_updated_cb.call_count == 1
|
|
assert mock_light.unregister_device_updated_cb.call_args[0][0] is cb
|
|
|
|
|
|
# Test availability functionality by using the light platform
|
|
async def test_entity_availability(
|
|
hass: HomeAssistant, mock_light: AsyncMock, caplog: pytest.LogCaptureFixture
|
|
) -> None:
|
|
"""Test that entity availability updates based on device connection status."""
|
|
|
|
entity_id = f"light.{mock_light.name.lower().replace(' ', '_')}"
|
|
|
|
# Initially connected
|
|
mock_light.pyvlx.get_connected.return_value = True
|
|
await update_callback_entity(hass, mock_light)
|
|
state = hass.states.get(entity_id)
|
|
assert state is not None
|
|
assert state.state != "unavailable"
|
|
|
|
# Simulate disconnection
|
|
mock_light.pyvlx.get_connected.return_value = False
|
|
await update_callback_entity(hass, mock_light)
|
|
state = hass.states.get(entity_id)
|
|
assert state is not None
|
|
assert state.state == "unavailable"
|
|
assert caplog.text.count(f"Entity {entity_id} is unavailable") == 1
|
|
|
|
# Simulate disconnection, check we don't log again
|
|
mock_light.pyvlx.get_connected.return_value = False
|
|
await update_callback_entity(hass, mock_light)
|
|
state = hass.states.get(entity_id)
|
|
assert state is not None
|
|
assert state.state == "unavailable"
|
|
assert caplog.text.count(f"Entity {entity_id} is unavailable") == 1
|
|
|
|
# Simulate reconnection
|
|
mock_light.pyvlx.get_connected.return_value = True
|
|
await update_callback_entity(hass, mock_light)
|
|
state = hass.states.get(entity_id)
|
|
assert state is not None
|
|
assert state.state != "unavailable"
|
|
assert caplog.text.count(f"Entity {entity_id} is back online") == 1
|
|
|
|
# Simulate reconnection, check we don't log again
|
|
mock_light.pyvlx.get_connected.return_value = True
|
|
await update_callback_entity(hass, mock_light)
|
|
state = hass.states.get(entity_id)
|
|
assert state is not None
|
|
assert state.state != "unavailable"
|
|
assert caplog.text.count(f"Entity {entity_id} is back online") == 1
|
|
|
|
|
|
async def test_light_brightness_and_is_on(
|
|
hass: HomeAssistant, mock_light: AsyncMock
|
|
) -> None:
|
|
"""Validate brightness mapping and on/off state from intensity."""
|
|
|
|
entity_id = f"light.{mock_light.name.lower().replace(' ', '_')}"
|
|
|
|
# Set initial intensity values
|
|
mock_light.intensity.intensity_percent = 20 # 20% "intensity" -> 80% brightness
|
|
mock_light.intensity.off = False
|
|
mock_light.intensity.known = True
|
|
|
|
# Trigger state write
|
|
await update_callback_entity(hass, mock_light)
|
|
|
|
state = hass.states.get(entity_id)
|
|
assert state is not None
|
|
# brightness = int((100 - 20) * 255 / 100) = int(204)
|
|
assert state.attributes.get("brightness") == 204
|
|
assert state.state == "on"
|
|
|
|
# Mark as off
|
|
mock_light.intensity.off = True
|
|
await update_callback_entity(hass, mock_light)
|
|
state = hass.states.get(entity_id)
|
|
assert state is not None
|
|
assert state.state == "off"
|
|
|
|
|
|
async def test_light_turn_on_with_brightness_uses_set_intensity(
|
|
hass: HomeAssistant, mock_light: AsyncMock
|
|
) -> None:
|
|
"""Turning on with brightness calls set_intensity with inverted percent."""
|
|
|
|
entity_id = f"light.{mock_light.name.lower().replace(' ', '_')}"
|
|
|
|
# Call turn_on with brightness=51 (20% when normalized)
|
|
await hass.services.async_call(
|
|
LIGHT_DOMAIN,
|
|
SERVICE_TURN_ON,
|
|
{"entity_id": entity_id, ATTR_BRIGHTNESS: 51},
|
|
blocking=True,
|
|
)
|
|
|
|
# set_intensity called once; turn_on should not be used in this path
|
|
assert mock_light.set_intensity.await_count == 1
|
|
assert mock_light.turn_on.await_count == 0
|
|
|
|
# Inspect the intensity argument (first positional)
|
|
args, kwargs = mock_light.set_intensity.await_args
|
|
intensity_obj = args[0]
|
|
# brightness 51 -> 20% normalized -> intensity_percent = 80
|
|
assert intensity_obj.intensity_percent == 80
|
|
assert kwargs.get("wait_for_completion") is True
|
|
|
|
|
|
async def test_light_turn_on_without_brightness_calls_turn_on(
|
|
hass: HomeAssistant, mock_light: AsyncMock
|
|
) -> None:
|
|
"""Turning on without brightness uses device.turn_on."""
|
|
|
|
entity_id = f"light.{mock_light.name.lower().replace(' ', '_')}"
|
|
|
|
await hass.services.async_call(
|
|
LIGHT_DOMAIN,
|
|
SERVICE_TURN_ON,
|
|
{"entity_id": entity_id},
|
|
blocking=True,
|
|
)
|
|
|
|
mock_light.turn_on.assert_awaited_once_with(wait_for_completion=True)
|
|
assert mock_light.set_intensity.await_count == 0
|
|
|
|
|
|
async def test_light_turn_off_calls_turn_off(
|
|
hass: HomeAssistant, mock_light: AsyncMock
|
|
) -> None:
|
|
"""Turning off calls device.turn_off with wait_for_completion."""
|
|
|
|
entity_id = f"light.{mock_light.name.lower().replace(' ', '_')}"
|
|
|
|
await hass.services.async_call(
|
|
LIGHT_DOMAIN,
|
|
SERVICE_TURN_OFF,
|
|
{"entity_id": entity_id},
|
|
blocking=True,
|
|
)
|
|
|
|
mock_light.turn_off.assert_awaited_once_with(wait_for_completion=True)
|