"""Test initialization of tedee.""" from http import HTTPStatus from typing import Any from unittest.mock import MagicMock, patch from urllib.parse import urlparse from pytedee_async.exception import ( TedeeAuthException, TedeeClientException, TedeeWebhookException, ) import pytest from syrupy import SnapshotAssertion from homeassistant.components.tedee.const import CONF_LOCAL_ACCESS_TOKEN, DOMAIN from homeassistant.components.webhook import async_generate_url from homeassistant.config_entries import ConfigEntryState from homeassistant.const import CONF_HOST, CONF_WEBHOOK_ID, EVENT_HOMEASSISTANT_STOP from homeassistant.core import HomeAssistant from homeassistant.helpers import device_registry as dr from .conftest import WEBHOOK_ID from tests.common import MockConfigEntry from tests.typing import ClientSessionGenerator async def test_load_unload_config_entry( hass: HomeAssistant, mock_config_entry: MockConfigEntry, mock_tedee: MagicMock, ) -> None: """Test loading and unloading the integration.""" mock_config_entry.add_to_hass(hass) await hass.config_entries.async_setup(mock_config_entry.entry_id) await hass.async_block_till_done() assert mock_config_entry.state is ConfigEntryState.LOADED await hass.config_entries.async_unload(mock_config_entry.entry_id) await hass.async_block_till_done() assert mock_config_entry.state is ConfigEntryState.NOT_LOADED @pytest.mark.parametrize( "side_effect", [TedeeClientException(""), TedeeAuthException("")] ) async def test_config_entry_not_ready( hass: HomeAssistant, mock_config_entry: MockConfigEntry, mock_tedee: MagicMock, side_effect: Exception, ) -> None: """Test the Tedee configuration entry not ready.""" mock_tedee.get_locks.side_effect = side_effect mock_config_entry.add_to_hass(hass) await hass.config_entries.async_setup(mock_config_entry.entry_id) await hass.async_block_till_done() assert len(mock_tedee.get_locks.mock_calls) == 1 assert mock_config_entry.state is ConfigEntryState.SETUP_RETRY async def test_cleanup_on_shutdown( hass: HomeAssistant, mock_config_entry: MockConfigEntry, mock_tedee: MagicMock, ) -> None: """Test the webhook is cleaned up on shutdown.""" mock_config_entry.add_to_hass(hass) await hass.config_entries.async_setup(mock_config_entry.entry_id) await hass.async_block_till_done() assert mock_config_entry.state is ConfigEntryState.LOADED hass.bus.async_fire(EVENT_HOMEASSISTANT_STOP) await hass.async_block_till_done() mock_tedee.delete_webhook.assert_called_once() async def test_webhook_cleanup_errors( hass: HomeAssistant, mock_config_entry: MockConfigEntry, mock_tedee: MagicMock, caplog: pytest.LogCaptureFixture, ) -> None: """Test the webhook is cleaned up on shutdown.""" mock_config_entry.add_to_hass(hass) await hass.config_entries.async_setup(mock_config_entry.entry_id) await hass.async_block_till_done() assert mock_config_entry.state is ConfigEntryState.LOADED mock_tedee.delete_webhook.side_effect = TedeeWebhookException("") hass.bus.async_fire(EVENT_HOMEASSISTANT_STOP) await hass.async_block_till_done() mock_tedee.delete_webhook.assert_called_once() assert "Failed to unregister Tedee webhook from bridge" in caplog.text async def test_webhook_registration_errors( hass: HomeAssistant, mock_config_entry: MockConfigEntry, mock_tedee: MagicMock, caplog: pytest.LogCaptureFixture, ) -> None: """Test the webhook is cleaned up on shutdown.""" mock_tedee.register_webhook.side_effect = TedeeWebhookException("") mock_config_entry.add_to_hass(hass) await hass.config_entries.async_setup(mock_config_entry.entry_id) await hass.async_block_till_done() assert mock_config_entry.state is ConfigEntryState.LOADED mock_tedee.register_webhook.assert_called_once() assert "Failed to register Tedee webhook from bridge" in caplog.text async def test_webhook_registration_cleanup_errors( hass: HomeAssistant, mock_config_entry: MockConfigEntry, mock_tedee: MagicMock, caplog: pytest.LogCaptureFixture, ) -> None: """Test the errors during webhook cleanup during registration.""" mock_tedee.cleanup_webhooks_by_host.side_effect = TedeeWebhookException("") mock_config_entry.add_to_hass(hass) await hass.config_entries.async_setup(mock_config_entry.entry_id) await hass.async_block_till_done() assert mock_config_entry.state is ConfigEntryState.LOADED mock_tedee.cleanup_webhooks_by_host.assert_called_once() assert "Failed to cleanup Tedee webhooks by host:" in caplog.text async def test_bridge_device( hass: HomeAssistant, mock_config_entry: MockConfigEntry, mock_tedee: MagicMock, device_registry: dr.DeviceRegistry, snapshot: SnapshotAssertion, ) -> None: """Ensure the bridge device is registered.""" mock_config_entry.add_to_hass(hass) await hass.config_entries.async_setup(mock_config_entry.entry_id) await hass.async_block_till_done() device = device_registry.async_get_device( {(mock_config_entry.domain, mock_tedee.get_local_bridge.return_value.serial)} ) assert device assert device == snapshot @pytest.mark.parametrize( ( "body", "expected_code", "side_effect", ), [ ( {"hello": "world"}, HTTPStatus.OK, None, ), # Success ( None, HTTPStatus.BAD_REQUEST, None, ), # Missing data ( {}, HTTPStatus.BAD_REQUEST, TedeeWebhookException, ), # Error ], ) async def test_webhook_post( hass: HomeAssistant, mock_config_entry: MockConfigEntry, mock_tedee: MagicMock, hass_client_no_auth: ClientSessionGenerator, body: dict[str, Any], expected_code: HTTPStatus, side_effect: Exception, ) -> None: """Test webhook callback.""" mock_config_entry.add_to_hass(hass) await hass.config_entries.async_setup(mock_config_entry.entry_id) await hass.async_block_till_done() client = await hass_client_no_auth() webhook_url = async_generate_url(hass, WEBHOOK_ID) mock_tedee.parse_webhook_message.side_effect = side_effect resp = await client.post(urlparse(webhook_url).path, json=body) # Wait for remaining tasks to complete. await hass.async_block_till_done() assert resp.status == expected_code async def test_config_flow_entry_migrate_2_1(hass: HomeAssistant) -> None: """Test that config entry fails setup if the version is from the future.""" entry = MockConfigEntry( domain=DOMAIN, version=2, minor_version=1, ) entry.add_to_hass(hass) assert not await hass.config_entries.async_setup(entry.entry_id) async def test_migration( hass: HomeAssistant, mock_tedee: MagicMock, ) -> None: """Test migration of the config entry.""" mock_config_entry = MockConfigEntry( title="My Tedee", domain=DOMAIN, data={ CONF_LOCAL_ACCESS_TOKEN: "api_token", CONF_HOST: "192.168.1.42", }, version=1, minor_version=1, unique_id="0000-0000", ) with patch( "homeassistant.components.tedee.webhook_generate_id", return_value=WEBHOOK_ID, ): mock_config_entry.add_to_hass(hass) await hass.config_entries.async_setup(mock_config_entry.entry_id) await hass.async_block_till_done() assert mock_config_entry.version == 1 assert mock_config_entry.minor_version == 2 assert mock_config_entry.data[CONF_WEBHOOK_ID] == WEBHOOK_ID assert mock_config_entry.state is ConfigEntryState.LOADED