"""Test for the LCN light platform.""" from unittest.mock import patch from freezegun.api import FrozenDateTimeFactory from pypck.inputs import ModStatusOutput, ModStatusRelays from pypck.lcn_addr import LcnAddr from pypck.lcn_defs import RelayStateModifier import pytest from syrupy.assertion import SnapshotAssertion from homeassistant.components.lcn.helpers import get_device_connection from homeassistant.components.lcn.light import SCAN_INTERVAL from homeassistant.components.light import ( ATTR_BRIGHTNESS, ATTR_TRANSITION, DOMAIN as DOMAIN_LIGHT, ) from homeassistant.const import ( ATTR_ENTITY_ID, SERVICE_TURN_OFF, SERVICE_TURN_ON, STATE_OFF, STATE_ON, STATE_UNAVAILABLE, Platform, ) from homeassistant.core import HomeAssistant from homeassistant.helpers import entity_registry as er from .conftest import MockConfigEntry, MockDeviceConnection, init_integration from tests.common import async_fire_time_changed, snapshot_platform LIGHT_OUTPUT1 = "light.testmodule_light_output1" LIGHT_OUTPUT2 = "light.testmodule_light_output2" LIGHT_RELAY1 = "light.testmodule_light_relay1" async def test_setup_lcn_light( hass: HomeAssistant, entity_registry: er.EntityRegistry, entry: MockConfigEntry, snapshot: SnapshotAssertion, ) -> None: """Test the setup of light.""" with patch("homeassistant.components.lcn.PLATFORMS", [Platform.LIGHT]): await init_integration(hass, entry) await snapshot_platform(hass, entity_registry, snapshot, entry.entry_id) async def test_output_turn_on(hass: HomeAssistant, entry: MockConfigEntry) -> None: """Test the output light turns on.""" await init_integration(hass, entry) with patch.object(MockDeviceConnection, "toggle_output") as toggle_output: # command failed toggle_output.return_value = False await hass.services.async_call( DOMAIN_LIGHT, SERVICE_TURN_ON, {ATTR_ENTITY_ID: LIGHT_OUTPUT1}, blocking=True, ) toggle_output.assert_awaited_with(0, 9, to_memory=True) state = hass.states.get(LIGHT_OUTPUT1) assert state is not None assert state.state != STATE_ON # command success toggle_output.reset_mock(return_value=True) toggle_output.return_value = True await hass.services.async_call( DOMAIN_LIGHT, SERVICE_TURN_ON, {ATTR_ENTITY_ID: LIGHT_OUTPUT1}, blocking=True, ) toggle_output.assert_awaited_with(0, 9, to_memory=True) state = hass.states.get(LIGHT_OUTPUT1) assert state is not None assert state.state == STATE_ON async def test_output_turn_on_with_attributes( hass: HomeAssistant, entry: MockConfigEntry ) -> None: """Test the output light turns on.""" await init_integration(hass, entry) with patch.object(MockDeviceConnection, "dim_output") as dim_output: dim_output.return_value = True await hass.services.async_call( DOMAIN_LIGHT, SERVICE_TURN_ON, { ATTR_ENTITY_ID: LIGHT_OUTPUT1, ATTR_BRIGHTNESS: 50, ATTR_TRANSITION: 2, }, blocking=True, ) dim_output.assert_awaited_with(0, 19, 6) state = hass.states.get(LIGHT_OUTPUT1) assert state is not None assert state.state == STATE_ON async def test_output_turn_off(hass: HomeAssistant, entry: MockConfigEntry) -> None: """Test the output light turns off.""" await init_integration(hass, entry) with patch.object(MockDeviceConnection, "toggle_output") as toggle_output: await hass.services.async_call( DOMAIN_LIGHT, SERVICE_TURN_ON, {ATTR_ENTITY_ID: LIGHT_OUTPUT1}, blocking=True, ) # command failed toggle_output.return_value = False await hass.services.async_call( DOMAIN_LIGHT, SERVICE_TURN_OFF, {ATTR_ENTITY_ID: LIGHT_OUTPUT1}, blocking=True, ) toggle_output.assert_awaited_with(0, 9, to_memory=True) state = hass.states.get(LIGHT_OUTPUT1) assert state is not None assert state.state != STATE_OFF # command success toggle_output.reset_mock(return_value=True) toggle_output.return_value = True await hass.services.async_call( DOMAIN_LIGHT, SERVICE_TURN_OFF, {ATTR_ENTITY_ID: LIGHT_OUTPUT1}, blocking=True, ) toggle_output.assert_awaited_with(0, 9, to_memory=True) state = hass.states.get(LIGHT_OUTPUT1) assert state is not None assert state.state == STATE_OFF async def test_relay_turn_on(hass: HomeAssistant, entry: MockConfigEntry) -> None: """Test the relay light turns on.""" await init_integration(hass, entry) with patch.object(MockDeviceConnection, "control_relays") as control_relays: states = [RelayStateModifier.NOCHANGE] * 8 states[0] = RelayStateModifier.ON # command failed control_relays.return_value = False await hass.services.async_call( DOMAIN_LIGHT, SERVICE_TURN_ON, {ATTR_ENTITY_ID: LIGHT_RELAY1}, blocking=True, ) control_relays.assert_awaited_with(states) state = hass.states.get(LIGHT_RELAY1) assert state is not None assert state.state != STATE_ON # command success control_relays.reset_mock(return_value=True) control_relays.return_value = True await hass.services.async_call( DOMAIN_LIGHT, SERVICE_TURN_ON, {ATTR_ENTITY_ID: LIGHT_RELAY1}, blocking=True, ) control_relays.assert_awaited_with(states) state = hass.states.get(LIGHT_RELAY1) assert state is not None assert state.state == STATE_ON async def test_relay_turn_off(hass: HomeAssistant, entry: MockConfigEntry) -> None: """Test the relay light turns off.""" await init_integration(hass, entry) with patch.object(MockDeviceConnection, "control_relays") as control_relays: states = [RelayStateModifier.NOCHANGE] * 8 states[0] = RelayStateModifier.OFF await hass.services.async_call( DOMAIN_LIGHT, SERVICE_TURN_ON, {ATTR_ENTITY_ID: LIGHT_RELAY1}, blocking=True, ) # command failed control_relays.return_value = False await hass.services.async_call( DOMAIN_LIGHT, SERVICE_TURN_OFF, {ATTR_ENTITY_ID: LIGHT_RELAY1}, blocking=True, ) control_relays.assert_awaited_with(states) state = hass.states.get(LIGHT_RELAY1) assert state is not None assert state.state != STATE_OFF # command success control_relays.reset_mock(return_value=True) control_relays.return_value = True await hass.services.async_call( DOMAIN_LIGHT, SERVICE_TURN_OFF, {ATTR_ENTITY_ID: LIGHT_RELAY1}, blocking=True, ) control_relays.assert_awaited_with(states) state = hass.states.get(LIGHT_RELAY1) assert state is not None assert state.state == STATE_OFF async def test_pushed_output_status_change( hass: HomeAssistant, entry: MockConfigEntry ) -> None: """Test the output light changes its state on status received.""" await init_integration(hass, entry) device_connection = get_device_connection(hass, (0, 7, False), entry) address = LcnAddr(0, 7, False) # push status "on" inp = ModStatusOutput(address, 0, 50) await device_connection.async_process_input(inp) await hass.async_block_till_done() state = hass.states.get(LIGHT_OUTPUT1) assert state is not None assert state.state == STATE_ON assert state.attributes[ATTR_BRIGHTNESS] == 128 # push status "off" inp = ModStatusOutput(address, 0, 0) await device_connection.async_process_input(inp) await hass.async_block_till_done() state = hass.states.get(LIGHT_OUTPUT1) assert state is not None assert state.state == STATE_OFF async def test_pushed_relay_status_change( hass: HomeAssistant, entry: MockConfigEntry ) -> None: """Test the relay light changes its state on status received.""" await init_integration(hass, entry) device_connection = get_device_connection(hass, (0, 7, False), entry) address = LcnAddr(0, 7, False) states = [False] * 8 # push status "on" states[0] = True inp = ModStatusRelays(address, states) await device_connection.async_process_input(inp) await hass.async_block_till_done() state = hass.states.get(LIGHT_RELAY1) assert state is not None assert state.state == STATE_ON # push status "off" states[0] = False inp = ModStatusRelays(address, states) await device_connection.async_process_input(inp) await hass.async_block_till_done() state = hass.states.get(LIGHT_RELAY1) assert state is not None assert state.state == STATE_OFF @pytest.mark.parametrize( ("entity_id", "request_method", "return_value"), [ ( LIGHT_OUTPUT1, "request_status_output", ModStatusOutput(LcnAddr(0, 7, False), 0, 0), ), ( LIGHT_RELAY1, "request_status_relays", ModStatusRelays(LcnAddr(0, 7, False), [False] * 8), ), ], ) async def test_availability( hass: HomeAssistant, freezer: FrozenDateTimeFactory, entry: MockConfigEntry, entity_id: str, request_method: str, return_value: ModStatusOutput | ModStatusRelays, ) -> None: """Test the availability of light entity.""" await init_integration(hass, entry) state = hass.states.get(entity_id) assert state is not None assert state.state != STATE_UNAVAILABLE # no response from device -> unavailable with patch.object(MockDeviceConnection, request_method, return_value=None): freezer.tick(SCAN_INTERVAL) async_fire_time_changed(hass) await hass.async_block_till_done(wait_background_tasks=True) state = hass.states.get(entity_id) assert state is not None assert state.state == STATE_UNAVAILABLE # response from device -> available with patch.object( MockDeviceConnection, request_method, return_value=return_value, ): freezer.tick(SCAN_INTERVAL) async_fire_time_changed(hass) await hass.async_block_till_done(wait_background_tasks=True) state = hass.states.get(entity_id) assert state is not None assert state.state != STATE_UNAVAILABLE async def test_unload_config_entry(hass: HomeAssistant, entry: MockConfigEntry) -> None: """Test the light is removed when the config entry is unloaded.""" await init_integration(hass, entry) await hass.config_entries.async_unload(entry.entry_id) assert hass.states.get(LIGHT_OUTPUT1).state == STATE_UNAVAILABLE