Improve config entry state transitions when unloading and removing entries (#138522)
* Improve config entry state transitions when unloading and removing entries * Update integrations which check for a single loaded entry * Update tests checking state after unload fails * Update homeassistant/config_entries.py Co-authored-by: Martin Hjelmare <marhje52@gmail.com> --------- Co-authored-by: Martin Hjelmare <marhje52@gmail.com>
This commit is contained in:
parent
ff16e587e8
commit
e0795e6d07
|
@ -7,7 +7,7 @@ from dataclasses import dataclass
|
|||
from adguardhome import AdGuardHome, AdGuardHomeConnectionError
|
||||
import voluptuous as vol
|
||||
|
||||
from homeassistant.config_entries import ConfigEntry, ConfigEntryState
|
||||
from homeassistant.config_entries import ConfigEntry
|
||||
from homeassistant.const import (
|
||||
CONF_HOST,
|
||||
CONF_NAME,
|
||||
|
@ -123,12 +123,7 @@ async def async_setup_entry(hass: HomeAssistant, entry: AdGuardConfigEntry) -> b
|
|||
async def async_unload_entry(hass: HomeAssistant, entry: AdGuardConfigEntry) -> bool:
|
||||
"""Unload AdGuard Home config entry."""
|
||||
unload_ok = await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
|
||||
loaded_entries = [
|
||||
entry
|
||||
for entry in hass.config_entries.async_entries(DOMAIN)
|
||||
if entry.state == ConfigEntryState.LOADED
|
||||
]
|
||||
if len(loaded_entries) == 1:
|
||||
if not hass.config_entries.async_loaded_entries(DOMAIN):
|
||||
# This is the last loaded instance of AdGuard, deregister any services
|
||||
hass.services.async_remove(DOMAIN, SERVICE_ADD_URL)
|
||||
hass.services.async_remove(DOMAIN, SERVICE_REMOVE_URL)
|
||||
|
|
|
@ -10,7 +10,7 @@ from google.oauth2.credentials import Credentials
|
|||
import voluptuous as vol
|
||||
|
||||
from homeassistant.components import conversation
|
||||
from homeassistant.config_entries import ConfigEntry, ConfigEntryState
|
||||
from homeassistant.config_entries import ConfigEntry
|
||||
from homeassistant.const import CONF_ACCESS_TOKEN, CONF_NAME, Platform
|
||||
from homeassistant.core import (
|
||||
HomeAssistant,
|
||||
|
@ -99,12 +99,7 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|||
async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
||||
"""Unload a config entry."""
|
||||
hass.data[DOMAIN].pop(entry.entry_id)
|
||||
loaded_entries = [
|
||||
entry
|
||||
for entry in hass.config_entries.async_entries(DOMAIN)
|
||||
if entry.state == ConfigEntryState.LOADED
|
||||
]
|
||||
if len(loaded_entries) == 1:
|
||||
if not hass.config_entries.async_loaded_entries(DOMAIN):
|
||||
for service_name in hass.services.async_services_for_domain(DOMAIN):
|
||||
hass.services.async_remove(DOMAIN, service_name)
|
||||
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
|
||||
from __future__ import annotations
|
||||
|
||||
from homeassistant.config_entries import ConfigEntry, ConfigEntryState
|
||||
from homeassistant.config_entries import ConfigEntry
|
||||
from homeassistant.const import CONF_NAME, Platform
|
||||
from homeassistant.core import HomeAssistant
|
||||
from homeassistant.helpers import config_validation as cv, discovery
|
||||
|
@ -59,12 +59,7 @@ async def async_setup_entry(hass: HomeAssistant, entry: GoogleMailConfigEntry) -
|
|||
|
||||
async def async_unload_entry(hass: HomeAssistant, entry: GoogleMailConfigEntry) -> bool:
|
||||
"""Unload a config entry."""
|
||||
loaded_entries = [
|
||||
entry
|
||||
for entry in hass.config_entries.async_entries(DOMAIN)
|
||||
if entry.state == ConfigEntryState.LOADED
|
||||
]
|
||||
if len(loaded_entries) == 1:
|
||||
if not hass.config_entries.async_loaded_entries(DOMAIN):
|
||||
for service_name in hass.services.async_services_for_domain(DOMAIN):
|
||||
hass.services.async_remove(DOMAIN, service_name)
|
||||
|
||||
|
|
|
@ -12,7 +12,7 @@ from gspread.exceptions import APIError
|
|||
from gspread.utils import ValueInputOption
|
||||
import voluptuous as vol
|
||||
|
||||
from homeassistant.config_entries import ConfigEntry, ConfigEntryState
|
||||
from homeassistant.config_entries import ConfigEntry
|
||||
from homeassistant.const import CONF_ACCESS_TOKEN, CONF_TOKEN
|
||||
from homeassistant.core import HomeAssistant, ServiceCall
|
||||
from homeassistant.exceptions import (
|
||||
|
@ -81,12 +81,7 @@ async def async_unload_entry(
|
|||
hass: HomeAssistant, entry: GoogleSheetsConfigEntry
|
||||
) -> bool:
|
||||
"""Unload a config entry."""
|
||||
loaded_entries = [
|
||||
entry
|
||||
for entry in hass.config_entries.async_entries(DOMAIN)
|
||||
if entry.state == ConfigEntryState.LOADED
|
||||
]
|
||||
if len(loaded_entries) == 1:
|
||||
if not hass.config_entries.async_loaded_entries(DOMAIN):
|
||||
for service_name in hass.services.async_services_for_domain(DOMAIN):
|
||||
hass.services.async_remove(DOMAIN, service_name)
|
||||
|
||||
|
|
|
@ -11,7 +11,7 @@ from aioguardian import Client
|
|||
from aioguardian.errors import GuardianError
|
||||
import voluptuous as vol
|
||||
|
||||
from homeassistant.config_entries import ConfigEntry, ConfigEntryState
|
||||
from homeassistant.config_entries import ConfigEntry
|
||||
from homeassistant.const import (
|
||||
ATTR_DEVICE_ID,
|
||||
CONF_DEVICE_ID,
|
||||
|
@ -247,12 +247,7 @@ async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|||
if unload_ok:
|
||||
hass.data[DOMAIN].pop(entry.entry_id)
|
||||
|
||||
loaded_entries = [
|
||||
entry
|
||||
for entry in hass.config_entries.async_entries(DOMAIN)
|
||||
if entry.state == ConfigEntryState.LOADED
|
||||
]
|
||||
if len(loaded_entries) == 1:
|
||||
if not hass.config_entries.async_loaded_entries(DOMAIN):
|
||||
# If this is the last loaded instance of Guardian, deregister any services
|
||||
# defined during integration setup:
|
||||
for service_name in SERVICES:
|
||||
|
|
|
@ -19,7 +19,7 @@ from aiolookin import (
|
|||
)
|
||||
from aiolookin.models import UDPCommandType, UDPEvent
|
||||
|
||||
from homeassistant.config_entries import ConfigEntry, ConfigEntryState
|
||||
from homeassistant.config_entries import ConfigEntry
|
||||
from homeassistant.const import CONF_HOST, Platform
|
||||
from homeassistant.core import HomeAssistant, callback
|
||||
from homeassistant.exceptions import ConfigEntryNotReady
|
||||
|
@ -192,12 +192,7 @@ async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|||
if unload_ok := await hass.config_entries.async_unload_platforms(entry, PLATFORMS):
|
||||
hass.data[DOMAIN].pop(entry.entry_id)
|
||||
|
||||
loaded_entries = [
|
||||
entry
|
||||
for entry in hass.config_entries.async_entries(DOMAIN)
|
||||
if entry.state == ConfigEntryState.LOADED
|
||||
]
|
||||
if len(loaded_entries) == 1:
|
||||
if not hass.config_entries.async_loaded_entries(DOMAIN):
|
||||
manager: LookinUDPManager = hass.data[DOMAIN][UDP_MANAGER]
|
||||
await manager.async_stop()
|
||||
return unload_ok
|
||||
|
|
|
@ -6,7 +6,7 @@ from typing import TYPE_CHECKING
|
|||
|
||||
from motionblinds import AsyncMotionMulticast
|
||||
|
||||
from homeassistant.config_entries import ConfigEntry, ConfigEntryState
|
||||
from homeassistant.config_entries import ConfigEntry
|
||||
from homeassistant.const import CONF_API_KEY, CONF_HOST, EVENT_HOMEASSISTANT_STOP
|
||||
from homeassistant.core import HomeAssistant
|
||||
from homeassistant.exceptions import ConfigEntryNotReady
|
||||
|
@ -124,12 +124,7 @@ async def async_unload_entry(hass: HomeAssistant, config_entry: ConfigEntry) ->
|
|||
multicast.Unregister_motion_gateway(config_entry.data[CONF_HOST])
|
||||
hass.data[DOMAIN].pop(config_entry.entry_id)
|
||||
|
||||
loaded_entries = [
|
||||
entry
|
||||
for entry in hass.config_entries.async_entries(DOMAIN)
|
||||
if entry.state == ConfigEntryState.LOADED
|
||||
]
|
||||
if len(loaded_entries) == 1:
|
||||
if not hass.config_entries.async_loaded_entries(DOMAIN):
|
||||
# No motion gateways left, stop Motion multicast
|
||||
unsub_stop = hass.data[DOMAIN].pop(KEY_UNSUB_STOP)
|
||||
unsub_stop()
|
||||
|
|
|
@ -6,7 +6,6 @@ from aiohttp.cookiejar import CookieJar
|
|||
import eternalegypt
|
||||
from eternalegypt.eternalegypt import SMS
|
||||
|
||||
from homeassistant.config_entries import ConfigEntryState
|
||||
from homeassistant.const import CONF_HOST, CONF_NAME, CONF_PASSWORD, Platform
|
||||
from homeassistant.core import HomeAssistant
|
||||
from homeassistant.exceptions import ConfigEntryNotReady
|
||||
|
@ -117,12 +116,7 @@ async def async_setup_entry(hass: HomeAssistant, entry: NetgearLTEConfigEntry) -
|
|||
async def async_unload_entry(hass: HomeAssistant, entry: NetgearLTEConfigEntry) -> bool:
|
||||
"""Unload a config entry."""
|
||||
unload_ok = await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
|
||||
loaded_entries = [
|
||||
entry
|
||||
for entry in hass.config_entries.async_entries(DOMAIN)
|
||||
if entry.state == ConfigEntryState.LOADED
|
||||
]
|
||||
if len(loaded_entries) == 1:
|
||||
if not hass.config_entries.async_loaded_entries(DOMAIN):
|
||||
hass.data.pop(DOMAIN, None)
|
||||
for service_name in hass.services.async_services()[DOMAIN]:
|
||||
hass.services.async_remove(DOMAIN, service_name)
|
||||
|
|
|
@ -13,7 +13,7 @@ from regenmaschine.controller import Controller
|
|||
from regenmaschine.errors import RainMachineError, UnknownAPICallError
|
||||
import voluptuous as vol
|
||||
|
||||
from homeassistant.config_entries import ConfigEntry, ConfigEntryState
|
||||
from homeassistant.config_entries import ConfigEntry
|
||||
from homeassistant.const import (
|
||||
CONF_DEVICE_ID,
|
||||
CONF_IP_ADDRESS,
|
||||
|
@ -465,12 +465,7 @@ async def async_unload_entry(
|
|||
) -> bool:
|
||||
"""Unload an RainMachine config entry."""
|
||||
unload_ok = await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
|
||||
loaded_entries = [
|
||||
entry
|
||||
for entry in hass.config_entries.async_entries(DOMAIN)
|
||||
if entry.state is ConfigEntryState.LOADED
|
||||
]
|
||||
if len(loaded_entries) == 1:
|
||||
if not hass.config_entries.async_loaded_entries(DOMAIN):
|
||||
# If this is the last loaded instance of RainMachine, deregister any services
|
||||
# defined during integration setup:
|
||||
for service_name in (
|
||||
|
|
|
@ -39,7 +39,7 @@ from simplipy.websocket import (
|
|||
)
|
||||
import voluptuous as vol
|
||||
|
||||
from homeassistant.config_entries import ConfigEntry, ConfigEntryState
|
||||
from homeassistant.config_entries import ConfigEntry
|
||||
from homeassistant.const import (
|
||||
ATTR_CODE,
|
||||
ATTR_DEVICE_ID,
|
||||
|
@ -402,12 +402,7 @@ async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
|||
if unload_ok:
|
||||
hass.data[DOMAIN].pop(entry.entry_id)
|
||||
|
||||
loaded_entries = [
|
||||
entry
|
||||
for entry in hass.config_entries.async_entries(DOMAIN)
|
||||
if entry.state == ConfigEntryState.LOADED
|
||||
]
|
||||
if len(loaded_entries) == 1:
|
||||
if not hass.config_entries.async_loaded_entries(DOMAIN):
|
||||
# If this is the last loaded instance of SimpliSafe, deregister any services
|
||||
# defined during integration setup:
|
||||
for service_name in SERVICES:
|
||||
|
|
|
@ -11,7 +11,7 @@ from tplink_omada_client.exceptions import (
|
|||
UnsupportedControllerVersion,
|
||||
)
|
||||
|
||||
from homeassistant.config_entries import ConfigEntry, ConfigEntryState
|
||||
from homeassistant.config_entries import ConfigEntry
|
||||
from homeassistant.const import Platform
|
||||
from homeassistant.core import HomeAssistant, ServiceCall
|
||||
from homeassistant.exceptions import ConfigEntryAuthFailed, ConfigEntryNotReady
|
||||
|
@ -80,12 +80,7 @@ async def async_setup_entry(hass: HomeAssistant, entry: OmadaConfigEntry) -> boo
|
|||
async def async_unload_entry(hass: HomeAssistant, entry: OmadaConfigEntry) -> bool:
|
||||
"""Unload a config entry."""
|
||||
unload_ok = await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
|
||||
loaded_entries = [
|
||||
entry
|
||||
for entry in hass.config_entries.async_entries(DOMAIN)
|
||||
if entry.state == ConfigEntryState.LOADED
|
||||
]
|
||||
if len(loaded_entries) == 1:
|
||||
if not hass.config_entries.async_loaded_entries(DOMAIN):
|
||||
# This is the last loaded instance of Omada, deregister any services
|
||||
hass.services.async_remove(DOMAIN, "reconnect_client")
|
||||
|
||||
|
|
|
@ -7,7 +7,7 @@ import voluptuous as vol
|
|||
from xiaomi_gateway import AsyncXiaomiGatewayMulticast, XiaomiGateway
|
||||
|
||||
from homeassistant.components import persistent_notification
|
||||
from homeassistant.config_entries import ConfigEntry, ConfigEntryState
|
||||
from homeassistant.config_entries import ConfigEntry
|
||||
from homeassistant.const import (
|
||||
ATTR_DEVICE_ID,
|
||||
CONF_HOST,
|
||||
|
@ -216,12 +216,7 @@ async def async_unload_entry(hass: HomeAssistant, config_entry: ConfigEntry) ->
|
|||
if unload_ok:
|
||||
hass.data[DOMAIN][GATEWAYS_KEY].pop(config_entry.entry_id)
|
||||
|
||||
loaded_entries = [
|
||||
entry
|
||||
for entry in hass.config_entries.async_entries(DOMAIN)
|
||||
if entry.state == ConfigEntryState.LOADED
|
||||
]
|
||||
if len(loaded_entries) == 1:
|
||||
if not hass.config_entries.async_loaded_entries(DOMAIN):
|
||||
# No gateways left, stop Xiaomi socket
|
||||
unsub_stop = hass.data[DOMAIN].pop(KEY_UNSUB_STOP)
|
||||
unsub_stop()
|
||||
|
|
|
@ -155,6 +155,8 @@ class ConfigEntryState(Enum):
|
|||
"""An error occurred when trying to unload the entry"""
|
||||
SETUP_IN_PROGRESS = "setup_in_progress", False
|
||||
"""The config entry is setting up."""
|
||||
UNLOAD_IN_PROGRESS = "unload_in_progress", False
|
||||
"""The config entry is being unloaded."""
|
||||
|
||||
_recoverable: bool
|
||||
|
||||
|
@ -955,18 +957,25 @@ class ConfigEntry[_DataT = Any]:
|
|||
)
|
||||
return False
|
||||
|
||||
if domain_is_integration:
|
||||
self._async_set_state(hass, ConfigEntryState.UNLOAD_IN_PROGRESS, None)
|
||||
try:
|
||||
result = await component.async_unload_entry(hass, self)
|
||||
|
||||
assert isinstance(result, bool)
|
||||
|
||||
# Only adjust state if we unloaded the component
|
||||
if domain_is_integration and result:
|
||||
await self._async_process_on_unload(hass)
|
||||
if hasattr(self, "runtime_data"):
|
||||
object.__delattr__(self, "runtime_data")
|
||||
# Only do side effects if we unloaded the integration
|
||||
if domain_is_integration:
|
||||
if result:
|
||||
await self._async_process_on_unload(hass)
|
||||
if hasattr(self, "runtime_data"):
|
||||
object.__delattr__(self, "runtime_data")
|
||||
|
||||
self._async_set_state(hass, ConfigEntryState.NOT_LOADED, None)
|
||||
self._async_set_state(hass, ConfigEntryState.NOT_LOADED, None)
|
||||
else:
|
||||
self._async_set_state(
|
||||
hass, ConfigEntryState.FAILED_UNLOAD, "Unload failed"
|
||||
)
|
||||
|
||||
except Exception as exc:
|
||||
_LOGGER.exception(
|
||||
|
@ -2052,9 +2061,9 @@ class ConfigEntries:
|
|||
else:
|
||||
unload_success = await self.async_unload(entry_id, _lock=False)
|
||||
|
||||
del self._entries[entry.entry_id]
|
||||
await entry.async_remove(self.hass)
|
||||
|
||||
del self._entries[entry.entry_id]
|
||||
self.async_update_issues()
|
||||
self._async_schedule_save()
|
||||
|
||||
|
|
|
@ -502,7 +502,7 @@ async def test_issue_registry_invalid_version(
|
|||
("stop_addon_side_effect", "entry_state"),
|
||||
[
|
||||
(None, ConfigEntryState.NOT_LOADED),
|
||||
(SupervisorError("Boom"), ConfigEntryState.LOADED),
|
||||
(SupervisorError("Boom"), ConfigEntryState.FAILED_UNLOAD),
|
||||
],
|
||||
)
|
||||
async def test_stop_addon(
|
||||
|
|
|
@ -76,7 +76,7 @@ async def test_reset_fails(
|
|||
return_value=False,
|
||||
):
|
||||
assert not await hass.config_entries.async_unload(config_entry_setup.entry_id)
|
||||
assert config_entry_setup.state is ConfigEntryState.LOADED
|
||||
assert config_entry_setup.state is ConfigEntryState.FAILED_UNLOAD
|
||||
|
||||
|
||||
@pytest.mark.usefixtures("mock_device_registry")
|
||||
|
|
|
@ -847,7 +847,7 @@ async def test_issue_registry(
|
|||
("stop_addon_side_effect", "entry_state"),
|
||||
[
|
||||
(None, ConfigEntryState.NOT_LOADED),
|
||||
(SupervisorError("Boom"), ConfigEntryState.LOADED),
|
||||
(SupervisorError("Boom"), ConfigEntryState.FAILED_UNLOAD),
|
||||
],
|
||||
)
|
||||
async def test_stop_addon(
|
||||
|
|
|
@ -468,8 +468,8 @@ async def test_remove_entry(
|
|||
hass: HomeAssistant, entry: config_entries.ConfigEntry
|
||||
) -> None:
|
||||
"""Mock removing an entry."""
|
||||
# Check that the entry is not yet removed from config entries
|
||||
assert hass.config_entries.async_get_entry(entry.entry_id)
|
||||
# Check that the entry is no longer in the config entries
|
||||
assert not hass.config_entries.async_get_entry(entry.entry_id)
|
||||
remove_entry_calls.append(None)
|
||||
|
||||
entity = MockEntity(unique_id="1234", name="Test Entity")
|
||||
|
@ -2623,7 +2623,7 @@ async def test_entry_setup_invalid_state(
|
|||
("unload_result", "expected_result", "expected_state", "has_runtime_data"),
|
||||
[
|
||||
(True, True, config_entries.ConfigEntryState.NOT_LOADED, False),
|
||||
(False, False, config_entries.ConfigEntryState.LOADED, True),
|
||||
(False, False, config_entries.ConfigEntryState.FAILED_UNLOAD, True),
|
||||
],
|
||||
)
|
||||
async def test_entry_unload(
|
||||
|
@ -2648,7 +2648,7 @@ async def test_entry_unload(
|
|||
"""Mock unload entry."""
|
||||
unload_entry_calls.append(None)
|
||||
verify_runtime_data()
|
||||
assert entry.state is config_entries.ConfigEntryState.LOADED
|
||||
assert entry.state is config_entries.ConfigEntryState.UNLOAD_IN_PROGRESS
|
||||
return unload_result
|
||||
|
||||
entry = MockConfigEntry(domain="comp", state=config_entries.ConfigEntryState.LOADED)
|
||||
|
|
Loading…
Reference in New Issue