172 lines
5.5 KiB
Python
172 lines
5.5 KiB
Python
"""Test initialization of tedee."""
|
|
from http import HTTPStatus
|
|
from typing import Any
|
|
from unittest.mock import MagicMock
|
|
from urllib.parse import urlparse
|
|
|
|
from pytedee_async.exception import (
|
|
TedeeAuthException,
|
|
TedeeClientException,
|
|
TedeeWebhookException,
|
|
)
|
|
import pytest
|
|
from syrupy import SnapshotAssertion
|
|
|
|
from homeassistant.components.webhook import async_generate_url
|
|
from homeassistant.config_entries import ConfigEntryState
|
|
from homeassistant.const import 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_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
|