1
0
mirror of https://github.com/home-assistant/core.git synced 2026-04-02 00:20:30 +01:00
Files
core/tests/components/intellifire/test_init.py
2026-03-10 21:11:57 +01:00

345 lines
13 KiB
Python

"""Test the IntelliFire config flow."""
from unittest.mock import AsyncMock, patch
from intellifire4py.const import IntelliFireApiMode
from homeassistant.components.intellifire import CONF_USER_ID
from homeassistant.components.intellifire.const import (
API_MODE_CLOUD,
API_MODE_LOCAL,
CONF_AUTH_COOKIE,
CONF_CONTROL_MODE,
CONF_READ_MODE,
CONF_SERIAL,
CONF_WEB_CLIENT_ID,
DOMAIN,
)
from homeassistant.config_entries import ConfigEntryState
from homeassistant.const import (
CONF_API_KEY,
CONF_HOST,
CONF_IP_ADDRESS,
CONF_PASSWORD,
CONF_USERNAME,
)
from homeassistant.core import HomeAssistant
from tests.common import MockConfigEntry
async def test_migration_v1_1_to_v1_3(
hass: HomeAssistant, mock_config_entry_old, mock_apis_single_fp
) -> None:
"""Test migration from v1.1 to v1.3."""
mock_config_entry_old.add_to_hass(hass)
await hass.config_entries.async_setup(mock_config_entry_old.entry_id)
# Verify the migration updated to v1.3
assert mock_config_entry_old.minor_version == 3
assert mock_config_entry_old.data == {
"ip_address": "192.168.2.108",
"host": "192.168.2.108",
"api_key": "B5C4DA27AAEF31D1FB21AFF9BFA6BCD2",
"serial": "3FB284769E4736F30C8973A7ED358123",
"auth_cookie": "B984F21A6378560019F8A1CDE41B6782",
"web_client_id": "FA2B1C3045601234D0AE17D72F8E975",
"user_id": "52C3F9E8B9D3AC99F8E4D12345678901FE9A2BC7D85F7654E28BF98BCD123456",
"username": "grumpypanda@china.cn",
"password": "you-stole-my-pandas",
}
# Verify options were set with new keys
assert mock_config_entry_old.options == {
CONF_READ_MODE: API_MODE_LOCAL,
CONF_CONTROL_MODE: API_MODE_LOCAL,
}
async def test_migration_v1_1_error(hass: HomeAssistant, mock_apis_single_fp) -> None:
"""Test migration failure when cloud lookup fails."""
mock_config_entry = MockConfigEntry(
domain=DOMAIN,
version=1,
minor_version=1,
title="Fireplace of testing",
data={
CONF_HOST: "11.168.2.218",
CONF_USERNAME: "grumpypanda@china.cn",
CONF_PASSWORD: "you-stole-my-pandas",
CONF_USER_ID: "52C3F9E8B9D3AC99F8E4D12345678901FE9A2BC7D85F7654E28BF98BCD123456",
},
)
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.MIGRATION_ERROR
async def test_migration_v1_2_to_v1_3(
hass: HomeAssistant, mock_config_entry_v1_2_old_options, mock_apis_single_fp
) -> None:
"""Test migration from v1.2 with old option keys to v1.3 with new keys."""
mock_config_entry_v1_2_old_options.add_to_hass(hass)
await hass.config_entries.async_setup(mock_config_entry_v1_2_old_options.entry_id)
await hass.async_block_till_done()
# Verify the migration updated the minor version
assert mock_config_entry_v1_2_old_options.minor_version == 3
# Verify the old option keys were migrated to new keys
# Old: {"cloud_read": "cloud", "cloud_control": "local"}
# New: {"read_mode": "cloud", "control_mode": "local"}
assert mock_config_entry_v1_2_old_options.options == {
CONF_READ_MODE: "cloud",
CONF_CONTROL_MODE: "local",
}
async def test_migration_v1_2_to_v1_3_defaults(
hass: HomeAssistant, mock_apis_single_fp
) -> None:
"""Test migration from v1.2 with no options defaults to local."""
mock_config_entry = MockConfigEntry(
domain=DOMAIN,
version=1,
minor_version=2,
data={
CONF_IP_ADDRESS: "192.168.2.108",
CONF_USERNAME: "grumpypanda@china.cn",
CONF_PASSWORD: "you-stole-my-pandas",
CONF_SERIAL: "3FB284769E4736F30C8973A7ED358123",
CONF_WEB_CLIENT_ID: "FA2B1C3045601234D0AE17D72F8E975",
CONF_API_KEY: "B5C4DA27AAEF31D1FB21AFF9BFA6BCD2",
CONF_AUTH_COOKIE: "B984F21A6378560019F8A1CDE41B6782",
CONF_USER_ID: "52C3F9E8B9D3AC99F8E4D12345678901FE9A2BC7D85F7654E28BF98BCD123456",
},
options={},
unique_id="3FB284769E4736F30C8973A7ED358123",
)
mock_config_entry.add_to_hass(hass)
await hass.config_entries.async_setup(mock_config_entry.entry_id)
await hass.async_block_till_done()
# Verify the migration updated the minor version
assert mock_config_entry.minor_version == 3
# Verify defaults were applied
assert mock_config_entry.options == {
CONF_READ_MODE: API_MODE_LOCAL,
CONF_CONTROL_MODE: API_MODE_LOCAL,
}
async def test_init_with_no_username(hass: HomeAssistant, mock_apis_single_fp) -> None:
"""Test the case where we completely fail to initialize."""
mock_config_entry = MockConfigEntry(
domain=DOMAIN,
version=1,
minor_version=2,
data={
CONF_IP_ADDRESS: "192.168.2.108",
CONF_PASSWORD: "you-stole-my-pandas",
CONF_SERIAL: "3FB284769E4736F30C8973A7ED358123",
CONF_WEB_CLIENT_ID: "FA2B1C3045601234D0AE17D72F8E975",
CONF_API_KEY: "B5C4DA27AAEF31D1FB21AFF9BFA6BCD2",
CONF_AUTH_COOKIE: "B984F21A6378560019F8A1CDE41B6782",
CONF_USER_ID: "52C3F9E8B9D3AC99F8E4D12345678901FE9A2BC7D85F7654E28BF98BCD123456",
},
options={CONF_READ_MODE: API_MODE_LOCAL, CONF_CONTROL_MODE: API_MODE_CLOUD},
unique_id="3FB284769E4736F30C8973A7ED358123",
)
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.SETUP_ERROR
async def test_connectivity_bad(
hass: HomeAssistant,
mock_config_entry_current,
mock_apis_single_fp,
) -> None:
"""Test a timeout error on the setup flow."""
with patch(
"homeassistant.components.intellifire.UnifiedFireplace.build_fireplace_from_common",
new_callable=AsyncMock,
side_effect=TimeoutError,
):
mock_config_entry_current.add_to_hass(hass)
await hass.config_entries.async_setup(mock_config_entry_current.entry_id)
await hass.async_block_till_done()
assert len(hass.states.async_all()) == 0
async def test_update_options_change_read_mode_only(
hass: HomeAssistant,
mock_config_entry_current: MockConfigEntry,
mock_apis_single_fp,
) -> None:
"""Test that changing only read mode triggers set_read_mode but not set_control_mode."""
_mock_local, _mock_cloud, mock_fp = mock_apis_single_fp
mock_config_entry_current.add_to_hass(hass)
await hass.config_entries.async_setup(mock_config_entry_current.entry_id)
await hass.async_block_till_done()
# Get the coordinator and mock async_request_refresh
coordinator = mock_config_entry_current.runtime_data
coordinator.async_request_refresh = AsyncMock()
# Reset mock call counts
mock_fp.set_read_mode.reset_mock()
mock_fp.set_control_mode.reset_mock()
# Change only read mode (local -> cloud), keep control mode same
hass.config_entries.async_update_entry(
mock_config_entry_current,
options={CONF_READ_MODE: API_MODE_CLOUD, CONF_CONTROL_MODE: API_MODE_LOCAL},
)
await hass.async_block_till_done()
# Only set_read_mode should be called
mock_fp.set_read_mode.assert_called_once()
mock_fp.set_control_mode.assert_not_called()
# async_request_refresh should always be called
coordinator.async_request_refresh.assert_called_once()
async def test_update_options_change_control_mode_only(
hass: HomeAssistant,
mock_config_entry_current: MockConfigEntry,
mock_apis_single_fp,
) -> None:
"""Test that changing only control mode triggers set_control_mode but not set_read_mode."""
_mock_local, _mock_cloud, mock_fp = mock_apis_single_fp
mock_config_entry_current.add_to_hass(hass)
await hass.config_entries.async_setup(mock_config_entry_current.entry_id)
await hass.async_block_till_done()
# Get the coordinator and mock async_request_refresh
coordinator = mock_config_entry_current.runtime_data
coordinator.async_request_refresh = AsyncMock()
# Reset mock call counts
mock_fp.set_read_mode.reset_mock()
mock_fp.set_control_mode.reset_mock()
# Change only control mode (local -> cloud), keep read mode same
hass.config_entries.async_update_entry(
mock_config_entry_current,
options={CONF_READ_MODE: API_MODE_LOCAL, CONF_CONTROL_MODE: API_MODE_CLOUD},
)
await hass.async_block_till_done()
# Only set_control_mode should be called
mock_fp.set_read_mode.assert_not_called()
mock_fp.set_control_mode.assert_called_once()
# async_request_refresh should always be called
coordinator.async_request_refresh.assert_called_once()
async def test_update_options_change_both_modes(
hass: HomeAssistant,
mock_config_entry_current: MockConfigEntry,
mock_apis_single_fp,
) -> None:
"""Test that changing both modes triggers both set methods."""
_mock_local, _mock_cloud, mock_fp = mock_apis_single_fp
mock_config_entry_current.add_to_hass(hass)
await hass.config_entries.async_setup(mock_config_entry_current.entry_id)
await hass.async_block_till_done()
# Get the coordinator and mock async_request_refresh
coordinator = mock_config_entry_current.runtime_data
coordinator.async_request_refresh = AsyncMock()
# Reset mock call counts
mock_fp.set_read_mode.reset_mock()
mock_fp.set_control_mode.reset_mock()
# Change both modes
hass.config_entries.async_update_entry(
mock_config_entry_current,
options={CONF_READ_MODE: API_MODE_CLOUD, CONF_CONTROL_MODE: API_MODE_CLOUD},
)
await hass.async_block_till_done()
# Both should be called
mock_fp.set_read_mode.assert_called_once()
mock_fp.set_control_mode.assert_called_once()
# async_request_refresh should always be called
coordinator.async_request_refresh.assert_called_once()
async def test_update_options_no_change(
hass: HomeAssistant,
mock_config_entry_current: MockConfigEntry,
mock_apis_single_fp,
) -> None:
"""Test that no mode change triggers neither set method but refresh is still called."""
_mock_local, _mock_cloud, mock_fp = mock_apis_single_fp
mock_config_entry_current.add_to_hass(hass)
await hass.config_entries.async_setup(mock_config_entry_current.entry_id)
await hass.async_block_till_done()
# Get the coordinator and mock async_request_refresh
coordinator = mock_config_entry_current.runtime_data
coordinator.async_request_refresh = AsyncMock()
# Reset mock call counts
mock_fp.set_read_mode.reset_mock()
mock_fp.set_control_mode.reset_mock()
# First change options to CLOUD/CLOUD to trigger listener
hass.config_entries.async_update_entry(
mock_config_entry_current,
options={CONF_READ_MODE: API_MODE_CLOUD, CONF_CONTROL_MODE: API_MODE_CLOUD},
)
await hass.async_block_till_done()
# Simulate that the fireplace updated its modes after the first change
# This makes the next update a true "no change" scenario
mock_fp.read_mode = IntelliFireApiMode.CLOUD
mock_fp.control_mode = IntelliFireApiMode.CLOUD
# Reset mocks after the first change
mock_fp.set_read_mode.reset_mock()
mock_fp.set_control_mode.reset_mock()
coordinator.async_request_refresh.reset_mock()
# Now update options to LOCAL/LOCAL - listener fires but fireplace modes
# were set to CLOUD/CLOUD, so this IS a mode change
# Instead, we update to the same CLOUD/CLOUD that the fireplace now has
# But wait - HA won't fire listener if options didn't change!
# To properly test "no mode change triggers neither setter":
# Change options to something different from current options (so listener fires)
# but the fireplace already has the target modes
# Set fireplace to LOCAL/LOCAL (matching what we'll update to)
mock_fp.read_mode = IntelliFireApiMode.LOCAL
mock_fp.control_mode = IntelliFireApiMode.LOCAL
# Update options back to LOCAL/LOCAL - listener fires because options changed
# from CLOUD/CLOUD, but fireplace already has LOCAL/LOCAL modes
hass.config_entries.async_update_entry(
mock_config_entry_current,
options={CONF_READ_MODE: API_MODE_LOCAL, CONF_CONTROL_MODE: API_MODE_LOCAL},
)
await hass.async_block_till_done()
# Neither set method should be called since new options match fireplace state
mock_fp.set_read_mode.assert_not_called()
mock_fp.set_control_mode.assert_not_called()
# But async_request_refresh should still be called
coordinator.async_request_refresh.assert_called_once()