2968 lines
103 KiB
Python
2968 lines
103 KiB
Python
"""Test the config manager."""
|
|
import asyncio
|
|
from datetime import timedelta
|
|
import logging
|
|
from unittest.mock import AsyncMock, Mock, patch
|
|
|
|
import pytest
|
|
|
|
from homeassistant import config_entries, data_entry_flow, loader
|
|
from homeassistant.const import EVENT_HOMEASSISTANT_STARTED, EVENT_HOMEASSISTANT_STOP
|
|
from homeassistant.core import CoreState, callback
|
|
from homeassistant.data_entry_flow import RESULT_TYPE_ABORT
|
|
from homeassistant.exceptions import (
|
|
ConfigEntryAuthFailed,
|
|
ConfigEntryNotReady,
|
|
HomeAssistantError,
|
|
)
|
|
from homeassistant.helpers import entity_registry as er
|
|
from homeassistant.helpers.update_coordinator import DataUpdateCoordinator
|
|
from homeassistant.setup import async_setup_component
|
|
from homeassistant.util import dt
|
|
|
|
from tests.common import (
|
|
MockConfigEntry,
|
|
MockEntity,
|
|
MockModule,
|
|
MockPlatform,
|
|
async_fire_time_changed,
|
|
mock_coro,
|
|
mock_entity_platform,
|
|
mock_integration,
|
|
mock_registry,
|
|
)
|
|
|
|
|
|
@pytest.fixture(autouse=True)
|
|
def mock_handlers():
|
|
"""Mock config flows."""
|
|
|
|
class MockFlowHandler(config_entries.ConfigFlow):
|
|
"""Define a mock flow handler."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_reauth(self, data):
|
|
"""Mock Reauth."""
|
|
return self.async_show_form(step_id="reauth")
|
|
|
|
with patch.dict(
|
|
config_entries.HANDLERS, {"comp": MockFlowHandler, "test": MockFlowHandler}
|
|
):
|
|
yield
|
|
|
|
|
|
@pytest.fixture
|
|
def manager(hass):
|
|
"""Fixture of a loaded config manager."""
|
|
manager = config_entries.ConfigEntries(hass, {})
|
|
manager._entries = {}
|
|
manager._store._async_ensure_stop_listener = lambda: None
|
|
hass.config_entries = manager
|
|
return manager
|
|
|
|
|
|
async def test_call_setup_entry(hass):
|
|
"""Test we call <component>.setup_entry."""
|
|
entry = MockConfigEntry(domain="comp")
|
|
entry.add_to_hass(hass)
|
|
assert not entry.supports_unload
|
|
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
mock_migrate_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"comp",
|
|
async_setup_entry=mock_setup_entry,
|
|
async_migrate_entry=mock_migrate_entry,
|
|
),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
with patch("homeassistant.config_entries.support_entry_unload", return_value=True):
|
|
result = await async_setup_component(hass, "comp", {})
|
|
await hass.async_block_till_done()
|
|
assert result
|
|
assert len(mock_migrate_entry.mock_calls) == 0
|
|
assert len(mock_setup_entry.mock_calls) == 1
|
|
assert entry.state is config_entries.ConfigEntryState.LOADED
|
|
assert entry.supports_unload
|
|
|
|
|
|
async def test_call_setup_entry_without_reload_support(hass):
|
|
"""Test we call <component>.setup_entry and the <component> does not support unloading."""
|
|
entry = MockConfigEntry(domain="comp")
|
|
entry.add_to_hass(hass)
|
|
assert not entry.supports_unload
|
|
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
mock_migrate_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"comp",
|
|
async_setup_entry=mock_setup_entry,
|
|
async_migrate_entry=mock_migrate_entry,
|
|
),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
with patch("homeassistant.config_entries.support_entry_unload", return_value=False):
|
|
result = await async_setup_component(hass, "comp", {})
|
|
await hass.async_block_till_done()
|
|
assert result
|
|
assert len(mock_migrate_entry.mock_calls) == 0
|
|
assert len(mock_setup_entry.mock_calls) == 1
|
|
assert entry.state is config_entries.ConfigEntryState.LOADED
|
|
assert not entry.supports_unload
|
|
|
|
|
|
async def test_call_async_migrate_entry(hass):
|
|
"""Test we call <component>.async_migrate_entry when version mismatch."""
|
|
entry = MockConfigEntry(domain="comp")
|
|
assert not entry.supports_unload
|
|
entry.version = 2
|
|
entry.add_to_hass(hass)
|
|
|
|
mock_migrate_entry = AsyncMock(return_value=True)
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"comp",
|
|
async_setup_entry=mock_setup_entry,
|
|
async_migrate_entry=mock_migrate_entry,
|
|
),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
with patch("homeassistant.config_entries.support_entry_unload", return_value=True):
|
|
result = await async_setup_component(hass, "comp", {})
|
|
await hass.async_block_till_done()
|
|
assert result
|
|
assert len(mock_migrate_entry.mock_calls) == 1
|
|
assert len(mock_setup_entry.mock_calls) == 1
|
|
assert entry.state is config_entries.ConfigEntryState.LOADED
|
|
assert entry.supports_unload
|
|
|
|
|
|
async def test_call_async_migrate_entry_failure_false(hass):
|
|
"""Test migration fails if returns false."""
|
|
entry = MockConfigEntry(domain="comp")
|
|
entry.version = 2
|
|
entry.add_to_hass(hass)
|
|
assert not entry.supports_unload
|
|
|
|
mock_migrate_entry = AsyncMock(return_value=False)
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"comp",
|
|
async_setup_entry=mock_setup_entry,
|
|
async_migrate_entry=mock_migrate_entry,
|
|
),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
result = await async_setup_component(hass, "comp", {})
|
|
assert result
|
|
assert len(mock_migrate_entry.mock_calls) == 1
|
|
assert len(mock_setup_entry.mock_calls) == 0
|
|
assert entry.state is config_entries.ConfigEntryState.MIGRATION_ERROR
|
|
assert not entry.supports_unload
|
|
|
|
|
|
async def test_call_async_migrate_entry_failure_exception(hass):
|
|
"""Test migration fails if exception raised."""
|
|
entry = MockConfigEntry(domain="comp")
|
|
entry.version = 2
|
|
entry.add_to_hass(hass)
|
|
assert not entry.supports_unload
|
|
|
|
mock_migrate_entry = AsyncMock(side_effect=Exception)
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"comp",
|
|
async_setup_entry=mock_setup_entry,
|
|
async_migrate_entry=mock_migrate_entry,
|
|
),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
result = await async_setup_component(hass, "comp", {})
|
|
assert result
|
|
assert len(mock_migrate_entry.mock_calls) == 1
|
|
assert len(mock_setup_entry.mock_calls) == 0
|
|
assert entry.state is config_entries.ConfigEntryState.MIGRATION_ERROR
|
|
assert not entry.supports_unload
|
|
|
|
|
|
async def test_call_async_migrate_entry_failure_not_bool(hass):
|
|
"""Test migration fails if boolean not returned."""
|
|
entry = MockConfigEntry(domain="comp")
|
|
entry.version = 2
|
|
entry.add_to_hass(hass)
|
|
assert not entry.supports_unload
|
|
|
|
mock_migrate_entry = AsyncMock(return_value=None)
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"comp",
|
|
async_setup_entry=mock_setup_entry,
|
|
async_migrate_entry=mock_migrate_entry,
|
|
),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
result = await async_setup_component(hass, "comp", {})
|
|
assert result
|
|
assert len(mock_migrate_entry.mock_calls) == 1
|
|
assert len(mock_setup_entry.mock_calls) == 0
|
|
assert entry.state is config_entries.ConfigEntryState.MIGRATION_ERROR
|
|
assert not entry.supports_unload
|
|
|
|
|
|
async def test_call_async_migrate_entry_failure_not_supported(hass):
|
|
"""Test migration fails if async_migrate_entry not implemented."""
|
|
entry = MockConfigEntry(domain="comp")
|
|
entry.version = 2
|
|
entry.add_to_hass(hass)
|
|
assert not entry.supports_unload
|
|
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(hass, MockModule("comp", async_setup_entry=mock_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
result = await async_setup_component(hass, "comp", {})
|
|
assert result
|
|
assert len(mock_setup_entry.mock_calls) == 0
|
|
assert entry.state is config_entries.ConfigEntryState.MIGRATION_ERROR
|
|
assert not entry.supports_unload
|
|
|
|
|
|
async def test_remove_entry(hass, manager):
|
|
"""Test that we can remove an entry."""
|
|
|
|
async def mock_setup_entry(hass, entry):
|
|
"""Mock setting up entry."""
|
|
hass.config_entries.async_setup_platforms(entry, ["light"])
|
|
return True
|
|
|
|
async def mock_unload_entry(hass, entry):
|
|
"""Mock unloading an entry."""
|
|
result = await hass.config_entries.async_unload_platforms(entry, ["light"])
|
|
assert result
|
|
return result
|
|
|
|
mock_remove_entry = AsyncMock(return_value=None)
|
|
|
|
entity = MockEntity(unique_id="1234", name="Test Entity")
|
|
|
|
async def mock_setup_entry_platform(hass, entry, async_add_entities):
|
|
"""Mock setting up platform."""
|
|
async_add_entities([entity])
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"test",
|
|
async_setup_entry=mock_setup_entry,
|
|
async_unload_entry=mock_unload_entry,
|
|
async_remove_entry=mock_remove_entry,
|
|
),
|
|
)
|
|
mock_entity_platform(
|
|
hass, "light.test", MockPlatform(async_setup_entry=mock_setup_entry_platform)
|
|
)
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
|
|
|
MockConfigEntry(domain="test_other", entry_id="test1").add_to_manager(manager)
|
|
entry = MockConfigEntry(domain="test", entry_id="test2")
|
|
entry.add_to_manager(manager)
|
|
MockConfigEntry(domain="test_other", entry_id="test3").add_to_manager(manager)
|
|
|
|
# Check all config entries exist
|
|
assert [item.entry_id for item in manager.async_entries()] == [
|
|
"test1",
|
|
"test2",
|
|
"test3",
|
|
]
|
|
|
|
# Setup entry
|
|
await entry.async_setup(hass)
|
|
await hass.async_block_till_done()
|
|
|
|
# Check entity state got added
|
|
assert hass.states.get("light.test_entity") is not None
|
|
assert len(hass.states.async_all()) == 1
|
|
|
|
# Check entity got added to entity registry
|
|
ent_reg = er.async_get(hass)
|
|
assert len(ent_reg.entities) == 1
|
|
entity_entry = list(ent_reg.entities.values())[0]
|
|
assert entity_entry.config_entry_id == entry.entry_id
|
|
|
|
# Remove entry
|
|
result = await manager.async_remove("test2")
|
|
await hass.async_block_till_done()
|
|
|
|
# Check that unload went well and so no need to restart
|
|
assert result == {"require_restart": False}
|
|
|
|
# Check the remove callback was invoked.
|
|
assert mock_remove_entry.call_count == 1
|
|
|
|
# Check that config entry was removed.
|
|
assert [item.entry_id for item in manager.async_entries()] == ["test1", "test3"]
|
|
|
|
# Check that entity state has been removed
|
|
assert hass.states.get("light.test_entity") is None
|
|
assert len(hass.states.async_all()) == 0
|
|
|
|
# Check that entity registry entry has been removed
|
|
entity_entry_list = list(ent_reg.entities.values())
|
|
assert not entity_entry_list
|
|
|
|
|
|
async def test_remove_entry_cancels_reauth(hass, manager):
|
|
"""Tests that removing a config entry, also aborts existing reauth flows."""
|
|
entry = MockConfigEntry(title="test_title", domain="test")
|
|
|
|
mock_setup_entry = AsyncMock(side_effect=ConfigEntryAuthFailed())
|
|
mock_integration(hass, MockModule("test", async_setup_entry=mock_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
|
|
|
entry.add_to_hass(hass)
|
|
await entry.async_setup(hass)
|
|
await hass.async_block_till_done()
|
|
|
|
flows = hass.config_entries.flow.async_progress()
|
|
assert len(flows) == 1
|
|
assert flows[0]["context"]["entry_id"] == entry.entry_id
|
|
assert flows[0]["context"]["source"] == config_entries.SOURCE_REAUTH
|
|
assert entry.state is config_entries.ConfigEntryState.SETUP_ERROR
|
|
|
|
await manager.async_remove(entry.entry_id)
|
|
|
|
flows = hass.config_entries.flow.async_progress()
|
|
assert len(flows) == 0
|
|
|
|
|
|
async def test_remove_entry_handles_callback_error(hass, manager):
|
|
"""Test that exceptions in the remove callback are handled."""
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
mock_unload_entry = AsyncMock(return_value=True)
|
|
mock_remove_entry = AsyncMock(return_value=None)
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"test",
|
|
async_setup_entry=mock_setup_entry,
|
|
async_unload_entry=mock_unload_entry,
|
|
async_remove_entry=mock_remove_entry,
|
|
),
|
|
)
|
|
entry = MockConfigEntry(domain="test", entry_id="test1")
|
|
entry.add_to_manager(manager)
|
|
# Check all config entries exist
|
|
assert [item.entry_id for item in manager.async_entries()] == ["test1"]
|
|
# Setup entry
|
|
await entry.async_setup(hass)
|
|
await hass.async_block_till_done()
|
|
|
|
# Remove entry
|
|
result = await manager.async_remove("test1")
|
|
await hass.async_block_till_done()
|
|
# Check that unload went well and so no need to restart
|
|
assert result == {"require_restart": False}
|
|
# Check the remove callback was invoked.
|
|
assert mock_remove_entry.call_count == 1
|
|
# Check that config entry was removed.
|
|
assert [item.entry_id for item in manager.async_entries()] == []
|
|
|
|
|
|
async def test_remove_entry_raises(hass, manager):
|
|
"""Test if a component raises while removing entry."""
|
|
|
|
async def mock_unload_entry(hass, entry):
|
|
"""Mock unload entry function."""
|
|
raise Exception("BROKEN")
|
|
|
|
mock_integration(hass, MockModule("comp", async_unload_entry=mock_unload_entry))
|
|
|
|
MockConfigEntry(domain="test", entry_id="test1").add_to_manager(manager)
|
|
MockConfigEntry(
|
|
domain="comp", entry_id="test2", state=config_entries.ConfigEntryState.LOADED
|
|
).add_to_manager(manager)
|
|
MockConfigEntry(domain="test", entry_id="test3").add_to_manager(manager)
|
|
|
|
assert [item.entry_id for item in manager.async_entries()] == [
|
|
"test1",
|
|
"test2",
|
|
"test3",
|
|
]
|
|
|
|
result = await manager.async_remove("test2")
|
|
|
|
assert result == {"require_restart": True}
|
|
assert [item.entry_id for item in manager.async_entries()] == ["test1", "test3"]
|
|
|
|
|
|
async def test_remove_entry_if_not_loaded(hass, manager):
|
|
"""Test that we can remove an entry that is not loaded."""
|
|
mock_unload_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(hass, MockModule("comp", async_unload_entry=mock_unload_entry))
|
|
|
|
MockConfigEntry(domain="test", entry_id="test1").add_to_manager(manager)
|
|
MockConfigEntry(domain="comp", entry_id="test2").add_to_manager(manager)
|
|
MockConfigEntry(domain="test", entry_id="test3").add_to_manager(manager)
|
|
|
|
assert [item.entry_id for item in manager.async_entries()] == [
|
|
"test1",
|
|
"test2",
|
|
"test3",
|
|
]
|
|
|
|
result = await manager.async_remove("test2")
|
|
|
|
assert result == {"require_restart": False}
|
|
assert [item.entry_id for item in manager.async_entries()] == ["test1", "test3"]
|
|
|
|
assert len(mock_unload_entry.mock_calls) == 0
|
|
|
|
|
|
async def test_remove_entry_if_integration_deleted(hass, manager):
|
|
"""Test that we can remove an entry when the integration is deleted."""
|
|
mock_unload_entry = AsyncMock(return_value=True)
|
|
|
|
MockConfigEntry(domain="test", entry_id="test1").add_to_manager(manager)
|
|
MockConfigEntry(domain="comp", entry_id="test2").add_to_manager(manager)
|
|
MockConfigEntry(domain="test", entry_id="test3").add_to_manager(manager)
|
|
|
|
assert [item.entry_id for item in manager.async_entries()] == [
|
|
"test1",
|
|
"test2",
|
|
"test3",
|
|
]
|
|
|
|
result = await manager.async_remove("test2")
|
|
|
|
assert result == {"require_restart": False}
|
|
assert [item.entry_id for item in manager.async_entries()] == ["test1", "test3"]
|
|
|
|
assert len(mock_unload_entry.mock_calls) == 0
|
|
|
|
|
|
async def test_add_entry_calls_setup_entry(hass, manager):
|
|
"""Test we call setup_config_entry."""
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(hass, MockModule("comp", async_setup_entry=mock_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Test user step."""
|
|
return self.async_create_entry(title="title", data={"token": "supersecret"})
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow, "beer": 5}):
|
|
await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert len(mock_setup_entry.mock_calls) == 1
|
|
p_hass, p_entry = mock_setup_entry.mock_calls[0][1]
|
|
|
|
assert p_hass is hass
|
|
assert p_entry.data == {"token": "supersecret"}
|
|
|
|
|
|
async def test_entries_gets_entries(manager):
|
|
"""Test entries are filtered by domain."""
|
|
MockConfigEntry(domain="test").add_to_manager(manager)
|
|
entry1 = MockConfigEntry(domain="test2")
|
|
entry1.add_to_manager(manager)
|
|
entry2 = MockConfigEntry(domain="test2")
|
|
entry2.add_to_manager(manager)
|
|
|
|
assert manager.async_entries("test2") == [entry1, entry2]
|
|
|
|
|
|
async def test_domains_gets_domains_uniques(manager):
|
|
"""Test we only return each domain once."""
|
|
MockConfigEntry(domain="test").add_to_manager(manager)
|
|
MockConfigEntry(domain="test2").add_to_manager(manager)
|
|
MockConfigEntry(domain="test2").add_to_manager(manager)
|
|
MockConfigEntry(domain="test").add_to_manager(manager)
|
|
MockConfigEntry(domain="test3").add_to_manager(manager)
|
|
|
|
assert manager.async_domains() == ["test", "test2", "test3"]
|
|
|
|
|
|
async def test_domains_gets_domains_excludes_ignore_and_disabled(manager):
|
|
"""Test we only return each domain once."""
|
|
MockConfigEntry(domain="test").add_to_manager(manager)
|
|
MockConfigEntry(domain="test2").add_to_manager(manager)
|
|
MockConfigEntry(domain="test2").add_to_manager(manager)
|
|
MockConfigEntry(
|
|
domain="ignored", source=config_entries.SOURCE_IGNORE
|
|
).add_to_manager(manager)
|
|
MockConfigEntry(domain="test3").add_to_manager(manager)
|
|
MockConfigEntry(
|
|
domain="disabled", disabled_by=config_entries.DISABLED_USER
|
|
).add_to_manager(manager)
|
|
assert manager.async_domains() == ["test", "test2", "test3"]
|
|
assert manager.async_domains(include_ignore=False) == ["test", "test2", "test3"]
|
|
assert manager.async_domains(include_disabled=False) == ["test", "test2", "test3"]
|
|
assert manager.async_domains(include_ignore=False, include_disabled=False) == [
|
|
"test",
|
|
"test2",
|
|
"test3",
|
|
]
|
|
|
|
assert manager.async_domains(include_ignore=True) == [
|
|
"test",
|
|
"test2",
|
|
"ignored",
|
|
"test3",
|
|
]
|
|
assert manager.async_domains(include_disabled=True) == [
|
|
"test",
|
|
"test2",
|
|
"test3",
|
|
"disabled",
|
|
]
|
|
assert manager.async_domains(include_ignore=True, include_disabled=True) == [
|
|
"test",
|
|
"test2",
|
|
"ignored",
|
|
"test3",
|
|
"disabled",
|
|
]
|
|
|
|
|
|
async def test_saving_and_loading(hass):
|
|
"""Test that we're saving and loading correctly."""
|
|
mock_integration(
|
|
hass, MockModule("test", async_setup_entry=lambda *args: mock_coro(True))
|
|
)
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 5
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Test user step."""
|
|
await self.async_set_unique_id("unique")
|
|
return self.async_create_entry(title="Test Title", data={"token": "abcd"})
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"test": TestFlow}):
|
|
await hass.config_entries.flow.async_init(
|
|
"test", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
|
|
class Test2Flow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 3
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Test user step."""
|
|
return self.async_create_entry(
|
|
title="Test 2 Title", data={"username": "bla"}
|
|
)
|
|
|
|
with patch("homeassistant.config_entries.HANDLERS.get", return_value=Test2Flow):
|
|
await hass.config_entries.flow.async_init(
|
|
"test", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
|
|
assert len(hass.config_entries.async_entries()) == 2
|
|
entry_1 = hass.config_entries.async_entries()[0]
|
|
|
|
hass.config_entries.async_update_entry(
|
|
entry_1,
|
|
pref_disable_polling=True,
|
|
pref_disable_new_entities=True,
|
|
)
|
|
|
|
# To trigger the call_later
|
|
async_fire_time_changed(hass, dt.utcnow() + timedelta(seconds=1))
|
|
# To execute the save
|
|
await hass.async_block_till_done()
|
|
|
|
# Now load written data in new config manager
|
|
manager = config_entries.ConfigEntries(hass, {})
|
|
await manager.async_initialize()
|
|
|
|
assert len(manager.async_entries()) == 2
|
|
|
|
# Ensure same order
|
|
for orig, loaded in zip(
|
|
hass.config_entries.async_entries(), manager.async_entries()
|
|
):
|
|
assert orig.version == loaded.version
|
|
assert orig.domain == loaded.domain
|
|
assert orig.title == loaded.title
|
|
assert orig.data == loaded.data
|
|
assert orig.source == loaded.source
|
|
assert orig.unique_id == loaded.unique_id
|
|
assert orig.pref_disable_new_entities == loaded.pref_disable_new_entities
|
|
assert orig.pref_disable_polling == loaded.pref_disable_polling
|
|
|
|
|
|
async def test_forward_entry_sets_up_component(hass):
|
|
"""Test we setup the component entry is forwarded to."""
|
|
entry = MockConfigEntry(domain="original")
|
|
|
|
mock_original_setup_entry = AsyncMock(return_value=True)
|
|
mock_integration(
|
|
hass, MockModule("original", async_setup_entry=mock_original_setup_entry)
|
|
)
|
|
|
|
mock_forwarded_setup_entry = AsyncMock(return_value=True)
|
|
mock_integration(
|
|
hass, MockModule("forwarded", async_setup_entry=mock_forwarded_setup_entry)
|
|
)
|
|
|
|
await hass.config_entries.async_forward_entry_setup(entry, "forwarded")
|
|
assert len(mock_original_setup_entry.mock_calls) == 0
|
|
assert len(mock_forwarded_setup_entry.mock_calls) == 1
|
|
|
|
|
|
async def test_forward_entry_does_not_setup_entry_if_setup_fails(hass):
|
|
"""Test we do not set up entry if component setup fails."""
|
|
entry = MockConfigEntry(domain="original")
|
|
|
|
mock_setup = AsyncMock(return_value=False)
|
|
mock_setup_entry = AsyncMock()
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"forwarded", async_setup=mock_setup, async_setup_entry=mock_setup_entry
|
|
),
|
|
)
|
|
|
|
await hass.config_entries.async_forward_entry_setup(entry, "forwarded")
|
|
assert len(mock_setup.mock_calls) == 1
|
|
assert len(mock_setup_entry.mock_calls) == 0
|
|
|
|
|
|
async def test_discovery_notification(hass):
|
|
"""Test that we create/dismiss a notification when source is discovery."""
|
|
mock_integration(hass, MockModule("test"))
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
|
|
|
with patch.dict(config_entries.HANDLERS):
|
|
|
|
class TestFlow(config_entries.ConfigFlow, domain="test"):
|
|
"""Test flow."""
|
|
|
|
VERSION = 5
|
|
|
|
async def async_step_discovery(self, discovery_info):
|
|
"""Test discovery step."""
|
|
return self.async_show_form(step_id="discovery_confirm")
|
|
|
|
async def async_step_discovery_confirm(self, discovery_info):
|
|
"""Test discovery confirm step."""
|
|
return self.async_create_entry(
|
|
title="Test Title", data={"token": "abcd"}
|
|
)
|
|
|
|
# Start first discovery flow to assert that reconfigure notification fires
|
|
flow1 = await hass.config_entries.flow.async_init(
|
|
"test", context={"source": config_entries.SOURCE_DISCOVERY}
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
state = hass.states.get("persistent_notification.config_entry_discovery")
|
|
assert state is not None
|
|
|
|
# Start a second discovery flow so we can finish the first and assert that
|
|
# the discovery notification persists until the second one is complete
|
|
flow2 = await hass.config_entries.flow.async_init(
|
|
"test", context={"source": config_entries.SOURCE_DISCOVERY}
|
|
)
|
|
|
|
flow1 = await hass.config_entries.flow.async_configure(flow1["flow_id"], {})
|
|
assert flow1["type"] == data_entry_flow.RESULT_TYPE_CREATE_ENTRY
|
|
|
|
await hass.async_block_till_done()
|
|
state = hass.states.get("persistent_notification.config_entry_discovery")
|
|
assert state is not None
|
|
|
|
flow2 = await hass.config_entries.flow.async_configure(flow2["flow_id"], {})
|
|
assert flow2["type"] == data_entry_flow.RESULT_TYPE_CREATE_ENTRY
|
|
|
|
await hass.async_block_till_done()
|
|
state = hass.states.get("persistent_notification.config_entry_discovery")
|
|
assert state is None
|
|
|
|
|
|
async def test_reauth_notification(hass):
|
|
"""Test that we create/dismiss a notification when source is reauth."""
|
|
mock_integration(hass, MockModule("test"))
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
|
|
|
with patch.dict(config_entries.HANDLERS):
|
|
|
|
class TestFlow(config_entries.ConfigFlow, domain="test"):
|
|
"""Test flow."""
|
|
|
|
VERSION = 5
|
|
|
|
async def async_step_user(self, user_input):
|
|
"""Test user step."""
|
|
return self.async_show_form(step_id="user_confirm")
|
|
|
|
async def async_step_user_confirm(self, user_input):
|
|
"""Test user confirm step."""
|
|
return self.async_show_form(step_id="user_confirm")
|
|
|
|
async def async_step_reauth(self, user_input):
|
|
"""Test reauth step."""
|
|
return self.async_show_form(step_id="reauth_confirm")
|
|
|
|
async def async_step_reauth_confirm(self, user_input):
|
|
"""Test reauth confirm step."""
|
|
return self.async_abort(reason="test")
|
|
|
|
# Start user flow to assert that reconfigure notification doesn't fire
|
|
await hass.config_entries.flow.async_init(
|
|
"test", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
state = hass.states.get("persistent_notification.config_entry_reconfigure")
|
|
assert state is None
|
|
|
|
# Start first reauth flow to assert that reconfigure notification fires
|
|
flow1 = await hass.config_entries.flow.async_init(
|
|
"test", context={"source": config_entries.SOURCE_REAUTH}
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
state = hass.states.get("persistent_notification.config_entry_reconfigure")
|
|
assert state is not None
|
|
|
|
# Start a second reauth flow so we can finish the first and assert that
|
|
# the reconfigure notification persists until the second one is complete
|
|
flow2 = await hass.config_entries.flow.async_init(
|
|
"test", context={"source": config_entries.SOURCE_REAUTH}
|
|
)
|
|
|
|
flow1 = await hass.config_entries.flow.async_configure(flow1["flow_id"], {})
|
|
assert flow1["type"] == data_entry_flow.RESULT_TYPE_ABORT
|
|
|
|
await hass.async_block_till_done()
|
|
state = hass.states.get("persistent_notification.config_entry_reconfigure")
|
|
assert state is not None
|
|
|
|
flow2 = await hass.config_entries.flow.async_configure(flow2["flow_id"], {})
|
|
assert flow2["type"] == data_entry_flow.RESULT_TYPE_ABORT
|
|
|
|
await hass.async_block_till_done()
|
|
state = hass.states.get("persistent_notification.config_entry_reconfigure")
|
|
assert state is None
|
|
|
|
|
|
async def test_discovery_notification_not_created(hass):
|
|
"""Test that we not create a notification when discovery is aborted."""
|
|
mock_integration(hass, MockModule("test"))
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 5
|
|
|
|
async def async_step_discovery(self, discovery_info):
|
|
"""Test discovery step."""
|
|
return self.async_abort(reason="test")
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"test": TestFlow}):
|
|
await hass.config_entries.flow.async_init(
|
|
"test", context={"source": config_entries.SOURCE_DISCOVERY}
|
|
)
|
|
|
|
await hass.async_block_till_done()
|
|
state = hass.states.get("persistent_notification.config_entry_discovery")
|
|
assert state is None
|
|
|
|
|
|
async def test_loading_default_config(hass):
|
|
"""Test loading the default config."""
|
|
manager = config_entries.ConfigEntries(hass, {})
|
|
|
|
with patch("homeassistant.util.json.open", side_effect=FileNotFoundError):
|
|
await manager.async_initialize()
|
|
|
|
assert len(manager.async_entries()) == 0
|
|
|
|
|
|
async def test_updating_entry_data(manager):
|
|
"""Test that we can update an entry data."""
|
|
entry = MockConfigEntry(
|
|
domain="test",
|
|
data={"first": True},
|
|
state=config_entries.ConfigEntryState.SETUP_ERROR,
|
|
)
|
|
entry.add_to_manager(manager)
|
|
|
|
assert manager.async_update_entry(entry) is False
|
|
assert entry.data == {"first": True}
|
|
|
|
assert manager.async_update_entry(entry, data={"second": True}) is True
|
|
assert entry.data == {"second": True}
|
|
|
|
|
|
async def test_updating_entry_system_options(manager):
|
|
"""Test that we can update an entry data."""
|
|
entry = MockConfigEntry(
|
|
domain="test",
|
|
data={"first": True},
|
|
state=config_entries.ConfigEntryState.SETUP_ERROR,
|
|
pref_disable_new_entities=True,
|
|
)
|
|
entry.add_to_manager(manager)
|
|
|
|
assert entry.pref_disable_new_entities is True
|
|
assert entry.pref_disable_polling is False
|
|
|
|
manager.async_update_entry(
|
|
entry, pref_disable_new_entities=False, pref_disable_polling=True
|
|
)
|
|
|
|
assert entry.pref_disable_new_entities is False
|
|
assert entry.pref_disable_polling is True
|
|
|
|
|
|
async def test_update_entry_options_and_trigger_listener(hass, manager):
|
|
"""Test that we can update entry options and trigger listener."""
|
|
entry = MockConfigEntry(domain="test", options={"first": True})
|
|
entry.add_to_manager(manager)
|
|
|
|
async def update_listener(hass, entry):
|
|
"""Test function."""
|
|
assert entry.options == {"second": True}
|
|
|
|
entry.add_update_listener(update_listener)
|
|
|
|
assert manager.async_update_entry(entry, options={"second": True}) is True
|
|
|
|
assert entry.options == {"second": True}
|
|
|
|
|
|
async def test_setup_raise_not_ready(hass, caplog):
|
|
"""Test a setup raising not ready."""
|
|
entry = MockConfigEntry(title="test_title", domain="test")
|
|
|
|
mock_setup_entry = AsyncMock(
|
|
side_effect=ConfigEntryNotReady("The internet connection is offline")
|
|
)
|
|
mock_integration(hass, MockModule("test", async_setup_entry=mock_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
|
|
|
with patch("homeassistant.helpers.event.async_call_later") as mock_call:
|
|
await entry.async_setup(hass)
|
|
|
|
assert len(mock_call.mock_calls) == 1
|
|
assert (
|
|
"Config entry 'test_title' for test integration not ready yet: The internet connection is offline"
|
|
in caplog.text
|
|
)
|
|
p_hass, p_wait_time, p_setup = mock_call.mock_calls[0][1]
|
|
|
|
assert p_hass is hass
|
|
assert p_wait_time == 5
|
|
assert entry.state is config_entries.ConfigEntryState.SETUP_RETRY
|
|
assert entry.reason == "The internet connection is offline"
|
|
|
|
mock_setup_entry.side_effect = None
|
|
mock_setup_entry.return_value = True
|
|
|
|
await p_setup(None)
|
|
assert entry.state is config_entries.ConfigEntryState.LOADED
|
|
assert entry.reason is None
|
|
|
|
|
|
async def test_setup_raise_not_ready_from_exception(hass, caplog):
|
|
"""Test a setup raising not ready from another exception."""
|
|
entry = MockConfigEntry(title="test_title", domain="test")
|
|
|
|
original_exception = HomeAssistantError("The device dropped the connection")
|
|
config_entry_exception = ConfigEntryNotReady()
|
|
config_entry_exception.__cause__ = original_exception
|
|
|
|
mock_setup_entry = AsyncMock(side_effect=config_entry_exception)
|
|
mock_integration(hass, MockModule("test", async_setup_entry=mock_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
|
|
|
with patch("homeassistant.helpers.event.async_call_later") as mock_call:
|
|
await entry.async_setup(hass)
|
|
|
|
assert len(mock_call.mock_calls) == 1
|
|
assert (
|
|
"Config entry 'test_title' for test integration not ready yet: The device dropped the connection"
|
|
in caplog.text
|
|
)
|
|
|
|
|
|
async def test_setup_retrying_during_unload(hass):
|
|
"""Test if we unload an entry that is in retry mode."""
|
|
entry = MockConfigEntry(domain="test")
|
|
|
|
mock_setup_entry = AsyncMock(side_effect=ConfigEntryNotReady)
|
|
mock_integration(hass, MockModule("test", async_setup_entry=mock_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
|
|
|
with patch("homeassistant.helpers.event.async_call_later") as mock_call:
|
|
await entry.async_setup(hass)
|
|
|
|
assert entry.state is config_entries.ConfigEntryState.SETUP_RETRY
|
|
assert len(mock_call.return_value.mock_calls) == 0
|
|
|
|
await entry.async_unload(hass)
|
|
|
|
assert entry.state is config_entries.ConfigEntryState.NOT_LOADED
|
|
assert len(mock_call.return_value.mock_calls) == 1
|
|
|
|
|
|
async def test_setup_retrying_during_unload_before_started(hass):
|
|
"""Test if we unload an entry that is in retry mode before started."""
|
|
entry = MockConfigEntry(domain="test")
|
|
hass.state = CoreState.starting
|
|
initial_listeners = hass.bus.async_listeners()[EVENT_HOMEASSISTANT_STARTED]
|
|
|
|
mock_setup_entry = AsyncMock(side_effect=ConfigEntryNotReady)
|
|
mock_integration(hass, MockModule("test", async_setup_entry=mock_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
|
|
|
await entry.async_setup(hass)
|
|
await hass.async_block_till_done()
|
|
|
|
assert entry.state is config_entries.ConfigEntryState.SETUP_RETRY
|
|
assert (
|
|
hass.bus.async_listeners()[EVENT_HOMEASSISTANT_STARTED] == initial_listeners + 1
|
|
)
|
|
|
|
await entry.async_unload(hass)
|
|
await hass.async_block_till_done()
|
|
|
|
assert entry.state is config_entries.ConfigEntryState.NOT_LOADED
|
|
assert (
|
|
hass.bus.async_listeners()[EVENT_HOMEASSISTANT_STARTED] == initial_listeners + 0
|
|
)
|
|
|
|
|
|
async def test_create_entry_options(hass):
|
|
"""Test a config entry being created with options."""
|
|
|
|
async def mock_async_setup(hass, config):
|
|
"""Mock setup."""
|
|
hass.async_create_task(
|
|
hass.config_entries.flow.async_init(
|
|
"comp",
|
|
context={"source": config_entries.SOURCE_IMPORT},
|
|
data={"data": "data", "option": "option"},
|
|
)
|
|
)
|
|
return True
|
|
|
|
async_setup_entry = AsyncMock(return_value=True)
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"comp", async_setup=mock_async_setup, async_setup_entry=async_setup_entry
|
|
),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_import(self, user_input):
|
|
"""Test import step creating entry, with options."""
|
|
return self.async_create_entry(
|
|
title="title",
|
|
data={"example": user_input["data"]},
|
|
options={"example": user_input["option"]},
|
|
)
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
assert await async_setup_component(hass, "comp", {})
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
assert len(async_setup_entry.mock_calls) == 1
|
|
|
|
entries = hass.config_entries.async_entries("comp")
|
|
assert len(entries) == 1
|
|
assert entries[0].data == {"example": "data"}
|
|
assert entries[0].options == {"example": "option"}
|
|
|
|
|
|
async def test_entry_options(hass, manager):
|
|
"""Test that we can set options on an entry."""
|
|
entry = MockConfigEntry(domain="test", data={"first": True}, options=None)
|
|
entry.add_to_manager(manager)
|
|
|
|
class TestFlow:
|
|
"""Test flow."""
|
|
|
|
@staticmethod
|
|
@callback
|
|
def async_get_options_flow(config_entry):
|
|
"""Test options flow."""
|
|
|
|
class OptionsFlowHandler(data_entry_flow.FlowHandler):
|
|
"""Test options flow handler."""
|
|
|
|
return OptionsFlowHandler()
|
|
|
|
config_entries.HANDLERS["test"] = TestFlow()
|
|
flow = await manager.options.async_create_flow(
|
|
entry.entry_id, context={"source": "test"}, data=None
|
|
)
|
|
|
|
flow.handler = entry.entry_id # Used to keep reference to config entry
|
|
|
|
await manager.options.async_finish_flow(
|
|
flow,
|
|
{"data": {"second": True}, "type": data_entry_flow.RESULT_TYPE_CREATE_ENTRY},
|
|
)
|
|
|
|
assert entry.data == {"first": True}
|
|
assert entry.options == {"second": True}
|
|
|
|
|
|
async def test_entry_options_abort(hass, manager):
|
|
"""Test that we can abort options flow."""
|
|
entry = MockConfigEntry(domain="test", data={"first": True}, options=None)
|
|
entry.add_to_manager(manager)
|
|
|
|
class TestFlow:
|
|
"""Test flow."""
|
|
|
|
@staticmethod
|
|
@callback
|
|
def async_get_options_flow(config_entry):
|
|
"""Test options flow."""
|
|
|
|
class OptionsFlowHandler(data_entry_flow.FlowHandler):
|
|
"""Test options flow handler."""
|
|
|
|
return OptionsFlowHandler()
|
|
|
|
config_entries.HANDLERS["test"] = TestFlow()
|
|
flow = await manager.options.async_create_flow(
|
|
entry.entry_id, context={"source": "test"}, data=None
|
|
)
|
|
|
|
flow.handler = entry.entry_id # Used to keep reference to config entry
|
|
|
|
assert await manager.options.async_finish_flow(
|
|
flow, {"type": data_entry_flow.RESULT_TYPE_ABORT, "reason": "test"}
|
|
)
|
|
|
|
|
|
async def test_entry_setup_succeed(hass, manager):
|
|
"""Test that we can setup an entry."""
|
|
entry = MockConfigEntry(
|
|
domain="comp", state=config_entries.ConfigEntryState.NOT_LOADED
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
mock_setup = AsyncMock(return_value=True)
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule("comp", async_setup=mock_setup, async_setup_entry=mock_setup_entry),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
assert await manager.async_setup(entry.entry_id)
|
|
assert len(mock_setup.mock_calls) == 1
|
|
assert len(mock_setup_entry.mock_calls) == 1
|
|
assert entry.state is config_entries.ConfigEntryState.LOADED
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"state",
|
|
(
|
|
config_entries.ConfigEntryState.LOADED,
|
|
config_entries.ConfigEntryState.SETUP_ERROR,
|
|
config_entries.ConfigEntryState.MIGRATION_ERROR,
|
|
config_entries.ConfigEntryState.SETUP_RETRY,
|
|
config_entries.ConfigEntryState.FAILED_UNLOAD,
|
|
),
|
|
)
|
|
async def test_entry_setup_invalid_state(hass, manager, state):
|
|
"""Test that we cannot setup an entry with invalid state."""
|
|
entry = MockConfigEntry(domain="comp", state=state)
|
|
entry.add_to_hass(hass)
|
|
|
|
mock_setup = AsyncMock(return_value=True)
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule("comp", async_setup=mock_setup, async_setup_entry=mock_setup_entry),
|
|
)
|
|
|
|
with pytest.raises(config_entries.OperationNotAllowed):
|
|
assert await manager.async_setup(entry.entry_id)
|
|
|
|
assert len(mock_setup.mock_calls) == 0
|
|
assert len(mock_setup_entry.mock_calls) == 0
|
|
assert entry.state is state
|
|
|
|
|
|
async def test_entry_unload_succeed(hass, manager):
|
|
"""Test that we can unload an entry."""
|
|
entry = MockConfigEntry(domain="comp", state=config_entries.ConfigEntryState.LOADED)
|
|
entry.add_to_hass(hass)
|
|
|
|
async_unload_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(hass, MockModule("comp", async_unload_entry=async_unload_entry))
|
|
|
|
assert await manager.async_unload(entry.entry_id)
|
|
assert len(async_unload_entry.mock_calls) == 1
|
|
assert entry.state is config_entries.ConfigEntryState.NOT_LOADED
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"state",
|
|
(
|
|
config_entries.ConfigEntryState.NOT_LOADED,
|
|
config_entries.ConfigEntryState.SETUP_ERROR,
|
|
config_entries.ConfigEntryState.SETUP_RETRY,
|
|
),
|
|
)
|
|
async def test_entry_unload_failed_to_load(hass, manager, state):
|
|
"""Test that we can unload an entry."""
|
|
entry = MockConfigEntry(domain="comp", state=state)
|
|
entry.add_to_hass(hass)
|
|
|
|
async_unload_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(hass, MockModule("comp", async_unload_entry=async_unload_entry))
|
|
|
|
assert await manager.async_unload(entry.entry_id)
|
|
assert len(async_unload_entry.mock_calls) == 0
|
|
assert entry.state is config_entries.ConfigEntryState.NOT_LOADED
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"state",
|
|
(
|
|
config_entries.ConfigEntryState.MIGRATION_ERROR,
|
|
config_entries.ConfigEntryState.FAILED_UNLOAD,
|
|
),
|
|
)
|
|
async def test_entry_unload_invalid_state(hass, manager, state):
|
|
"""Test that we cannot unload an entry with invalid state."""
|
|
entry = MockConfigEntry(domain="comp", state=state)
|
|
entry.add_to_hass(hass)
|
|
|
|
async_unload_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(hass, MockModule("comp", async_unload_entry=async_unload_entry))
|
|
|
|
with pytest.raises(config_entries.OperationNotAllowed):
|
|
assert await manager.async_unload(entry.entry_id)
|
|
|
|
assert len(async_unload_entry.mock_calls) == 0
|
|
assert entry.state is state
|
|
|
|
|
|
async def test_entry_reload_succeed(hass, manager):
|
|
"""Test that we can reload an entry."""
|
|
entry = MockConfigEntry(domain="comp", state=config_entries.ConfigEntryState.LOADED)
|
|
entry.add_to_hass(hass)
|
|
|
|
async_setup = AsyncMock(return_value=True)
|
|
async_setup_entry = AsyncMock(return_value=True)
|
|
async_unload_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"comp",
|
|
async_setup=async_setup,
|
|
async_setup_entry=async_setup_entry,
|
|
async_unload_entry=async_unload_entry,
|
|
),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
assert await manager.async_reload(entry.entry_id)
|
|
assert len(async_unload_entry.mock_calls) == 1
|
|
assert len(async_setup.mock_calls) == 1
|
|
assert len(async_setup_entry.mock_calls) == 1
|
|
assert entry.state is config_entries.ConfigEntryState.LOADED
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"state",
|
|
(
|
|
config_entries.ConfigEntryState.NOT_LOADED,
|
|
config_entries.ConfigEntryState.SETUP_ERROR,
|
|
config_entries.ConfigEntryState.SETUP_RETRY,
|
|
),
|
|
)
|
|
async def test_entry_reload_not_loaded(hass, manager, state):
|
|
"""Test that we can reload an entry."""
|
|
entry = MockConfigEntry(domain="comp", state=state)
|
|
entry.add_to_hass(hass)
|
|
|
|
async_setup = AsyncMock(return_value=True)
|
|
async_setup_entry = AsyncMock(return_value=True)
|
|
async_unload_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"comp",
|
|
async_setup=async_setup,
|
|
async_setup_entry=async_setup_entry,
|
|
async_unload_entry=async_unload_entry,
|
|
),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
assert await manager.async_reload(entry.entry_id)
|
|
assert len(async_unload_entry.mock_calls) == 0
|
|
assert len(async_setup.mock_calls) == 1
|
|
assert len(async_setup_entry.mock_calls) == 1
|
|
assert entry.state is config_entries.ConfigEntryState.LOADED
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"state",
|
|
(
|
|
config_entries.ConfigEntryState.MIGRATION_ERROR,
|
|
config_entries.ConfigEntryState.FAILED_UNLOAD,
|
|
),
|
|
)
|
|
async def test_entry_reload_error(hass, manager, state):
|
|
"""Test that we can reload an entry."""
|
|
entry = MockConfigEntry(domain="comp", state=state)
|
|
entry.add_to_hass(hass)
|
|
|
|
async_setup = AsyncMock(return_value=True)
|
|
async_setup_entry = AsyncMock(return_value=True)
|
|
async_unload_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"comp",
|
|
async_setup=async_setup,
|
|
async_setup_entry=async_setup_entry,
|
|
async_unload_entry=async_unload_entry,
|
|
),
|
|
)
|
|
|
|
with pytest.raises(config_entries.OperationNotAllowed):
|
|
assert await manager.async_reload(entry.entry_id)
|
|
|
|
assert len(async_unload_entry.mock_calls) == 0
|
|
assert len(async_setup.mock_calls) == 0
|
|
assert len(async_setup_entry.mock_calls) == 0
|
|
|
|
assert entry.state == state
|
|
|
|
|
|
async def test_entry_disable_succeed(hass, manager):
|
|
"""Test that we can disable an entry."""
|
|
entry = MockConfigEntry(domain="comp", state=config_entries.ConfigEntryState.LOADED)
|
|
entry.add_to_hass(hass)
|
|
|
|
async_setup = AsyncMock(return_value=True)
|
|
async_setup_entry = AsyncMock(return_value=True)
|
|
async_unload_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"comp",
|
|
async_setup=async_setup,
|
|
async_setup_entry=async_setup_entry,
|
|
async_unload_entry=async_unload_entry,
|
|
),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
# Disable
|
|
assert await manager.async_set_disabled_by(
|
|
entry.entry_id, config_entries.DISABLED_USER
|
|
)
|
|
assert len(async_unload_entry.mock_calls) == 1
|
|
assert len(async_setup.mock_calls) == 0
|
|
assert len(async_setup_entry.mock_calls) == 0
|
|
assert entry.state is config_entries.ConfigEntryState.NOT_LOADED
|
|
|
|
# Enable
|
|
assert await manager.async_set_disabled_by(entry.entry_id, None)
|
|
assert len(async_unload_entry.mock_calls) == 1
|
|
assert len(async_setup.mock_calls) == 1
|
|
assert len(async_setup_entry.mock_calls) == 1
|
|
assert entry.state is config_entries.ConfigEntryState.LOADED
|
|
|
|
|
|
async def test_entry_disable_without_reload_support(hass, manager):
|
|
"""Test that we can disable an entry without reload support."""
|
|
entry = MockConfigEntry(domain="comp", state=config_entries.ConfigEntryState.LOADED)
|
|
entry.add_to_hass(hass)
|
|
|
|
async_setup = AsyncMock(return_value=True)
|
|
async_setup_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"comp",
|
|
async_setup=async_setup,
|
|
async_setup_entry=async_setup_entry,
|
|
),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
# Disable
|
|
assert not await manager.async_set_disabled_by(
|
|
entry.entry_id, config_entries.DISABLED_USER
|
|
)
|
|
assert len(async_setup.mock_calls) == 0
|
|
assert len(async_setup_entry.mock_calls) == 0
|
|
assert entry.state is config_entries.ConfigEntryState.FAILED_UNLOAD
|
|
|
|
# Enable
|
|
with pytest.raises(config_entries.OperationNotAllowed):
|
|
await manager.async_set_disabled_by(entry.entry_id, None)
|
|
assert len(async_setup.mock_calls) == 0
|
|
assert len(async_setup_entry.mock_calls) == 0
|
|
assert entry.state is config_entries.ConfigEntryState.FAILED_UNLOAD
|
|
|
|
|
|
async def test_entry_enable_without_reload_support(hass, manager):
|
|
"""Test that we can disable an entry without reload support."""
|
|
entry = MockConfigEntry(domain="comp", disabled_by=config_entries.DISABLED_USER)
|
|
entry.add_to_hass(hass)
|
|
|
|
async_setup = AsyncMock(return_value=True)
|
|
async_setup_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"comp",
|
|
async_setup=async_setup,
|
|
async_setup_entry=async_setup_entry,
|
|
),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
# Enable
|
|
assert await manager.async_set_disabled_by(entry.entry_id, None)
|
|
assert len(async_setup.mock_calls) == 1
|
|
assert len(async_setup_entry.mock_calls) == 1
|
|
assert entry.state is config_entries.ConfigEntryState.LOADED
|
|
|
|
# Disable
|
|
assert not await manager.async_set_disabled_by(
|
|
entry.entry_id, config_entries.DISABLED_USER
|
|
)
|
|
assert len(async_setup.mock_calls) == 1
|
|
assert len(async_setup_entry.mock_calls) == 1
|
|
assert entry.state is config_entries.ConfigEntryState.FAILED_UNLOAD
|
|
|
|
|
|
async def test_init_custom_integration(hass):
|
|
"""Test initializing flow for custom integration."""
|
|
integration = loader.Integration(
|
|
hass,
|
|
"custom_components.hue",
|
|
None,
|
|
{"name": "Hue", "dependencies": [], "requirements": [], "domain": "hue"},
|
|
)
|
|
with pytest.raises(data_entry_flow.UnknownHandler), patch(
|
|
"homeassistant.loader.async_get_integration",
|
|
return_value=integration,
|
|
):
|
|
await hass.config_entries.flow.async_init("bla")
|
|
|
|
|
|
async def test_support_entry_unload(hass):
|
|
"""Test unloading entry."""
|
|
assert await config_entries.support_entry_unload(hass, "light")
|
|
assert not await config_entries.support_entry_unload(hass, "auth")
|
|
|
|
|
|
async def test_reload_entry_entity_registry_ignores_no_entry(hass):
|
|
"""Test reloading entry in entity registry skips if no config entry linked."""
|
|
handler = config_entries.EntityRegistryDisabledHandler(hass)
|
|
registry = mock_registry(hass)
|
|
|
|
# Test we ignore entities without config entry
|
|
entry = registry.async_get_or_create("light", "hue", "123")
|
|
registry.async_update_entity(entry.entity_id, disabled_by=er.DISABLED_USER)
|
|
await hass.async_block_till_done()
|
|
assert not handler.changed
|
|
assert handler._remove_call_later is None
|
|
|
|
|
|
async def test_reload_entry_entity_registry_works(hass):
|
|
"""Test we schedule an entry to be reloaded if disabled_by is updated."""
|
|
handler = config_entries.EntityRegistryDisabledHandler(hass)
|
|
handler.async_setup()
|
|
registry = mock_registry(hass)
|
|
|
|
config_entry = MockConfigEntry(
|
|
domain="comp", state=config_entries.ConfigEntryState.LOADED
|
|
)
|
|
config_entry.supports_unload = True
|
|
config_entry.add_to_hass(hass)
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
mock_unload_entry = AsyncMock(return_value=True)
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"comp",
|
|
async_setup_entry=mock_setup_entry,
|
|
async_unload_entry=mock_unload_entry,
|
|
),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
# Only changing disabled_by should update trigger
|
|
entity_entry = registry.async_get_or_create(
|
|
"light", "hue", "123", config_entry=config_entry
|
|
)
|
|
registry.async_update_entity(entity_entry.entity_id, name="yo")
|
|
await hass.async_block_till_done()
|
|
assert not handler.changed
|
|
assert handler._remove_call_later is None
|
|
|
|
# Disable entity, we should not do anything, only act when enabled.
|
|
registry.async_update_entity(entity_entry.entity_id, disabled_by=er.DISABLED_USER)
|
|
await hass.async_block_till_done()
|
|
assert not handler.changed
|
|
assert handler._remove_call_later is None
|
|
|
|
# Enable entity, check we are reloading config entry.
|
|
registry.async_update_entity(entity_entry.entity_id, disabled_by=None)
|
|
await hass.async_block_till_done()
|
|
assert handler.changed == {config_entry.entry_id}
|
|
assert handler._remove_call_later is not None
|
|
|
|
async_fire_time_changed(
|
|
hass,
|
|
dt.utcnow() + timedelta(seconds=config_entries.RELOAD_AFTER_UPDATE_DELAY + 1),
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert len(mock_unload_entry.mock_calls) == 1
|
|
|
|
|
|
async def test_unique_id_persisted(hass, manager):
|
|
"""Test that a unique ID is stored in the config entry."""
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(hass, MockModule("comp", async_setup_entry=mock_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Test user step."""
|
|
await self.async_set_unique_id("mock-unique-id")
|
|
return self.async_create_entry(title="mock-title", data={})
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
|
|
assert len(mock_setup_entry.mock_calls) == 1
|
|
p_hass, p_entry = mock_setup_entry.mock_calls[0][1]
|
|
|
|
assert p_hass is hass
|
|
assert p_entry.unique_id == "mock-unique-id"
|
|
|
|
|
|
async def test_unique_id_existing_entry(hass, manager):
|
|
"""Test that we remove an entry if there already is an entry with unique ID."""
|
|
hass.config.components.add("comp")
|
|
MockConfigEntry(
|
|
domain="comp",
|
|
state=config_entries.ConfigEntryState.LOADED,
|
|
unique_id="mock-unique-id",
|
|
).add_to_hass(hass)
|
|
|
|
async_setup_entry = AsyncMock(return_value=True)
|
|
async_unload_entry = AsyncMock(return_value=True)
|
|
async_remove_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"comp",
|
|
async_setup_entry=async_setup_entry,
|
|
async_unload_entry=async_unload_entry,
|
|
async_remove_entry=async_remove_entry,
|
|
),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Test user step."""
|
|
existing_entry = await self.async_set_unique_id("mock-unique-id")
|
|
|
|
assert existing_entry is not None
|
|
|
|
return self.async_create_entry(title="mock-title", data={"via": "flow"})
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
result = await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
|
|
assert result["type"] == data_entry_flow.RESULT_TYPE_CREATE_ENTRY
|
|
|
|
entries = hass.config_entries.async_entries("comp")
|
|
assert len(entries) == 1
|
|
assert entries[0].data == {"via": "flow"}
|
|
|
|
assert len(async_setup_entry.mock_calls) == 1
|
|
assert len(async_unload_entry.mock_calls) == 1
|
|
assert len(async_remove_entry.mock_calls) == 1
|
|
|
|
|
|
async def test_entry_id_existing_entry(hass, manager):
|
|
"""Test that we throw when the entry id collides."""
|
|
collide_entry_id = "collide"
|
|
hass.config.components.add("comp")
|
|
MockConfigEntry(
|
|
entry_id=collide_entry_id,
|
|
domain="comp",
|
|
state=config_entries.ConfigEntryState.LOADED,
|
|
unique_id="mock-unique-id",
|
|
).add_to_hass(hass)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule("comp"),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Test user step."""
|
|
return self.async_create_entry(title="mock-title", data={"via": "flow"})
|
|
|
|
with pytest.raises(HomeAssistantError), patch.dict(
|
|
config_entries.HANDLERS, {"comp": TestFlow}
|
|
), patch(
|
|
"homeassistant.config_entries.uuid_util.random_uuid_hex",
|
|
return_value=collide_entry_id,
|
|
):
|
|
await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
|
|
|
|
async def test_unique_id_update_existing_entry_without_reload(hass, manager):
|
|
"""Test that we update an entry if there already is an entry with unique ID."""
|
|
hass.config.components.add("comp")
|
|
entry = MockConfigEntry(
|
|
domain="comp",
|
|
data={"additional": "data", "host": "0.0.0.0"},
|
|
unique_id="mock-unique-id",
|
|
state=config_entries.ConfigEntryState.LOADED,
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule("comp"),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Test user step."""
|
|
await self.async_set_unique_id("mock-unique-id")
|
|
await self._abort_if_unique_id_configured(
|
|
updates={"host": "1.1.1.1"}, reload_on_update=False
|
|
)
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}), patch(
|
|
"homeassistant.config_entries.ConfigEntries.async_reload"
|
|
) as async_reload:
|
|
result = await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert result["type"] == RESULT_TYPE_ABORT
|
|
assert result["reason"] == "already_configured"
|
|
assert entry.data["host"] == "1.1.1.1"
|
|
assert entry.data["additional"] == "data"
|
|
assert len(async_reload.mock_calls) == 0
|
|
|
|
|
|
async def test_unique_id_update_existing_entry_with_reload(hass, manager):
|
|
"""Test that we update an entry if there already is an entry with unique ID and we reload on changes."""
|
|
hass.config.components.add("comp")
|
|
entry = MockConfigEntry(
|
|
domain="comp",
|
|
data={"additional": "data", "host": "0.0.0.0"},
|
|
unique_id="mock-unique-id",
|
|
state=config_entries.ConfigEntryState.LOADED,
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule("comp"),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
updates = {"host": "1.1.1.1"}
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Test user step."""
|
|
await self.async_set_unique_id("mock-unique-id")
|
|
await self._abort_if_unique_id_configured(
|
|
updates=updates, reload_on_update=True
|
|
)
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}), patch(
|
|
"homeassistant.config_entries.ConfigEntries.async_reload"
|
|
) as async_reload:
|
|
result = await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert result["type"] == RESULT_TYPE_ABORT
|
|
assert result["reason"] == "already_configured"
|
|
assert entry.data["host"] == "1.1.1.1"
|
|
assert entry.data["additional"] == "data"
|
|
assert len(async_reload.mock_calls) == 1
|
|
|
|
# Test we don't reload if entry not started
|
|
updates["host"] = "2.2.2.2"
|
|
entry.state = config_entries.ConfigEntryState.NOT_LOADED
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}), patch(
|
|
"homeassistant.config_entries.ConfigEntries.async_reload"
|
|
) as async_reload:
|
|
result = await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert result["type"] == RESULT_TYPE_ABORT
|
|
assert result["reason"] == "already_configured"
|
|
assert entry.data["host"] == "2.2.2.2"
|
|
assert entry.data["additional"] == "data"
|
|
assert len(async_reload.mock_calls) == 0
|
|
|
|
|
|
async def test_unique_id_not_update_existing_entry(hass, manager):
|
|
"""Test that we do not update an entry if existing entry has the data."""
|
|
hass.config.components.add("comp")
|
|
entry = MockConfigEntry(
|
|
domain="comp",
|
|
data={"additional": "data", "host": "0.0.0.0"},
|
|
unique_id="mock-unique-id",
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule("comp"),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Test user step."""
|
|
await self.async_set_unique_id("mock-unique-id")
|
|
await self._abort_if_unique_id_configured(
|
|
updates={"host": "0.0.0.0"}, reload_on_update=True
|
|
)
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}), patch(
|
|
"homeassistant.config_entries.ConfigEntries.async_reload"
|
|
) as async_reload:
|
|
result = await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert result["type"] == RESULT_TYPE_ABORT
|
|
assert result["reason"] == "already_configured"
|
|
assert entry.data["host"] == "0.0.0.0"
|
|
assert entry.data["additional"] == "data"
|
|
assert len(async_reload.mock_calls) == 0
|
|
|
|
|
|
async def test_unique_id_in_progress(hass, manager):
|
|
"""Test that we abort if there is already a flow in progress with same unique id."""
|
|
mock_integration(hass, MockModule("comp"))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Test user step."""
|
|
await self.async_set_unique_id("mock-unique-id")
|
|
return self.async_show_form(step_id="discovery")
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
# Create one to be in progress
|
|
result = await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
assert result["type"] == data_entry_flow.RESULT_TYPE_FORM
|
|
|
|
# Will be canceled
|
|
result2 = await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
|
|
assert result2["type"] == data_entry_flow.RESULT_TYPE_ABORT
|
|
assert result2["reason"] == "already_in_progress"
|
|
|
|
|
|
async def test_finish_flow_aborts_progress(hass, manager):
|
|
"""Test that when finishing a flow, we abort other flows in progress with unique ID."""
|
|
mock_integration(
|
|
hass,
|
|
MockModule("comp", async_setup_entry=AsyncMock(return_value=True)),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Test user step."""
|
|
await self.async_set_unique_id("mock-unique-id", raise_on_progress=False)
|
|
|
|
if user_input is None:
|
|
return self.async_show_form(step_id="discovery")
|
|
|
|
return self.async_create_entry(title="yo", data={})
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
# Create one to be in progress
|
|
result = await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
assert result["type"] == data_entry_flow.RESULT_TYPE_FORM
|
|
|
|
# Will finish and cancel other one.
|
|
result2 = await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_USER}, data={}
|
|
)
|
|
|
|
assert result2["type"] == data_entry_flow.RESULT_TYPE_CREATE_ENTRY
|
|
|
|
assert len(hass.config_entries.flow.async_progress()) == 0
|
|
|
|
|
|
async def test_unique_id_ignore(hass, manager):
|
|
"""Test that we can ignore flows that are in progress and have a unique ID."""
|
|
async_setup_entry = AsyncMock(return_value=False)
|
|
mock_integration(hass, MockModule("comp", async_setup_entry=async_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Test user flow."""
|
|
await self.async_set_unique_id("mock-unique-id")
|
|
return self.async_show_form(step_id="discovery")
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
# Create one to be in progress
|
|
result = await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
assert result["type"] == data_entry_flow.RESULT_TYPE_FORM
|
|
|
|
result2 = await manager.flow.async_init(
|
|
"comp",
|
|
context={"source": config_entries.SOURCE_IGNORE},
|
|
data={"unique_id": "mock-unique-id", "title": "Ignored Title"},
|
|
)
|
|
|
|
assert result2["type"] == data_entry_flow.RESULT_TYPE_CREATE_ENTRY
|
|
|
|
# assert len(hass.config_entries.flow.async_progress()) == 0
|
|
|
|
# We should never set up an ignored entry.
|
|
assert len(async_setup_entry.mock_calls) == 0
|
|
|
|
entry = hass.config_entries.async_entries("comp")[0]
|
|
|
|
assert entry.source == "ignore"
|
|
assert entry.unique_id == "mock-unique-id"
|
|
assert entry.title == "Ignored Title"
|
|
|
|
|
|
async def test_manual_add_overrides_ignored_entry(hass, manager):
|
|
"""Test that we can ignore manually add entry, overriding ignored entry."""
|
|
hass.config.components.add("comp")
|
|
entry = MockConfigEntry(
|
|
domain="comp",
|
|
data={"additional": "data", "host": "0.0.0.0"},
|
|
unique_id="mock-unique-id",
|
|
state=config_entries.ConfigEntryState.LOADED,
|
|
source=config_entries.SOURCE_IGNORE,
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule("comp"),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Test user step."""
|
|
await self.async_set_unique_id("mock-unique-id")
|
|
self._abort_if_unique_id_configured(
|
|
updates={"host": "1.1.1.1"}, reload_on_update=False
|
|
)
|
|
return self.async_show_form(step_id="step2")
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}), patch(
|
|
"homeassistant.config_entries.ConfigEntries.async_reload"
|
|
) as async_reload:
|
|
result = await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert result["type"] == data_entry_flow.RESULT_TYPE_FORM
|
|
assert entry.data["host"] == "1.1.1.1"
|
|
assert entry.data["additional"] == "data"
|
|
assert len(async_reload.mock_calls) == 0
|
|
|
|
|
|
async def test_manual_add_overrides_ignored_entry_singleton(hass, manager):
|
|
"""Test that we can ignore manually add entry, overriding ignored entry."""
|
|
hass.config.components.add("comp")
|
|
entry = MockConfigEntry(
|
|
domain="comp",
|
|
state=config_entries.ConfigEntryState.LOADED,
|
|
source=config_entries.SOURCE_IGNORE,
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(hass, MockModule("comp", async_setup_entry=mock_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Test user step."""
|
|
if self._async_current_entries():
|
|
return self.async_abort(reason="single_instance_allowed")
|
|
return self.async_create_entry(title="title", data={"token": "supersecret"})
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow, "beer": 5}):
|
|
await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert len(mock_setup_entry.mock_calls) == 1
|
|
p_hass, p_entry = mock_setup_entry.mock_calls[0][1]
|
|
|
|
assert p_hass is hass
|
|
assert p_entry.data == {"token": "supersecret"}
|
|
|
|
|
|
async def test__async_current_entries_does_not_skip_ignore_non_user(hass, manager):
|
|
"""Test that _async_current_entries does not skip ignore by default for non user step."""
|
|
hass.config.components.add("comp")
|
|
entry = MockConfigEntry(
|
|
domain="comp",
|
|
state=config_entries.ConfigEntryState.LOADED,
|
|
source=config_entries.SOURCE_IGNORE,
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(hass, MockModule("comp", async_setup_entry=mock_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_import(self, user_input=None):
|
|
"""Test not the user step."""
|
|
if self._async_current_entries():
|
|
return self.async_abort(reason="single_instance_allowed")
|
|
return self.async_create_entry(title="title", data={"token": "supersecret"})
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow, "beer": 5}):
|
|
await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_IMPORT}
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert len(mock_setup_entry.mock_calls) == 0
|
|
|
|
|
|
async def test__async_current_entries_explicit_skip_ignore(hass, manager):
|
|
"""Test that _async_current_entries can explicitly include ignore."""
|
|
hass.config.components.add("comp")
|
|
entry = MockConfigEntry(
|
|
domain="comp",
|
|
state=config_entries.ConfigEntryState.LOADED,
|
|
source=config_entries.SOURCE_IGNORE,
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(hass, MockModule("comp", async_setup_entry=mock_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_import(self, user_input=None):
|
|
"""Test not the user step."""
|
|
if self._async_current_entries(include_ignore=False):
|
|
return self.async_abort(reason="single_instance_allowed")
|
|
return self.async_create_entry(title="title", data={"token": "supersecret"})
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow, "beer": 5}):
|
|
await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_IMPORT}
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert len(mock_setup_entry.mock_calls) == 1
|
|
p_hass, p_entry = mock_setup_entry.mock_calls[0][1]
|
|
|
|
assert p_hass is hass
|
|
assert p_entry.data == {"token": "supersecret"}
|
|
|
|
|
|
async def test__async_current_entries_explicit_include_ignore(hass, manager):
|
|
"""Test that _async_current_entries can explicitly include ignore."""
|
|
hass.config.components.add("comp")
|
|
entry = MockConfigEntry(
|
|
domain="comp",
|
|
state=config_entries.ConfigEntryState.LOADED,
|
|
source=config_entries.SOURCE_IGNORE,
|
|
)
|
|
entry.add_to_hass(hass)
|
|
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(hass, MockModule("comp", async_setup_entry=mock_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_import(self, user_input=None):
|
|
"""Test not the user step."""
|
|
if self._async_current_entries(include_ignore=True):
|
|
return self.async_abort(reason="single_instance_allowed")
|
|
return self.async_create_entry(title="title", data={"token": "supersecret"})
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow, "beer": 5}):
|
|
await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_IMPORT}
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert len(mock_setup_entry.mock_calls) == 0
|
|
|
|
|
|
async def test_unignore_step_form(hass, manager):
|
|
"""Test that we can ignore flows that are in progress and have a unique ID, then rediscover them."""
|
|
async_setup_entry = AsyncMock(return_value=True)
|
|
mock_integration(hass, MockModule("comp", async_setup_entry=async_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_unignore(self, user_input):
|
|
"""Test unignore step."""
|
|
unique_id = user_input["unique_id"]
|
|
await self.async_set_unique_id(unique_id)
|
|
return self.async_show_form(step_id="discovery")
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
result = await manager.flow.async_init(
|
|
"comp",
|
|
context={"source": config_entries.SOURCE_IGNORE},
|
|
data={"unique_id": "mock-unique-id", "title": "Ignored Title"},
|
|
)
|
|
assert result["type"] == data_entry_flow.RESULT_TYPE_CREATE_ENTRY
|
|
|
|
entry = hass.config_entries.async_entries("comp")[0]
|
|
assert entry.source == "ignore"
|
|
assert entry.unique_id == "mock-unique-id"
|
|
assert entry.domain == "comp"
|
|
assert entry.title == "Ignored Title"
|
|
|
|
await manager.async_remove(entry.entry_id)
|
|
|
|
# Right after removal there shouldn't be an entry or active flows
|
|
assert len(hass.config_entries.async_entries("comp")) == 0
|
|
assert len(hass.config_entries.flow.async_progress()) == 0
|
|
|
|
# But after a 'tick' the unignore step has run and we can see an active flow again.
|
|
await hass.async_block_till_done()
|
|
assert len(hass.config_entries.flow.async_progress()) == 1
|
|
|
|
# and still not config entries
|
|
assert len(hass.config_entries.async_entries("comp")) == 0
|
|
|
|
|
|
async def test_unignore_create_entry(hass, manager):
|
|
"""Test that we can ignore flows that are in progress and have a unique ID, then rediscover them."""
|
|
async_setup_entry = AsyncMock(return_value=True)
|
|
mock_integration(hass, MockModule("comp", async_setup_entry=async_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_unignore(self, user_input):
|
|
"""Test unignore step."""
|
|
unique_id = user_input["unique_id"]
|
|
await self.async_set_unique_id(unique_id)
|
|
return self.async_create_entry(title="yo", data={})
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
result = await manager.flow.async_init(
|
|
"comp",
|
|
context={"source": config_entries.SOURCE_IGNORE},
|
|
data={"unique_id": "mock-unique-id", "title": "Ignored Title"},
|
|
)
|
|
assert result["type"] == data_entry_flow.RESULT_TYPE_CREATE_ENTRY
|
|
|
|
entry = hass.config_entries.async_entries("comp")[0]
|
|
assert entry.source == "ignore"
|
|
assert entry.unique_id == "mock-unique-id"
|
|
assert entry.domain == "comp"
|
|
assert entry.title == "Ignored Title"
|
|
|
|
await manager.async_remove(entry.entry_id)
|
|
|
|
# Right after removal there shouldn't be an entry or flow
|
|
assert len(hass.config_entries.flow.async_progress()) == 0
|
|
assert len(hass.config_entries.async_entries("comp")) == 0
|
|
|
|
# But after a 'tick' the unignore step has run and we can see a config entry.
|
|
await hass.async_block_till_done()
|
|
entry = hass.config_entries.async_entries("comp")[0]
|
|
assert entry.source == config_entries.SOURCE_UNIGNORE
|
|
assert entry.unique_id == "mock-unique-id"
|
|
assert entry.title == "yo"
|
|
|
|
# And still no active flow
|
|
assert len(hass.config_entries.flow.async_progress()) == 0
|
|
|
|
|
|
async def test_unignore_default_impl(hass, manager):
|
|
"""Test that resdicovery is a no-op by default."""
|
|
async_setup_entry = AsyncMock(return_value=True)
|
|
mock_integration(hass, MockModule("comp", async_setup_entry=async_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
result = await manager.flow.async_init(
|
|
"comp",
|
|
context={"source": config_entries.SOURCE_IGNORE},
|
|
data={"unique_id": "mock-unique-id", "title": "Ignored Title"},
|
|
)
|
|
assert result["type"] == data_entry_flow.RESULT_TYPE_CREATE_ENTRY
|
|
|
|
entry = hass.config_entries.async_entries("comp")[0]
|
|
assert entry.source == "ignore"
|
|
assert entry.unique_id == "mock-unique-id"
|
|
assert entry.domain == "comp"
|
|
assert entry.title == "Ignored Title"
|
|
|
|
await manager.async_remove(entry.entry_id)
|
|
await hass.async_block_till_done()
|
|
|
|
assert len(hass.config_entries.async_entries("comp")) == 0
|
|
assert len(hass.config_entries.flow.async_progress()) == 0
|
|
|
|
|
|
async def test_partial_flows_hidden(hass, manager):
|
|
"""Test that flows that don't have a cur_step and haven't finished initing are hidden."""
|
|
async_setup_entry = AsyncMock(return_value=True)
|
|
mock_integration(hass, MockModule("comp", async_setup_entry=async_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
# A flag to test our assertion that `async_step_discovery` was called and is in its blocked state
|
|
# This simulates if the step was e.g. doing network i/o
|
|
discovery_started = asyncio.Event()
|
|
|
|
# A flag to allow `async_step_discovery` to resume after we have verified the uninited flow is not
|
|
# visible and has not triggered a discovery alert. This lets us control when the mocked network
|
|
# i/o is complete.
|
|
pause_discovery = asyncio.Event()
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_discovery(self, discovery_info):
|
|
"""Test discovery step."""
|
|
discovery_started.set()
|
|
await pause_discovery.wait()
|
|
return self.async_show_form(step_id="someform")
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
# Start a config entry flow and wait for it to be blocked
|
|
init_task = asyncio.ensure_future(
|
|
manager.flow.async_init(
|
|
"comp",
|
|
context={"source": config_entries.SOURCE_DISCOVERY},
|
|
data={"unique_id": "mock-unique-id"},
|
|
)
|
|
)
|
|
await discovery_started.wait()
|
|
|
|
# While it's blocked it shouldn't be visible or trigger discovery notifications
|
|
assert len(hass.config_entries.flow.async_progress()) == 0
|
|
|
|
await hass.async_block_till_done()
|
|
state = hass.states.get("persistent_notification.config_entry_discovery")
|
|
assert state is None
|
|
|
|
# Let the flow init complete
|
|
pause_discovery.set()
|
|
|
|
# When it's complete it should now be visible in async_progress and have triggered
|
|
# discovery notifications
|
|
result = await init_task
|
|
assert result["type"] == data_entry_flow.RESULT_TYPE_FORM
|
|
assert len(hass.config_entries.flow.async_progress()) == 1
|
|
|
|
await hass.async_block_till_done()
|
|
state = hass.states.get("persistent_notification.config_entry_discovery")
|
|
assert state is not None
|
|
|
|
|
|
async def test_async_setup_init_entry(hass):
|
|
"""Test a config entry being initialized during integration setup."""
|
|
|
|
async def mock_async_setup(hass, config):
|
|
"""Mock setup."""
|
|
hass.async_create_task(
|
|
hass.config_entries.flow.async_init(
|
|
"comp",
|
|
context={"source": config_entries.SOURCE_IMPORT},
|
|
data={},
|
|
)
|
|
)
|
|
return True
|
|
|
|
async_setup_entry = AsyncMock(return_value=True)
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"comp", async_setup=mock_async_setup, async_setup_entry=async_setup_entry
|
|
),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_import(self, user_input):
|
|
"""Test import step creating entry."""
|
|
return self.async_create_entry(title="title", data={})
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
assert await async_setup_component(hass, "comp", {})
|
|
|
|
await hass.async_block_till_done()
|
|
|
|
assert len(async_setup_entry.mock_calls) == 1
|
|
|
|
entries = hass.config_entries.async_entries("comp")
|
|
assert len(entries) == 1
|
|
assert entries[0].state is config_entries.ConfigEntryState.LOADED
|
|
|
|
|
|
async def test_async_setup_update_entry(hass):
|
|
"""Test a config entry being updated during integration setup."""
|
|
entry = MockConfigEntry(domain="comp", data={"value": "initial"})
|
|
entry.add_to_hass(hass)
|
|
|
|
async def mock_async_setup(hass, config):
|
|
"""Mock setup."""
|
|
hass.async_create_task(
|
|
hass.config_entries.flow.async_init(
|
|
"comp",
|
|
context={"source": config_entries.SOURCE_IMPORT},
|
|
data={},
|
|
)
|
|
)
|
|
return True
|
|
|
|
async def mock_async_setup_entry(hass, entry):
|
|
"""Mock setting up an entry."""
|
|
assert entry.data["value"] == "updated"
|
|
return True
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"comp",
|
|
async_setup=mock_async_setup,
|
|
async_setup_entry=mock_async_setup_entry,
|
|
),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_import(self, user_input):
|
|
"""Test import step updating existing entry."""
|
|
assert (
|
|
self.hass.config_entries.async_update_entry(
|
|
entry, data={"value": "updated"}
|
|
)
|
|
is True
|
|
)
|
|
return self.async_abort(reason="yo")
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
assert await async_setup_component(hass, "comp", {})
|
|
|
|
entries = hass.config_entries.async_entries("comp")
|
|
assert len(entries) == 1
|
|
assert entries[0].state is config_entries.ConfigEntryState.LOADED
|
|
assert entries[0].data == {"value": "updated"}
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"discovery_source",
|
|
(
|
|
config_entries.SOURCE_DISCOVERY,
|
|
config_entries.SOURCE_SSDP,
|
|
config_entries.SOURCE_USB,
|
|
config_entries.SOURCE_HOMEKIT,
|
|
config_entries.SOURCE_DHCP,
|
|
config_entries.SOURCE_ZEROCONF,
|
|
config_entries.SOURCE_HASSIO,
|
|
),
|
|
)
|
|
async def test_flow_with_default_discovery(hass, manager, discovery_source):
|
|
"""Test that finishing a default discovery flow removes the unique ID in the entry."""
|
|
mock_integration(
|
|
hass,
|
|
MockModule("comp", async_setup_entry=AsyncMock(return_value=True)),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Test user step."""
|
|
if user_input is None:
|
|
return self.async_show_form(step_id="user")
|
|
|
|
return self.async_create_entry(title="yo", data={})
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
# Create one to be in progress
|
|
result = await manager.flow.async_init(
|
|
"comp", context={"source": discovery_source}
|
|
)
|
|
assert result["type"] == data_entry_flow.RESULT_TYPE_FORM
|
|
|
|
flows = hass.config_entries.flow.async_progress()
|
|
assert len(flows) == 1
|
|
assert (
|
|
flows[0]["context"]["unique_id"]
|
|
== config_entries.DEFAULT_DISCOVERY_UNIQUE_ID
|
|
)
|
|
|
|
# Finish flow
|
|
result2 = await manager.flow.async_configure(
|
|
result["flow_id"], user_input={"fake": "data"}
|
|
)
|
|
assert result2["type"] == data_entry_flow.RESULT_TYPE_CREATE_ENTRY
|
|
|
|
assert len(hass.config_entries.flow.async_progress()) == 0
|
|
|
|
entry = hass.config_entries.async_entries("comp")[0]
|
|
assert entry.title == "yo"
|
|
assert entry.source == discovery_source
|
|
assert entry.unique_id is None
|
|
|
|
|
|
async def test_flow_with_default_discovery_with_unique_id(hass, manager):
|
|
"""Test discovery flow using the default discovery is ignored when unique ID is set."""
|
|
mock_integration(hass, MockModule("comp"))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_discovery(self, discovery_info):
|
|
"""Test discovery step."""
|
|
await self.async_set_unique_id("mock-unique-id")
|
|
# This call should make no difference, as a unique ID is set
|
|
await self._async_handle_discovery_without_unique_id()
|
|
return self.async_show_form(step_id="mock")
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
result = await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_DISCOVERY}
|
|
)
|
|
assert result["type"] == data_entry_flow.RESULT_TYPE_FORM
|
|
|
|
flows = hass.config_entries.flow.async_progress()
|
|
assert len(flows) == 1
|
|
assert flows[0]["context"]["unique_id"] == "mock-unique-id"
|
|
|
|
|
|
async def test_default_discovery_abort_existing_entries(hass, manager):
|
|
"""Test that a flow without discovery implementation aborts when a config entry exists."""
|
|
hass.config.components.add("comp")
|
|
entry = MockConfigEntry(domain="comp", data={}, unique_id="mock-unique-id")
|
|
entry.add_to_hass(hass)
|
|
|
|
mock_integration(hass, MockModule("comp"))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
result = await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_DISCOVERY}
|
|
)
|
|
assert result["type"] == data_entry_flow.RESULT_TYPE_ABORT
|
|
assert result["reason"] == "already_configured"
|
|
|
|
|
|
async def test_default_discovery_in_progress(hass, manager):
|
|
"""Test that a flow using default discovery can only be triggered once."""
|
|
mock_integration(hass, MockModule("comp"))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_discovery(self, discovery_info):
|
|
"""Test discovery step."""
|
|
await self.async_set_unique_id(discovery_info.get("unique_id"))
|
|
await self._async_handle_discovery_without_unique_id()
|
|
return self.async_show_form(step_id="mock")
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
result = await manager.flow.async_init(
|
|
"comp",
|
|
context={"source": config_entries.SOURCE_DISCOVERY},
|
|
data={"unique_id": "mock-unique-id"},
|
|
)
|
|
assert result["type"] == data_entry_flow.RESULT_TYPE_FORM
|
|
|
|
# Second discovery without a unique ID
|
|
result2 = await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_DISCOVERY}, data={}
|
|
)
|
|
assert result2["type"] == data_entry_flow.RESULT_TYPE_ABORT
|
|
|
|
flows = hass.config_entries.flow.async_progress()
|
|
assert len(flows) == 1
|
|
assert flows[0]["context"]["unique_id"] == "mock-unique-id"
|
|
|
|
|
|
async def test_default_discovery_abort_on_new_unique_flow(hass, manager):
|
|
"""Test that a flow using default discovery is aborted when a second flow with unique ID is created."""
|
|
mock_integration(hass, MockModule("comp"))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_discovery(self, discovery_info):
|
|
"""Test discovery step."""
|
|
await self.async_set_unique_id(discovery_info.get("unique_id"))
|
|
await self._async_handle_discovery_without_unique_id()
|
|
return self.async_show_form(step_id="mock")
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
# First discovery with default, no unique ID
|
|
result2 = await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_DISCOVERY}, data={}
|
|
)
|
|
assert result2["type"] == data_entry_flow.RESULT_TYPE_FORM
|
|
|
|
# Second discovery brings in a unique ID
|
|
result = await manager.flow.async_init(
|
|
"comp",
|
|
context={"source": config_entries.SOURCE_DISCOVERY},
|
|
data={"unique_id": "mock-unique-id"},
|
|
)
|
|
assert result["type"] == data_entry_flow.RESULT_TYPE_FORM
|
|
|
|
# Ensure the first one is cancelled and we end up with just the last one
|
|
flows = hass.config_entries.flow.async_progress()
|
|
assert len(flows) == 1
|
|
assert flows[0]["context"]["unique_id"] == "mock-unique-id"
|
|
|
|
|
|
async def test_default_discovery_abort_on_user_flow_complete(hass, manager):
|
|
"""Test that a flow using default discovery is aborted when a second flow completes."""
|
|
mock_integration(hass, MockModule("comp"))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Test user step."""
|
|
if user_input is None:
|
|
return self.async_show_form(step_id="user")
|
|
return self.async_create_entry(title="title", data={"token": "supersecret"})
|
|
|
|
async def async_step_discovery(self, discovery_info=None):
|
|
"""Test discovery step."""
|
|
await self._async_handle_discovery_without_unique_id()
|
|
return self.async_show_form(step_id="mock")
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
# First discovery with default, no unique ID
|
|
flow1 = await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_DISCOVERY}, data={}
|
|
)
|
|
assert flow1["type"] == data_entry_flow.RESULT_TYPE_FORM
|
|
|
|
flows = hass.config_entries.flow.async_progress()
|
|
assert len(flows) == 1
|
|
|
|
# User sets up a manual flow
|
|
flow2 = await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
assert flow2["type"] == data_entry_flow.RESULT_TYPE_FORM
|
|
|
|
flows = hass.config_entries.flow.async_progress()
|
|
assert len(flows) == 2
|
|
|
|
# Complete the manual flow
|
|
result = await hass.config_entries.flow.async_configure(flow2["flow_id"], {})
|
|
assert result["type"] == data_entry_flow.RESULT_TYPE_CREATE_ENTRY
|
|
|
|
# Ensure the first flow is gone now
|
|
flows = hass.config_entries.flow.async_progress()
|
|
assert len(flows) == 0
|
|
|
|
|
|
async def test_flow_same_device_multiple_sources(hass, manager):
|
|
"""Test discovery of the same devices from multiple discovery sources."""
|
|
mock_integration(
|
|
hass,
|
|
MockModule("comp", async_setup_entry=AsyncMock(return_value=True)),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_zeroconf(self, discovery_info=None):
|
|
"""Test zeroconf step."""
|
|
return await self._async_discovery_handler(discovery_info)
|
|
|
|
async def async_step_homekit(self, discovery_info=None):
|
|
"""Test homekit step."""
|
|
return await self._async_discovery_handler(discovery_info)
|
|
|
|
async def _async_discovery_handler(self, discovery_info=None):
|
|
"""Test any discovery handler."""
|
|
await self.async_set_unique_id("thisid")
|
|
self._abort_if_unique_id_configured()
|
|
await asyncio.sleep(0.1)
|
|
return await self.async_step_link()
|
|
|
|
async def async_step_link(self, user_input=None):
|
|
"""Test a link step."""
|
|
if user_input is None:
|
|
return self.async_show_form(step_id="link")
|
|
return self.async_create_entry(title="title", data={"token": "supersecret"})
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow}):
|
|
# Create one to be in progress
|
|
flow1 = manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_ZEROCONF}
|
|
)
|
|
flow2 = manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_ZEROCONF}
|
|
)
|
|
flow3 = manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_HOMEKIT}
|
|
)
|
|
result1, result2, result3 = await asyncio.gather(flow1, flow2, flow3)
|
|
|
|
flows = hass.config_entries.flow.async_progress()
|
|
assert len(flows) == 1
|
|
assert flows[0]["context"]["unique_id"] == "thisid"
|
|
|
|
# Finish flow
|
|
result2 = await manager.flow.async_configure(
|
|
flows[0]["flow_id"], user_input={"fake": "data"}
|
|
)
|
|
assert result2["type"] == data_entry_flow.RESULT_TYPE_CREATE_ENTRY
|
|
|
|
assert len(hass.config_entries.flow.async_progress()) == 0
|
|
|
|
entry = hass.config_entries.async_entries("comp")[0]
|
|
assert entry.title == "title"
|
|
assert entry.source in {
|
|
config_entries.SOURCE_ZEROCONF,
|
|
config_entries.SOURCE_HOMEKIT,
|
|
}
|
|
assert entry.unique_id == "thisid"
|
|
|
|
|
|
async def test_updating_entry_with_and_without_changes(manager):
|
|
"""Test that we can update an entry data."""
|
|
entry = MockConfigEntry(
|
|
domain="test",
|
|
data={"first": True},
|
|
title="thetitle",
|
|
options={"option": True},
|
|
unique_id="abc123",
|
|
state=config_entries.ConfigEntryState.SETUP_ERROR,
|
|
)
|
|
entry.add_to_manager(manager)
|
|
|
|
assert manager.async_update_entry(entry) is False
|
|
|
|
for change in (
|
|
{"data": {"second": True, "third": 456}},
|
|
{"data": {"second": True}},
|
|
{"options": {"hello": True}},
|
|
{"pref_disable_new_entities": True},
|
|
{"pref_disable_polling": True},
|
|
{"title": "sometitle"},
|
|
{"unique_id": "abcd1234"},
|
|
):
|
|
assert manager.async_update_entry(entry, **change) is True
|
|
assert manager.async_update_entry(entry, **change) is False
|
|
|
|
|
|
async def test_entry_reload_calls_on_unload_listeners(hass, manager):
|
|
"""Test reload calls the on unload listeners."""
|
|
entry = MockConfigEntry(domain="comp", state=config_entries.ConfigEntryState.LOADED)
|
|
entry.add_to_hass(hass)
|
|
|
|
async_setup = AsyncMock(return_value=True)
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
async_unload_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(
|
|
hass,
|
|
MockModule(
|
|
"comp",
|
|
async_setup=async_setup,
|
|
async_setup_entry=mock_setup_entry,
|
|
async_unload_entry=async_unload_entry,
|
|
),
|
|
)
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
mock_unload_callback = Mock()
|
|
|
|
entry.async_on_unload(mock_unload_callback)
|
|
|
|
assert await manager.async_reload(entry.entry_id)
|
|
assert len(async_unload_entry.mock_calls) == 1
|
|
assert len(mock_setup_entry.mock_calls) == 1
|
|
assert len(mock_unload_callback.mock_calls) == 1
|
|
assert entry.state is config_entries.ConfigEntryState.LOADED
|
|
|
|
assert await manager.async_reload(entry.entry_id)
|
|
assert len(async_unload_entry.mock_calls) == 2
|
|
assert len(mock_setup_entry.mock_calls) == 2
|
|
# Since we did not register another async_on_unload it should
|
|
# have only been called once
|
|
assert len(mock_unload_callback.mock_calls) == 1
|
|
assert entry.state is config_entries.ConfigEntryState.LOADED
|
|
|
|
|
|
async def test_setup_raise_auth_failed(hass, caplog):
|
|
"""Test a setup raising ConfigEntryAuthFailed."""
|
|
entry = MockConfigEntry(title="test_title", domain="test")
|
|
|
|
mock_setup_entry = AsyncMock(
|
|
side_effect=ConfigEntryAuthFailed("The password is no longer valid")
|
|
)
|
|
mock_integration(hass, MockModule("test", async_setup_entry=mock_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
|
|
|
await entry.async_setup(hass)
|
|
await hass.async_block_till_done()
|
|
assert "could not authenticate: The password is no longer valid" in caplog.text
|
|
|
|
assert entry.state is config_entries.ConfigEntryState.SETUP_ERROR
|
|
assert entry.reason == "The password is no longer valid"
|
|
flows = hass.config_entries.flow.async_progress()
|
|
assert len(flows) == 1
|
|
assert flows[0]["context"]["entry_id"] == entry.entry_id
|
|
assert flows[0]["context"]["source"] == config_entries.SOURCE_REAUTH
|
|
|
|
caplog.clear()
|
|
entry.state = config_entries.ConfigEntryState.NOT_LOADED
|
|
entry.reason = None
|
|
|
|
await entry.async_setup(hass)
|
|
await hass.async_block_till_done()
|
|
assert "could not authenticate: The password is no longer valid" in caplog.text
|
|
|
|
# Verify multiple ConfigEntryAuthFailed does not generate a second flow
|
|
assert entry.state is config_entries.ConfigEntryState.SETUP_ERROR
|
|
flows = hass.config_entries.flow.async_progress()
|
|
assert len(flows) == 1
|
|
|
|
|
|
async def test_setup_raise_auth_failed_from_first_coordinator_update(hass, caplog):
|
|
"""Test async_config_entry_first_refresh raises ConfigEntryAuthFailed."""
|
|
entry = MockConfigEntry(title="test_title", domain="test")
|
|
|
|
async def async_setup_entry(hass, entry):
|
|
"""Mock setup entry with a simple coordinator."""
|
|
|
|
async def _async_update_data():
|
|
raise ConfigEntryAuthFailed("The password is no longer valid")
|
|
|
|
coordinator = DataUpdateCoordinator(
|
|
hass,
|
|
logging.getLogger(__name__),
|
|
name="any",
|
|
update_method=_async_update_data,
|
|
update_interval=timedelta(seconds=1000),
|
|
)
|
|
|
|
await coordinator.async_config_entry_first_refresh()
|
|
return True
|
|
|
|
mock_integration(hass, MockModule("test", async_setup_entry=async_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
|
|
|
await entry.async_setup(hass)
|
|
await hass.async_block_till_done()
|
|
assert "could not authenticate: The password is no longer valid" in caplog.text
|
|
|
|
assert entry.state is config_entries.ConfigEntryState.SETUP_ERROR
|
|
flows = hass.config_entries.flow.async_progress()
|
|
assert len(flows) == 1
|
|
assert flows[0]["context"]["entry_id"] == entry.entry_id
|
|
assert flows[0]["context"]["source"] == config_entries.SOURCE_REAUTH
|
|
|
|
caplog.clear()
|
|
entry.state = config_entries.ConfigEntryState.NOT_LOADED
|
|
|
|
await entry.async_setup(hass)
|
|
await hass.async_block_till_done()
|
|
assert "could not authenticate: The password is no longer valid" in caplog.text
|
|
|
|
# Verify multiple ConfigEntryAuthFailed does not generate a second flow
|
|
assert entry.state is config_entries.ConfigEntryState.SETUP_ERROR
|
|
flows = hass.config_entries.flow.async_progress()
|
|
assert len(flows) == 1
|
|
|
|
|
|
async def test_setup_raise_auth_failed_from_future_coordinator_update(hass, caplog):
|
|
"""Test a coordinator raises ConfigEntryAuthFailed in the future."""
|
|
entry = MockConfigEntry(title="test_title", domain="test")
|
|
|
|
async def async_setup_entry(hass, entry):
|
|
"""Mock setup entry with a simple coordinator."""
|
|
|
|
async def _async_update_data():
|
|
raise ConfigEntryAuthFailed("The password is no longer valid")
|
|
|
|
coordinator = DataUpdateCoordinator(
|
|
hass,
|
|
logging.getLogger(__name__),
|
|
name="any",
|
|
update_method=_async_update_data,
|
|
update_interval=timedelta(seconds=1000),
|
|
)
|
|
|
|
await coordinator.async_refresh()
|
|
return True
|
|
|
|
mock_integration(hass, MockModule("test", async_setup_entry=async_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
|
|
|
await entry.async_setup(hass)
|
|
await hass.async_block_till_done()
|
|
assert "Authentication failed while fetching" in caplog.text
|
|
assert "The password is no longer valid" in caplog.text
|
|
|
|
assert entry.state is config_entries.ConfigEntryState.LOADED
|
|
flows = hass.config_entries.flow.async_progress()
|
|
assert len(flows) == 1
|
|
assert flows[0]["context"]["entry_id"] == entry.entry_id
|
|
assert flows[0]["context"]["source"] == config_entries.SOURCE_REAUTH
|
|
|
|
caplog.clear()
|
|
entry.state = config_entries.ConfigEntryState.NOT_LOADED
|
|
|
|
await entry.async_setup(hass)
|
|
await hass.async_block_till_done()
|
|
assert "Authentication failed while fetching" in caplog.text
|
|
assert "The password is no longer valid" in caplog.text
|
|
|
|
# Verify multiple ConfigEntryAuthFailed does not generate a second flow
|
|
assert entry.state is config_entries.ConfigEntryState.LOADED
|
|
flows = hass.config_entries.flow.async_progress()
|
|
assert len(flows) == 1
|
|
|
|
|
|
async def test_initialize_and_shutdown(hass):
|
|
"""Test we call the shutdown function at stop."""
|
|
manager = config_entries.ConfigEntries(hass, {})
|
|
|
|
with patch.object(manager, "_async_shutdown") as mock_async_shutdown:
|
|
await manager.async_initialize()
|
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_STOP)
|
|
await hass.async_block_till_done()
|
|
|
|
assert mock_async_shutdown.called
|
|
|
|
|
|
async def test_setup_retrying_during_shutdown(hass):
|
|
"""Test if we shutdown an entry that is in retry mode."""
|
|
entry = MockConfigEntry(domain="test")
|
|
|
|
mock_setup_entry = AsyncMock(side_effect=ConfigEntryNotReady)
|
|
mock_integration(hass, MockModule("test", async_setup_entry=mock_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.test", None)
|
|
|
|
with patch("homeassistant.helpers.event.async_call_later") as mock_call:
|
|
await entry.async_setup(hass)
|
|
|
|
assert entry.state is config_entries.ConfigEntryState.SETUP_RETRY
|
|
assert len(mock_call.return_value.mock_calls) == 0
|
|
|
|
hass.bus.async_fire(EVENT_HOMEASSISTANT_STOP)
|
|
await hass.async_block_till_done()
|
|
|
|
assert len(mock_call.return_value.mock_calls) == 0
|
|
|
|
async_fire_time_changed(hass, dt.utcnow() + timedelta(hours=4))
|
|
await hass.async_block_till_done()
|
|
|
|
assert len(mock_call.return_value.mock_calls) == 0
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"matchers, reason",
|
|
[
|
|
({}, "already_configured"),
|
|
({"host": "3.3.3.3"}, "no_match"),
|
|
({"host": "3.4.5.6"}, "already_configured"),
|
|
({"host": "3.4.5.6", "ip": "3.4.5.6"}, "no_match"),
|
|
({"host": "3.4.5.6", "ip": "1.2.3.4"}, "already_configured"),
|
|
({"host": "3.4.5.6", "ip": "1.2.3.4", "port": 23}, "already_configured"),
|
|
({"ip": "9.9.9.9"}, "already_configured"),
|
|
({"ip": "7.7.7.7"}, "no_match"), # ignored
|
|
],
|
|
)
|
|
async def test__async_abort_entries_match(hass, manager, matchers, reason):
|
|
"""Test aborting if matching config entries exist."""
|
|
MockConfigEntry(
|
|
domain="comp", data={"ip": "1.2.3.4", "host": "4.5.6.7", "port": 23}
|
|
).add_to_hass(hass)
|
|
MockConfigEntry(
|
|
domain="comp", data={"ip": "9.9.9.9", "host": "4.5.6.7", "port": 23}
|
|
).add_to_hass(hass)
|
|
MockConfigEntry(
|
|
domain="comp", data={"ip": "1.2.3.4", "host": "3.4.5.6", "port": 23}
|
|
).add_to_hass(hass)
|
|
MockConfigEntry(
|
|
domain="comp",
|
|
source=config_entries.SOURCE_IGNORE,
|
|
data={"ip": "7.7.7.7", "host": "4.5.6.7", "port": 23},
|
|
).add_to_hass(hass)
|
|
|
|
mock_setup_entry = AsyncMock(return_value=True)
|
|
|
|
mock_integration(hass, MockModule("comp", async_setup_entry=mock_setup_entry))
|
|
mock_entity_platform(hass, "config_flow.comp", None)
|
|
|
|
class TestFlow(config_entries.ConfigFlow):
|
|
"""Test flow."""
|
|
|
|
VERSION = 1
|
|
|
|
async def async_step_user(self, user_input=None):
|
|
"""Test user step."""
|
|
self._async_abort_entries_match(matchers)
|
|
return self.async_abort(reason="no_match")
|
|
|
|
with patch.dict(config_entries.HANDLERS, {"comp": TestFlow, "beer": 5}):
|
|
result = await manager.flow.async_init(
|
|
"comp", context={"source": config_entries.SOURCE_USER}
|
|
)
|
|
await hass.async_block_till_done()
|
|
|
|
assert result["type"] == RESULT_TYPE_ABORT
|
|
assert result["reason"] == reason
|
|
|
|
|
|
async def test_loading_old_data(hass, hass_storage):
|
|
"""Test automatically migrating old data."""
|
|
hass_storage[config_entries.STORAGE_KEY] = {
|
|
"version": 1,
|
|
"data": {
|
|
"entries": [
|
|
{
|
|
"version": 5,
|
|
"domain": "my_domain",
|
|
"entry_id": "mock-id",
|
|
"data": {"my": "data"},
|
|
"source": "user",
|
|
"title": "Mock title",
|
|
"system_options": {"disable_new_entities": True},
|
|
}
|
|
]
|
|
},
|
|
}
|
|
manager = config_entries.ConfigEntries(hass, {})
|
|
await manager.async_initialize()
|
|
|
|
entries = manager.async_entries()
|
|
assert len(entries) == 1
|
|
entry = entries[0]
|
|
assert entry.version == 5
|
|
assert entry.domain == "my_domain"
|
|
assert entry.entry_id == "mock-id"
|
|
assert entry.title == "Mock title"
|
|
assert entry.data == {"my": "data"}
|
|
assert entry.pref_disable_new_entities is True
|