262 lines
7.3 KiB
Python
262 lines
7.3 KiB
Python
"""Tests for init module."""
|
|
|
|
import http
|
|
import time
|
|
from unittest.mock import MagicMock
|
|
|
|
from aiohttp import ClientConnectionError
|
|
import pytest
|
|
from syrupy import SnapshotAssertion
|
|
|
|
from homeassistant.components.myuplink.const import DOMAIN, OAUTH2_TOKEN
|
|
from homeassistant.config_entries import ConfigEntryState
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.helpers import device_registry as dr
|
|
from homeassistant.setup import async_setup_component
|
|
|
|
from . import setup_integration
|
|
from .const import UNIQUE_ID
|
|
|
|
from tests.common import MockConfigEntry, load_fixture
|
|
from tests.test_util.aiohttp import AiohttpClientMocker
|
|
from tests.typing import WebSocketGenerator
|
|
|
|
|
|
async def test_load_unload_entry(
|
|
hass: HomeAssistant,
|
|
mock_myuplink_client: MagicMock,
|
|
mock_config_entry: MockConfigEntry,
|
|
) -> None:
|
|
"""Test load and unload entry."""
|
|
await setup_integration(hass, mock_config_entry)
|
|
entry = hass.config_entries.async_entries(DOMAIN)[0]
|
|
|
|
assert entry.state is ConfigEntryState.LOADED
|
|
|
|
await hass.config_entries.async_remove(entry.entry_id)
|
|
await hass.async_block_till_done()
|
|
|
|
assert entry.state is ConfigEntryState.NOT_LOADED
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
("expires_at", "status", "expected_state"),
|
|
[
|
|
(
|
|
time.time() - 3600,
|
|
http.HTTPStatus.UNAUTHORIZED,
|
|
ConfigEntryState.SETUP_ERROR,
|
|
),
|
|
(
|
|
time.time() - 3600,
|
|
http.HTTPStatus.INTERNAL_SERVER_ERROR,
|
|
ConfigEntryState.SETUP_RETRY,
|
|
),
|
|
],
|
|
ids=["unauthorized", "internal_server_error"],
|
|
)
|
|
async def test_expired_token_refresh_failure(
|
|
hass: HomeAssistant,
|
|
mock_config_entry: MockConfigEntry,
|
|
aioclient_mock: AiohttpClientMocker,
|
|
status: http.HTTPStatus,
|
|
expected_state: ConfigEntryState,
|
|
) -> None:
|
|
"""Test failure while refreshing token with a transient error."""
|
|
|
|
aioclient_mock.clear_requests()
|
|
aioclient_mock.post(
|
|
OAUTH2_TOKEN,
|
|
status=status,
|
|
)
|
|
|
|
await setup_integration(hass, mock_config_entry)
|
|
|
|
assert mock_config_entry.state is expected_state
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
("expires_at", "expected_state"),
|
|
[
|
|
(
|
|
time.time() - 3600,
|
|
ConfigEntryState.SETUP_RETRY,
|
|
),
|
|
],
|
|
ids=[
|
|
"client_connection_error",
|
|
],
|
|
)
|
|
async def test_expired_token_refresh_connection_failure(
|
|
hass: HomeAssistant,
|
|
mock_config_entry: MockConfigEntry,
|
|
aioclient_mock: AiohttpClientMocker,
|
|
expected_state: ConfigEntryState,
|
|
) -> None:
|
|
"""Test failure while refreshing token with a ClientError."""
|
|
|
|
aioclient_mock.clear_requests()
|
|
aioclient_mock.post(
|
|
OAUTH2_TOKEN,
|
|
exc=ClientConnectionError(),
|
|
)
|
|
|
|
await setup_integration(hass, mock_config_entry)
|
|
|
|
assert mock_config_entry.state is expected_state
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"load_systems_file",
|
|
[load_fixture("systems.json", DOMAIN)],
|
|
)
|
|
async def test_devices_created_count(
|
|
hass: HomeAssistant,
|
|
device_registry: dr.DeviceRegistry,
|
|
mock_myuplink_client: MagicMock,
|
|
mock_config_entry: MockConfigEntry,
|
|
) -> None:
|
|
"""Test that one device is created."""
|
|
await setup_integration(hass, mock_config_entry)
|
|
|
|
assert len(device_registry.devices) == 1
|
|
|
|
|
|
async def test_devices_multiple_created_count(
|
|
hass: HomeAssistant,
|
|
device_registry: dr.DeviceRegistry,
|
|
mock_myuplink_client: MagicMock,
|
|
mock_config_entry: MockConfigEntry,
|
|
) -> None:
|
|
"""Test that multiple devices are created."""
|
|
await setup_integration(hass, mock_config_entry)
|
|
|
|
assert len(device_registry.devices) == 2
|
|
|
|
|
|
async def test_migrate_config_entry(
|
|
hass: HomeAssistant,
|
|
mock_config_entry: MockConfigEntry,
|
|
mock_myuplink_client: MagicMock,
|
|
expires_at: float,
|
|
access_token: str,
|
|
) -> None:
|
|
"""Test migration of config entry."""
|
|
mock_entry_v1_1 = MockConfigEntry(
|
|
version=1,
|
|
minor_version=1,
|
|
domain=DOMAIN,
|
|
title="myUplink test",
|
|
data={
|
|
"auth_implementation": DOMAIN,
|
|
"token": {
|
|
"access_token": access_token,
|
|
"scope": "WRITESYSTEM READSYSTEM offline_access",
|
|
"expires_in": 86399,
|
|
"refresh_token": "3012bc9f-7a65-4240-b817-9154ffdcc30f",
|
|
"token_type": "Bearer",
|
|
"expires_at": expires_at,
|
|
},
|
|
},
|
|
entry_id="myuplink_test",
|
|
)
|
|
|
|
await setup_integration(hass, mock_entry_v1_1)
|
|
assert mock_entry_v1_1.version == 1
|
|
assert mock_entry_v1_1.minor_version == 2
|
|
assert mock_entry_v1_1.unique_id == UNIQUE_ID
|
|
|
|
|
|
async def test_oaut2_scope_failure(
|
|
hass: HomeAssistant,
|
|
mock_myuplink_client: MagicMock,
|
|
mock_config_entry: MockConfigEntry,
|
|
) -> None:
|
|
"""Test that an incorrect OAuth2 scope fails."""
|
|
|
|
mock_config_entry.data["token"]["scope"] = "wrong_scope"
|
|
await setup_integration(hass, mock_config_entry)
|
|
|
|
assert mock_config_entry.state is ConfigEntryState.SETUP_ERROR
|
|
|
|
|
|
async def test_device_remove_devices(
|
|
hass: HomeAssistant,
|
|
hass_ws_client: WebSocketGenerator,
|
|
mock_config_entry: MockConfigEntry,
|
|
mock_myuplink_client: MagicMock,
|
|
device_registry: dr.DeviceRegistry,
|
|
) -> None:
|
|
"""Test we can only remove a device that no longer exists."""
|
|
assert await async_setup_component(hass, "config", {})
|
|
|
|
mock_config_entry.add_to_hass(hass)
|
|
|
|
assert await hass.config_entries.async_setup(mock_config_entry.entry_id)
|
|
await hass.async_block_till_done()
|
|
|
|
device_entry = device_registry.async_get_device(
|
|
identifiers={
|
|
(
|
|
DOMAIN,
|
|
"batman-r-1234-20240201-123456-aa-bb-cc-dd-ee-ff",
|
|
)
|
|
},
|
|
)
|
|
client = await hass_ws_client(hass)
|
|
response = await client.remove_device(device_entry.id, mock_config_entry.entry_id)
|
|
assert not response["success"]
|
|
|
|
old_device_entry = device_registry.async_get_or_create(
|
|
config_entry_id=mock_config_entry.entry_id,
|
|
identifiers={(DOMAIN, "OLD-DEVICE-UUID")},
|
|
)
|
|
response = await client.remove_device(
|
|
old_device_entry.id, mock_config_entry.entry_id
|
|
)
|
|
assert response["success"]
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"load_systems_file",
|
|
[load_fixture("systems-multi.json", DOMAIN)],
|
|
ids=[
|
|
"multi",
|
|
],
|
|
)
|
|
@pytest.mark.parametrize(
|
|
("load_device_file", "device_id"),
|
|
[
|
|
(
|
|
load_fixture("device-alfred.json", DOMAIN),
|
|
"alfred-r-1234-20240201-123456-aa-bb-cc-dd-ee-ff",
|
|
),
|
|
(
|
|
load_fixture("device-batman.json", DOMAIN),
|
|
"batman-r-1234-20240201-123456-aa-bb-cc-dd-ee-ff",
|
|
),
|
|
(
|
|
load_fixture("device-robin.json", DOMAIN),
|
|
"robin-r-1234-20240201-123456-aa-bb-cc-dd-ee-ff",
|
|
),
|
|
],
|
|
ids=[
|
|
"alfred",
|
|
"batman",
|
|
"robin",
|
|
],
|
|
)
|
|
async def test_device_info(
|
|
hass: HomeAssistant,
|
|
snapshot: SnapshotAssertion,
|
|
mock_myuplink_client: MagicMock,
|
|
mock_config_entry: MockConfigEntry,
|
|
device_registry: dr.DeviceRegistry,
|
|
device_id: str,
|
|
) -> None:
|
|
"""Test device registry integration."""
|
|
await setup_integration(hass, mock_config_entry)
|
|
device_entry = device_registry.async_get_device(identifiers={(DOMAIN, device_id)})
|
|
assert device_entry is not None
|
|
assert device_entry == snapshot
|