2019-06-12 16:29:28 +00:00
|
|
|
"""Manage config entries in Home Assistant."""
|
2019-03-01 04:27:20 +00:00
|
|
|
import asyncio
|
2019-02-15 17:30:47 +00:00
|
|
|
import functools
|
2019-12-09 15:42:10 +00:00
|
|
|
import logging
|
2020-03-09 21:07:50 +00:00
|
|
|
from types import MappingProxyType
|
2019-12-16 11:27:43 +00:00
|
|
|
from typing import Any, Callable, Dict, List, Optional, Set, Union, cast
|
2019-02-22 16:59:43 +00:00
|
|
|
import weakref
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-08-18 04:34:11 +00:00
|
|
|
import attr
|
|
|
|
|
2019-04-11 08:26:36 +00:00
|
|
|
from homeassistant import data_entry_flow, loader
|
2020-07-22 15:06:37 +00:00
|
|
|
from homeassistant.core import CALLBACK_TYPE, HomeAssistant, callback
|
2019-12-09 15:42:10 +00:00
|
|
|
from homeassistant.exceptions import ConfigEntryNotReady, HomeAssistantError
|
2019-08-23 00:32:43 +00:00
|
|
|
from homeassistant.helpers import entity_registry
|
2019-10-28 20:36:26 +00:00
|
|
|
from homeassistant.helpers.event import Event
|
2019-12-09 15:42:10 +00:00
|
|
|
from homeassistant.setup import async_process_deps_reqs, async_setup_component
|
|
|
|
from homeassistant.util.decorator import Registry
|
2020-08-24 15:21:30 +00:00
|
|
|
import homeassistant.util.uuid as uuid_util
|
2019-07-25 06:08:20 +00:00
|
|
|
|
2018-02-16 22:07:38 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2019-10-28 20:36:26 +00:00
|
|
|
_UNDEF: dict = {}
|
2018-08-09 11:24:14 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
SOURCE_DISCOVERY = "discovery"
|
2020-06-15 11:38:38 +00:00
|
|
|
SOURCE_HASSIO = "hassio"
|
|
|
|
SOURCE_HOMEKIT = "homekit"
|
2019-07-31 19:25:30 +00:00
|
|
|
SOURCE_IMPORT = "import"
|
2020-05-13 13:11:00 +00:00
|
|
|
SOURCE_INTEGRATION_DISCOVERY = "integration_discovery"
|
2019-10-29 06:32:57 +00:00
|
|
|
SOURCE_SSDP = "ssdp"
|
|
|
|
SOURCE_USER = "user"
|
|
|
|
SOURCE_ZEROCONF = "zeroconf"
|
2019-12-21 10:22:07 +00:00
|
|
|
|
|
|
|
# If a user wants to hide a discovery from the UI they can "Ignore" it. The config_entries/ignore_flow
|
|
|
|
# websocket command creates a config entry with this source and while it exists normal discoveries
|
|
|
|
# with the same unique id are ignored.
|
2019-12-18 06:41:01 +00:00
|
|
|
SOURCE_IGNORE = "ignore"
|
2018-08-09 11:24:14 +00:00
|
|
|
|
2019-12-21 10:22:07 +00:00
|
|
|
# This is used when a user uses the "Stop Ignoring" button in the UI (the
|
|
|
|
# config_entries/ignore_flow websocket command). It's triggered after the "ignore" config entry has
|
|
|
|
# been removed and unloaded.
|
|
|
|
SOURCE_UNIGNORE = "unignore"
|
|
|
|
|
2018-02-16 22:07:38 +00:00
|
|
|
HANDLERS = Registry()
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
STORAGE_KEY = "core.config_entries"
|
2018-06-25 16:53:49 +00:00
|
|
|
STORAGE_VERSION = 1
|
|
|
|
|
|
|
|
# Deprecated since 0.73
|
2019-07-31 19:25:30 +00:00
|
|
|
PATH_CONFIG = ".config_entries.json"
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
SAVE_DELAY = 1
|
|
|
|
|
2018-10-04 13:53:50 +00:00
|
|
|
# The config entry has been set up successfully
|
2019-07-31 19:25:30 +00:00
|
|
|
ENTRY_STATE_LOADED = "loaded"
|
2018-10-04 13:53:50 +00:00
|
|
|
# There was an error while trying to set up this config entry
|
2019-07-31 19:25:30 +00:00
|
|
|
ENTRY_STATE_SETUP_ERROR = "setup_error"
|
2019-02-15 17:30:47 +00:00
|
|
|
# There was an error while trying to migrate the config entry to a new version
|
2019-07-31 19:25:30 +00:00
|
|
|
ENTRY_STATE_MIGRATION_ERROR = "migration_error"
|
2018-10-04 13:53:50 +00:00
|
|
|
# The config entry was not ready to be set up yet, but might be later
|
2019-07-31 19:25:30 +00:00
|
|
|
ENTRY_STATE_SETUP_RETRY = "setup_retry"
|
2018-10-04 13:53:50 +00:00
|
|
|
# The config entry has not been loaded
|
2019-07-31 19:25:30 +00:00
|
|
|
ENTRY_STATE_NOT_LOADED = "not_loaded"
|
2018-10-04 13:53:50 +00:00
|
|
|
# An error occurred when trying to unload the entry
|
2019-07-31 19:25:30 +00:00
|
|
|
ENTRY_STATE_FAILED_UNLOAD = "failed_unload"
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
UNRECOVERABLE_STATES = (ENTRY_STATE_MIGRATION_ERROR, ENTRY_STATE_FAILED_UNLOAD)
|
2019-03-01 04:27:20 +00:00
|
|
|
|
2020-06-15 11:38:38 +00:00
|
|
|
DEFAULT_DISCOVERY_UNIQUE_ID = "default_discovery_unique_id"
|
2019-07-31 19:25:30 +00:00
|
|
|
DISCOVERY_NOTIFICATION_ID = "config_entry_discovery"
|
2020-01-03 16:28:05 +00:00
|
|
|
DISCOVERY_SOURCES = (
|
|
|
|
SOURCE_SSDP,
|
|
|
|
SOURCE_ZEROCONF,
|
|
|
|
SOURCE_DISCOVERY,
|
|
|
|
SOURCE_IMPORT,
|
|
|
|
SOURCE_UNIGNORE,
|
|
|
|
)
|
2018-04-22 19:00:24 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
EVENT_FLOW_DISCOVERED = "config_entry_discovered"
|
2018-06-18 03:03:29 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
CONN_CLASS_CLOUD_PUSH = "cloud_push"
|
|
|
|
CONN_CLASS_CLOUD_POLL = "cloud_poll"
|
|
|
|
CONN_CLASS_LOCAL_PUSH = "local_push"
|
|
|
|
CONN_CLASS_LOCAL_POLL = "local_poll"
|
|
|
|
CONN_CLASS_ASSUMED = "assumed"
|
|
|
|
CONN_CLASS_UNKNOWN = "unknown"
|
2018-09-17 08:12:46 +00:00
|
|
|
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-03-01 04:27:20 +00:00
|
|
|
class ConfigError(HomeAssistantError):
|
|
|
|
"""Error while configuring an account."""
|
|
|
|
|
|
|
|
|
|
|
|
class UnknownEntry(ConfigError):
|
|
|
|
"""Unknown entry specified."""
|
|
|
|
|
|
|
|
|
|
|
|
class OperationNotAllowed(ConfigError):
|
|
|
|
"""Raised when a config entry operation is not allowed."""
|
|
|
|
|
|
|
|
|
2020-07-22 15:06:37 +00:00
|
|
|
UpdateListenerType = Callable[[HomeAssistant, "ConfigEntry"], Any]
|
|
|
|
|
|
|
|
|
2018-02-16 22:07:38 +00:00
|
|
|
class ConfigEntry:
|
|
|
|
"""Hold a configuration entry."""
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
__slots__ = (
|
|
|
|
"entry_id",
|
|
|
|
"version",
|
|
|
|
"domain",
|
|
|
|
"title",
|
|
|
|
"data",
|
|
|
|
"options",
|
2019-12-16 11:27:43 +00:00
|
|
|
"unique_id",
|
2020-08-25 22:59:22 +00:00
|
|
|
"supports_unload",
|
2019-08-18 04:34:11 +00:00
|
|
|
"system_options",
|
2019-07-31 19:25:30 +00:00
|
|
|
"source",
|
|
|
|
"connection_class",
|
|
|
|
"state",
|
|
|
|
"_setup_lock",
|
|
|
|
"update_listeners",
|
|
|
|
"_async_cancel_retry_setup",
|
|
|
|
)
|
|
|
|
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
version: int,
|
|
|
|
domain: str,
|
|
|
|
title: str,
|
|
|
|
data: dict,
|
|
|
|
source: str,
|
|
|
|
connection_class: str,
|
2019-08-18 04:34:11 +00:00
|
|
|
system_options: dict,
|
2019-07-31 19:25:30 +00:00
|
|
|
options: Optional[dict] = None,
|
2019-12-16 11:27:43 +00:00
|
|
|
unique_id: Optional[str] = None,
|
2019-07-31 19:25:30 +00:00
|
|
|
entry_id: Optional[str] = None,
|
|
|
|
state: str = ENTRY_STATE_NOT_LOADED,
|
|
|
|
) -> None:
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Initialize a config entry."""
|
|
|
|
# Unique id of the config entry
|
2020-08-24 15:21:30 +00:00
|
|
|
self.entry_id = entry_id or uuid_util.uuid_v1mc_hex()
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
# Version of the configuration.
|
|
|
|
self.version = version
|
|
|
|
|
|
|
|
# Domain the configuration belongs to
|
|
|
|
self.domain = domain
|
|
|
|
|
|
|
|
# Title of the configuration
|
|
|
|
self.title = title
|
|
|
|
|
|
|
|
# Config data
|
2020-03-09 21:07:50 +00:00
|
|
|
self.data = MappingProxyType(data)
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-02-22 16:59:43 +00:00
|
|
|
# Entry options
|
2020-03-09 21:07:50 +00:00
|
|
|
self.options = MappingProxyType(options or {})
|
2019-02-22 16:59:43 +00:00
|
|
|
|
2019-08-18 04:34:11 +00:00
|
|
|
# Entry system options
|
|
|
|
self.system_options = SystemOptions(**system_options)
|
|
|
|
|
2018-02-16 22:07:38 +00:00
|
|
|
# Source of the configuration (user, discovery, cloud)
|
|
|
|
self.source = source
|
|
|
|
|
2018-09-17 08:12:46 +00:00
|
|
|
# Connection class
|
|
|
|
self.connection_class = connection_class
|
|
|
|
|
2018-02-16 22:07:38 +00:00
|
|
|
# State of the entry (LOADED, NOT_LOADED)
|
|
|
|
self.state = state
|
|
|
|
|
2019-12-16 11:27:43 +00:00
|
|
|
# Unique ID of this entry.
|
|
|
|
self.unique_id = unique_id
|
|
|
|
|
2020-08-25 22:59:22 +00:00
|
|
|
# Supports unload
|
|
|
|
self.supports_unload = False
|
|
|
|
|
2019-02-22 16:59:43 +00:00
|
|
|
# Listeners to call on update
|
2020-07-22 15:06:37 +00:00
|
|
|
self.update_listeners: List[weakref.ReferenceType[UpdateListenerType]] = []
|
2019-02-22 16:59:43 +00:00
|
|
|
|
2018-10-04 13:53:50 +00:00
|
|
|
# Function to cancel a scheduled retry
|
2019-09-04 03:36:04 +00:00
|
|
|
self._async_cancel_retry_setup: Optional[Callable[[], Any]] = None
|
2018-10-04 13:53:50 +00:00
|
|
|
|
2018-07-23 08:24:39 +00:00
|
|
|
async def async_setup(
|
2019-07-31 19:25:30 +00:00
|
|
|
self,
|
|
|
|
hass: HomeAssistant,
|
|
|
|
*,
|
|
|
|
integration: Optional[loader.Integration] = None,
|
|
|
|
tries: int = 0,
|
|
|
|
) -> None:
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Set up an entry."""
|
2019-12-18 06:41:01 +00:00
|
|
|
if self.source == SOURCE_IGNORE:
|
|
|
|
return
|
|
|
|
|
2019-04-15 02:07:05 +00:00
|
|
|
if integration is None:
|
|
|
|
integration = await loader.async_get_integration(hass, self.domain)
|
|
|
|
|
2020-08-25 22:59:22 +00:00
|
|
|
self.supports_unload = await support_entry_unload(hass, self.domain)
|
|
|
|
|
2019-05-13 08:16:55 +00:00
|
|
|
try:
|
|
|
|
component = integration.get_component()
|
|
|
|
except ImportError as err:
|
|
|
|
_LOGGER.error(
|
2020-02-13 16:27:00 +00:00
|
|
|
"Error importing integration %s to set up %s configuration entry: %s",
|
2019-07-31 19:25:30 +00:00
|
|
|
integration.domain,
|
|
|
|
self.domain,
|
|
|
|
err,
|
|
|
|
)
|
2019-05-13 08:16:55 +00:00
|
|
|
if self.domain == integration.domain:
|
|
|
|
self.state = ENTRY_STATE_SETUP_ERROR
|
|
|
|
return
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-08-23 00:32:43 +00:00
|
|
|
if self.domain == integration.domain:
|
|
|
|
try:
|
|
|
|
integration.get_platform("config_flow")
|
|
|
|
except ImportError as err:
|
|
|
|
_LOGGER.error(
|
2020-02-13 16:27:00 +00:00
|
|
|
"Error importing platform config_flow from integration %s to set up %s configuration entry: %s",
|
2019-08-23 00:32:43 +00:00
|
|
|
integration.domain,
|
|
|
|
self.domain,
|
|
|
|
err,
|
|
|
|
)
|
|
|
|
self.state = ENTRY_STATE_SETUP_ERROR
|
|
|
|
return
|
|
|
|
|
|
|
|
# Perform migration
|
2019-02-15 17:30:47 +00:00
|
|
|
if not await self.async_migrate(hass):
|
|
|
|
self.state = ENTRY_STATE_MIGRATION_ERROR
|
|
|
|
return
|
|
|
|
|
2018-02-16 22:07:38 +00:00
|
|
|
try:
|
2020-08-27 11:56:20 +00:00
|
|
|
result = await component.async_setup_entry(hass, self) # type: ignore
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
if not isinstance(result, bool):
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.error(
|
|
|
|
"%s.async_setup_entry did not return boolean", integration.domain
|
|
|
|
)
|
2018-02-16 22:07:38 +00:00
|
|
|
result = False
|
2018-10-04 13:53:50 +00:00
|
|
|
except ConfigEntryNotReady:
|
|
|
|
self.state = ENTRY_STATE_SETUP_RETRY
|
2019-07-31 19:25:30 +00:00
|
|
|
wait_time = 2 ** min(tries, 4) * 5
|
2018-10-04 13:53:50 +00:00
|
|
|
tries += 1
|
|
|
|
_LOGGER.warning(
|
2020-07-05 21:04:19 +00:00
|
|
|
"Config entry for %s not ready yet. Retrying in %d seconds",
|
2019-07-31 19:25:30 +00:00
|
|
|
self.domain,
|
|
|
|
wait_time,
|
|
|
|
)
|
2018-10-04 13:53:50 +00:00
|
|
|
|
2019-10-28 20:36:26 +00:00
|
|
|
async def setup_again(now: Any) -> None:
|
2018-10-04 13:53:50 +00:00
|
|
|
"""Run setup again."""
|
|
|
|
self._async_cancel_retry_setup = None
|
2019-07-31 19:25:30 +00:00
|
|
|
await self.async_setup(hass, integration=integration, tries=tries)
|
2018-10-04 13:53:50 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
self._async_cancel_retry_setup = hass.helpers.event.async_call_later(
|
|
|
|
wait_time, setup_again
|
|
|
|
)
|
2018-10-04 13:53:50 +00:00
|
|
|
return
|
2018-02-16 22:07:38 +00:00
|
|
|
except Exception: # pylint: disable=broad-except
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.exception(
|
|
|
|
"Error setting up entry %s for %s", self.title, integration.domain
|
|
|
|
)
|
2018-02-16 22:07:38 +00:00
|
|
|
result = False
|
|
|
|
|
2018-04-09 14:09:08 +00:00
|
|
|
# Only store setup result as state if it was not forwarded.
|
2019-04-15 02:07:05 +00:00
|
|
|
if self.domain != integration.domain:
|
2018-04-09 14:09:08 +00:00
|
|
|
return
|
|
|
|
|
2018-02-16 22:07:38 +00:00
|
|
|
if result:
|
|
|
|
self.state = ENTRY_STATE_LOADED
|
|
|
|
else:
|
|
|
|
self.state = ENTRY_STATE_SETUP_ERROR
|
|
|
|
|
2019-07-20 21:35:59 +00:00
|
|
|
async def async_unload(
|
2019-07-31 19:25:30 +00:00
|
|
|
self, hass: HomeAssistant, *, integration: Optional[loader.Integration] = None
|
|
|
|
) -> bool:
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Unload an entry.
|
|
|
|
|
|
|
|
Returns if unload is possible and was successful.
|
|
|
|
"""
|
2019-12-20 20:49:07 +00:00
|
|
|
if self.source == SOURCE_IGNORE:
|
|
|
|
self.state = ENTRY_STATE_NOT_LOADED
|
|
|
|
return True
|
|
|
|
|
2019-04-15 02:07:05 +00:00
|
|
|
if integration is None:
|
2020-05-25 19:40:06 +00:00
|
|
|
try:
|
|
|
|
integration = await loader.async_get_integration(hass, self.domain)
|
|
|
|
except loader.IntegrationNotFound:
|
|
|
|
# The integration was likely a custom_component
|
|
|
|
# that was uninstalled, or an integration
|
|
|
|
# that has been renamed without removing the config
|
|
|
|
# entry.
|
|
|
|
self.state = ENTRY_STATE_NOT_LOADED
|
|
|
|
return True
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-04-15 02:07:05 +00:00
|
|
|
component = integration.get_component()
|
|
|
|
|
|
|
|
if integration.domain == self.domain:
|
2019-03-01 04:27:20 +00:00
|
|
|
if self.state in UNRECOVERABLE_STATES:
|
|
|
|
return False
|
2018-10-04 13:53:50 +00:00
|
|
|
|
|
|
|
if self.state != ENTRY_STATE_LOADED:
|
2019-03-01 04:27:20 +00:00
|
|
|
if self._async_cancel_retry_setup is not None:
|
|
|
|
self._async_cancel_retry_setup()
|
|
|
|
self._async_cancel_retry_setup = None
|
|
|
|
|
|
|
|
self.state = ENTRY_STATE_NOT_LOADED
|
2018-10-04 13:53:50 +00:00
|
|
|
return True
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
supports_unload = hasattr(component, "async_unload_entry")
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
if not supports_unload:
|
2019-04-15 02:07:05 +00:00
|
|
|
if integration.domain == self.domain:
|
2019-03-01 04:27:20 +00:00
|
|
|
self.state = ENTRY_STATE_FAILED_UNLOAD
|
2018-02-16 22:07:38 +00:00
|
|
|
return False
|
|
|
|
|
|
|
|
try:
|
2020-08-27 11:56:20 +00:00
|
|
|
result = await component.async_unload_entry(hass, self) # type: ignore
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2018-10-04 13:53:50 +00:00
|
|
|
assert isinstance(result, bool)
|
|
|
|
|
|
|
|
# Only adjust state if we unloaded the component
|
2019-04-15 02:07:05 +00:00
|
|
|
if result and integration.domain == self.domain:
|
2018-10-04 13:53:50 +00:00
|
|
|
self.state = ENTRY_STATE_NOT_LOADED
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
return result
|
|
|
|
except Exception: # pylint: disable=broad-except
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.exception(
|
|
|
|
"Error unloading entry %s for %s", self.title, integration.domain
|
|
|
|
)
|
2019-04-15 02:07:05 +00:00
|
|
|
if integration.domain == self.domain:
|
2018-10-04 13:53:50 +00:00
|
|
|
self.state = ENTRY_STATE_FAILED_UNLOAD
|
2018-02-16 22:07:38 +00:00
|
|
|
return False
|
|
|
|
|
2019-03-02 05:13:55 +00:00
|
|
|
async def async_remove(self, hass: HomeAssistant) -> None:
|
|
|
|
"""Invoke remove callback on component."""
|
2019-12-20 20:49:07 +00:00
|
|
|
if self.source == SOURCE_IGNORE:
|
|
|
|
return
|
|
|
|
|
2020-05-25 19:40:06 +00:00
|
|
|
try:
|
|
|
|
integration = await loader.async_get_integration(hass, self.domain)
|
|
|
|
except loader.IntegrationNotFound:
|
|
|
|
# The integration was likely a custom_component
|
|
|
|
# that was uninstalled, or an integration
|
|
|
|
# that has been renamed without removing the config
|
|
|
|
# entry.
|
|
|
|
return
|
|
|
|
|
2019-04-15 02:07:05 +00:00
|
|
|
component = integration.get_component()
|
2019-07-31 19:25:30 +00:00
|
|
|
if not hasattr(component, "async_remove_entry"):
|
2019-03-02 05:13:55 +00:00
|
|
|
return
|
|
|
|
try:
|
2020-08-27 11:56:20 +00:00
|
|
|
await component.async_remove_entry(hass, self) # type: ignore
|
2019-03-02 05:13:55 +00:00
|
|
|
except Exception: # pylint: disable=broad-except
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.exception(
|
|
|
|
"Error calling entry remove callback %s for %s",
|
|
|
|
self.title,
|
|
|
|
integration.domain,
|
|
|
|
)
|
2019-03-02 05:13:55 +00:00
|
|
|
|
2019-02-15 17:30:47 +00:00
|
|
|
async def async_migrate(self, hass: HomeAssistant) -> bool:
|
|
|
|
"""Migrate an entry.
|
|
|
|
|
|
|
|
Returns True if config entry is up-to-date or has been migrated.
|
|
|
|
"""
|
|
|
|
handler = HANDLERS.get(self.domain)
|
|
|
|
if handler is None:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.error(
|
|
|
|
"Flow handler not found for entry %s for %s", self.title, self.domain
|
|
|
|
)
|
2019-02-15 17:30:47 +00:00
|
|
|
return False
|
|
|
|
# Handler may be a partial
|
2019-07-07 01:58:33 +00:00
|
|
|
while isinstance(handler, functools.partial):
|
2019-02-15 17:30:47 +00:00
|
|
|
handler = handler.func
|
|
|
|
|
|
|
|
if self.version == handler.VERSION:
|
|
|
|
return True
|
|
|
|
|
2019-05-13 08:16:55 +00:00
|
|
|
integration = await loader.async_get_integration(hass, self.domain)
|
|
|
|
component = integration.get_component()
|
2019-07-31 19:25:30 +00:00
|
|
|
supports_migrate = hasattr(component, "async_migrate_entry")
|
2019-02-15 17:30:47 +00:00
|
|
|
if not supports_migrate:
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.error(
|
|
|
|
"Migration handler not found for entry %s for %s",
|
|
|
|
self.title,
|
|
|
|
self.domain,
|
|
|
|
)
|
2019-02-15 17:30:47 +00:00
|
|
|
return False
|
|
|
|
|
|
|
|
try:
|
2020-08-27 11:56:20 +00:00
|
|
|
result = await component.async_migrate_entry(hass, self) # type: ignore
|
2019-02-15 17:30:47 +00:00
|
|
|
if not isinstance(result, bool):
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.error(
|
|
|
|
"%s.async_migrate_entry did not return boolean", self.domain
|
|
|
|
)
|
2019-02-15 17:30:47 +00:00
|
|
|
return False
|
|
|
|
if result:
|
|
|
|
# pylint: disable=protected-access
|
2019-10-18 20:06:33 +00:00
|
|
|
hass.config_entries._async_schedule_save()
|
2019-02-15 17:30:47 +00:00
|
|
|
return result
|
|
|
|
except Exception: # pylint: disable=broad-except
|
2019-07-31 19:25:30 +00:00
|
|
|
_LOGGER.exception(
|
|
|
|
"Error migrating entry %s for %s", self.title, self.domain
|
|
|
|
)
|
2019-02-15 17:30:47 +00:00
|
|
|
return False
|
|
|
|
|
2020-07-22 15:06:37 +00:00
|
|
|
def add_update_listener(self, listener: UpdateListenerType) -> CALLBACK_TYPE:
|
2019-02-22 16:59:43 +00:00
|
|
|
"""Listen for when entry is updated.
|
|
|
|
|
|
|
|
Returns function to unlisten.
|
|
|
|
"""
|
|
|
|
weak_listener = weakref.ref(listener)
|
|
|
|
self.update_listeners.append(weak_listener)
|
|
|
|
|
|
|
|
return lambda: self.update_listeners.remove(weak_listener)
|
|
|
|
|
2019-10-28 20:36:26 +00:00
|
|
|
def as_dict(self) -> Dict[str, Any]:
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Return dictionary version of this entry."""
|
|
|
|
return {
|
2019-07-31 19:25:30 +00:00
|
|
|
"entry_id": self.entry_id,
|
|
|
|
"version": self.version,
|
|
|
|
"domain": self.domain,
|
|
|
|
"title": self.title,
|
2020-03-09 21:07:50 +00:00
|
|
|
"data": dict(self.data),
|
|
|
|
"options": dict(self.options),
|
2019-08-18 04:34:11 +00:00
|
|
|
"system_options": self.system_options.as_dict(),
|
2019-07-31 19:25:30 +00:00
|
|
|
"source": self.source,
|
|
|
|
"connection_class": self.connection_class,
|
2019-12-16 18:45:09 +00:00
|
|
|
"unique_id": self.unique_id,
|
2018-02-16 22:07:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-01-03 10:52:01 +00:00
|
|
|
class ConfigEntriesFlowManager(data_entry_flow.FlowManager):
|
|
|
|
"""Manage all the config entry flows that are in progress."""
|
|
|
|
|
|
|
|
def __init__(
|
|
|
|
self, hass: HomeAssistant, config_entries: "ConfigEntries", hass_config: dict
|
|
|
|
):
|
|
|
|
"""Initialize the config entry flow manager."""
|
|
|
|
super().__init__(hass)
|
|
|
|
self.config_entries = config_entries
|
|
|
|
self._hass_config = hass_config
|
|
|
|
|
|
|
|
async def async_finish_flow(
|
|
|
|
self, flow: data_entry_flow.FlowHandler, result: Dict[str, Any]
|
|
|
|
) -> Dict[str, Any]:
|
|
|
|
"""Finish a config flow and add an entry."""
|
|
|
|
flow = cast(ConfigFlow, flow)
|
|
|
|
|
|
|
|
# Remove notification if no other discovery config entries in progress
|
|
|
|
if not any(
|
|
|
|
ent["context"]["source"] in DISCOVERY_SOURCES
|
|
|
|
for ent in self.hass.config_entries.flow.async_progress()
|
|
|
|
if ent["flow_id"] != flow.flow_id
|
|
|
|
):
|
|
|
|
self.hass.components.persistent_notification.async_dismiss(
|
|
|
|
DISCOVERY_NOTIFICATION_ID
|
|
|
|
)
|
|
|
|
|
|
|
|
if result["type"] != data_entry_flow.RESULT_TYPE_CREATE_ENTRY:
|
|
|
|
return result
|
|
|
|
|
|
|
|
# Check if config entry exists with unique ID. Unload it.
|
|
|
|
existing_entry = None
|
|
|
|
|
|
|
|
if flow.unique_id is not None:
|
|
|
|
# Abort all flows in progress with same unique ID.
|
|
|
|
for progress_flow in self.async_progress():
|
|
|
|
if (
|
|
|
|
progress_flow["handler"] == flow.handler
|
|
|
|
and progress_flow["flow_id"] != flow.flow_id
|
|
|
|
and progress_flow["context"].get("unique_id") == flow.unique_id
|
|
|
|
):
|
|
|
|
self.async_abort(progress_flow["flow_id"])
|
|
|
|
|
2020-06-15 11:38:38 +00:00
|
|
|
# Reset unique ID when the default discovery ID has been used
|
|
|
|
if flow.unique_id == DEFAULT_DISCOVERY_UNIQUE_ID:
|
|
|
|
await flow.async_set_unique_id(None)
|
|
|
|
|
2020-01-03 10:52:01 +00:00
|
|
|
# Find existing entry.
|
|
|
|
for check_entry in self.config_entries.async_entries(result["handler"]):
|
|
|
|
if check_entry.unique_id == flow.unique_id:
|
|
|
|
existing_entry = check_entry
|
|
|
|
break
|
|
|
|
|
|
|
|
# Unload the entry before setting up the new one.
|
|
|
|
# We will remove it only after the other one is set up,
|
|
|
|
# so that device customizations are not getting lost.
|
|
|
|
if (
|
|
|
|
existing_entry is not None
|
|
|
|
and existing_entry.state not in UNRECOVERABLE_STATES
|
|
|
|
):
|
|
|
|
await self.config_entries.async_unload(existing_entry.entry_id)
|
|
|
|
|
|
|
|
entry = ConfigEntry(
|
|
|
|
version=result["version"],
|
|
|
|
domain=result["handler"],
|
|
|
|
title=result["title"],
|
|
|
|
data=result["data"],
|
|
|
|
options={},
|
|
|
|
system_options={},
|
|
|
|
source=flow.context["source"],
|
|
|
|
connection_class=flow.CONNECTION_CLASS,
|
|
|
|
unique_id=flow.unique_id,
|
|
|
|
)
|
|
|
|
|
|
|
|
await self.config_entries.async_add(entry)
|
|
|
|
|
|
|
|
if existing_entry is not None:
|
|
|
|
await self.config_entries.async_remove(existing_entry.entry_id)
|
|
|
|
|
|
|
|
result["result"] = entry
|
|
|
|
return result
|
|
|
|
|
|
|
|
async def async_create_flow(
|
|
|
|
self, handler_key: Any, *, context: Optional[Dict] = None, data: Any = None
|
|
|
|
) -> "ConfigFlow":
|
|
|
|
"""Create a flow for specified handler.
|
|
|
|
|
|
|
|
Handler key is the domain of the component that we want to set up.
|
|
|
|
"""
|
|
|
|
try:
|
|
|
|
integration = await loader.async_get_integration(self.hass, handler_key)
|
2020-08-28 11:50:32 +00:00
|
|
|
except loader.IntegrationNotFound as err:
|
2020-01-03 10:52:01 +00:00
|
|
|
_LOGGER.error("Cannot find integration %s", handler_key)
|
2020-08-28 11:50:32 +00:00
|
|
|
raise data_entry_flow.UnknownHandler from err
|
2020-01-03 10:52:01 +00:00
|
|
|
|
|
|
|
# Make sure requirements and dependencies of component are resolved
|
|
|
|
await async_process_deps_reqs(self.hass, self._hass_config, integration)
|
|
|
|
|
|
|
|
try:
|
|
|
|
integration.get_platform("config_flow")
|
|
|
|
except ImportError as err:
|
|
|
|
_LOGGER.error(
|
2020-02-13 16:27:00 +00:00
|
|
|
"Error occurred loading configuration flow for integration %s: %s",
|
2020-01-03 10:52:01 +00:00
|
|
|
handler_key,
|
|
|
|
err,
|
|
|
|
)
|
|
|
|
raise data_entry_flow.UnknownHandler
|
|
|
|
|
|
|
|
handler = HANDLERS.get(handler_key)
|
|
|
|
|
|
|
|
if handler is None:
|
|
|
|
raise data_entry_flow.UnknownHandler
|
|
|
|
|
|
|
|
if not context or "source" not in context:
|
|
|
|
raise KeyError("Context not set or doesn't have a source set")
|
|
|
|
|
2020-01-03 16:28:05 +00:00
|
|
|
flow = cast(ConfigFlow, handler())
|
|
|
|
flow.init_step = context["source"]
|
|
|
|
return flow
|
|
|
|
|
|
|
|
async def async_post_init(
|
|
|
|
self, flow: data_entry_flow.FlowHandler, result: dict
|
|
|
|
) -> None:
|
|
|
|
"""After a flow is initialised trigger new flow notifications."""
|
|
|
|
source = flow.context["source"]
|
2020-01-03 10:52:01 +00:00
|
|
|
|
|
|
|
# Create notification.
|
|
|
|
if source in DISCOVERY_SOURCES:
|
|
|
|
self.hass.bus.async_fire(EVENT_FLOW_DISCOVERED)
|
|
|
|
self.hass.components.persistent_notification.async_create(
|
|
|
|
title="New devices discovered",
|
|
|
|
message=(
|
|
|
|
"We have discovered new devices on your network. "
|
|
|
|
"[Check it out](/config/integrations)"
|
|
|
|
),
|
|
|
|
notification_id=DISCOVERY_NOTIFICATION_ID,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2018-02-16 22:07:38 +00:00
|
|
|
class ConfigEntries:
|
|
|
|
"""Manage the configuration entries.
|
|
|
|
|
|
|
|
An instance of this object is available via `hass.config_entries`.
|
|
|
|
"""
|
|
|
|
|
2018-07-31 14:00:17 +00:00
|
|
|
def __init__(self, hass: HomeAssistant, hass_config: dict) -> None:
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Initialize the entry manager."""
|
|
|
|
self.hass = hass
|
2020-01-03 10:52:01 +00:00
|
|
|
self.flow = ConfigEntriesFlowManager(hass, self, hass_config)
|
2019-02-22 16:59:43 +00:00
|
|
|
self.options = OptionsFlowManager(hass)
|
2018-02-16 22:07:38 +00:00
|
|
|
self._hass_config = hass_config
|
2019-09-04 03:36:04 +00:00
|
|
|
self._entries: List[ConfigEntry] = []
|
2018-06-25 16:53:49 +00:00
|
|
|
self._store = hass.helpers.storage.Store(STORAGE_VERSION, STORAGE_KEY)
|
2019-08-23 00:32:43 +00:00
|
|
|
EntityRegistryDisabledHandler(hass).async_setup()
|
2018-02-16 22:07:38 +00:00
|
|
|
|
|
|
|
@callback
|
2018-07-31 14:00:17 +00:00
|
|
|
def async_domains(self) -> List[str]:
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Return domains for which we have entries."""
|
2019-09-04 03:36:04 +00:00
|
|
|
seen: Set[str] = set()
|
2018-02-16 22:07:38 +00:00
|
|
|
result = []
|
|
|
|
|
|
|
|
for entry in self._entries:
|
|
|
|
if entry.domain not in seen:
|
|
|
|
seen.add(entry.domain)
|
|
|
|
result.append(entry.domain)
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
2019-02-22 16:59:43 +00:00
|
|
|
@callback
|
|
|
|
def async_get_entry(self, entry_id: str) -> Optional[ConfigEntry]:
|
|
|
|
"""Return entry with matching entry_id."""
|
|
|
|
for entry in self._entries:
|
|
|
|
if entry_id == entry.entry_id:
|
|
|
|
return entry
|
|
|
|
return None
|
|
|
|
|
2018-02-16 22:07:38 +00:00
|
|
|
@callback
|
2018-08-17 18:22:49 +00:00
|
|
|
def async_entries(self, domain: Optional[str] = None) -> List[ConfigEntry]:
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Return all entries or entries for a specific domain."""
|
|
|
|
if domain is None:
|
|
|
|
return list(self._entries)
|
|
|
|
return [entry for entry in self._entries if entry.domain == domain]
|
|
|
|
|
2020-01-03 10:52:01 +00:00
|
|
|
async def async_add(self, entry: ConfigEntry) -> None:
|
|
|
|
"""Add and setup an entry."""
|
|
|
|
self._entries.append(entry)
|
|
|
|
await self.async_setup(entry.entry_id)
|
|
|
|
self._async_schedule_save()
|
|
|
|
|
2019-10-28 20:36:26 +00:00
|
|
|
async def async_remove(self, entry_id: str) -> Dict[str, Any]:
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Remove an entry."""
|
2019-03-01 04:27:20 +00:00
|
|
|
entry = self.async_get_entry(entry_id)
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-03-01 04:27:20 +00:00
|
|
|
if entry is None:
|
2018-02-16 22:07:38 +00:00
|
|
|
raise UnknownEntry
|
|
|
|
|
2019-03-01 04:27:20 +00:00
|
|
|
if entry.state in UNRECOVERABLE_STATES:
|
|
|
|
unload_success = entry.state != ENTRY_STATE_FAILED_UNLOAD
|
|
|
|
else:
|
|
|
|
unload_success = await self.async_unload(entry_id)
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-03-02 05:13:55 +00:00
|
|
|
await entry.async_remove(self.hass)
|
|
|
|
|
2019-03-01 04:27:20 +00:00
|
|
|
self._entries.remove(entry)
|
|
|
|
self._async_schedule_save()
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-03-01 04:27:20 +00:00
|
|
|
dev_reg, ent_reg = await asyncio.gather(
|
|
|
|
self.hass.helpers.device_registry.async_get_registry(),
|
|
|
|
self.hass.helpers.entity_registry.async_get_registry(),
|
|
|
|
)
|
2018-09-04 07:00:14 +00:00
|
|
|
|
2019-03-01 04:27:20 +00:00
|
|
|
dev_reg.async_clear_config_entry(entry_id)
|
|
|
|
ent_reg.async_clear_config_entry(entry_id)
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-12-21 10:22:07 +00:00
|
|
|
# After we have fully removed an "ignore" config entry we can try and rediscover it so that a
|
|
|
|
# user is able to immediately start configuring it. We do this by starting a new flow with
|
|
|
|
# the 'unignore' step. If the integration doesn't implement async_step_unignore then
|
|
|
|
# this will be a no-op.
|
|
|
|
if entry.source == SOURCE_IGNORE:
|
|
|
|
self.hass.async_create_task(
|
|
|
|
self.hass.config_entries.flow.async_init(
|
|
|
|
entry.domain,
|
|
|
|
context={"source": SOURCE_UNIGNORE},
|
|
|
|
data={"unique_id": entry.unique_id},
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
return {"require_restart": not unload_success}
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-03-01 04:27:20 +00:00
|
|
|
async def async_initialize(self) -> None:
|
|
|
|
"""Initialize config entry config."""
|
2018-06-25 16:53:49 +00:00
|
|
|
# Migrating for config entries stored before 0.73
|
|
|
|
config = await self.hass.helpers.storage.async_migrator(
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass.config.path(PATH_CONFIG),
|
|
|
|
self._store,
|
|
|
|
old_conf_migrate_func=_old_conf_migrator,
|
2018-06-25 16:53:49 +00:00
|
|
|
)
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2018-06-25 21:21:38 +00:00
|
|
|
if config is None:
|
|
|
|
self._entries = []
|
|
|
|
return
|
|
|
|
|
2018-09-17 08:12:46 +00:00
|
|
|
self._entries = [
|
|
|
|
ConfigEntry(
|
2019-07-31 19:25:30 +00:00
|
|
|
version=entry["version"],
|
|
|
|
domain=entry["domain"],
|
|
|
|
entry_id=entry["entry_id"],
|
|
|
|
data=entry["data"],
|
|
|
|
source=entry["source"],
|
|
|
|
title=entry["title"],
|
2018-09-17 08:12:46 +00:00
|
|
|
# New in 0.79
|
2019-07-31 19:25:30 +00:00
|
|
|
connection_class=entry.get("connection_class", CONN_CLASS_UNKNOWN),
|
2019-02-22 16:59:43 +00:00
|
|
|
# New in 0.89
|
2019-07-31 19:25:30 +00:00
|
|
|
options=entry.get("options"),
|
2019-08-18 04:34:11 +00:00
|
|
|
# New in 0.98
|
|
|
|
system_options=entry.get("system_options", {}),
|
2019-12-16 18:45:09 +00:00
|
|
|
# New in 0.104
|
|
|
|
unique_id=entry.get("unique_id"),
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
for entry in config["entries"]
|
|
|
|
]
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-03-01 04:27:20 +00:00
|
|
|
async def async_setup(self, entry_id: str) -> bool:
|
|
|
|
"""Set up a config entry.
|
|
|
|
|
|
|
|
Return True if entry has been successfully loaded.
|
|
|
|
"""
|
|
|
|
entry = self.async_get_entry(entry_id)
|
|
|
|
|
|
|
|
if entry is None:
|
|
|
|
raise UnknownEntry
|
|
|
|
|
|
|
|
if entry.state != ENTRY_STATE_NOT_LOADED:
|
|
|
|
raise OperationNotAllowed
|
|
|
|
|
|
|
|
# Setup Component if not set up yet
|
|
|
|
if entry.domain in self.hass.config.components:
|
|
|
|
await entry.async_setup(self.hass)
|
|
|
|
else:
|
|
|
|
# Setting up the component will set up all its config entries
|
|
|
|
result = await async_setup_component(
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass, entry.domain, self._hass_config
|
|
|
|
)
|
2019-03-01 04:27:20 +00:00
|
|
|
|
|
|
|
if not result:
|
|
|
|
return result
|
|
|
|
|
|
|
|
return entry.state == ENTRY_STATE_LOADED
|
|
|
|
|
|
|
|
async def async_unload(self, entry_id: str) -> bool:
|
|
|
|
"""Unload a config entry."""
|
|
|
|
entry = self.async_get_entry(entry_id)
|
|
|
|
|
|
|
|
if entry is None:
|
|
|
|
raise UnknownEntry
|
|
|
|
|
|
|
|
if entry.state in UNRECOVERABLE_STATES:
|
|
|
|
raise OperationNotAllowed
|
|
|
|
|
|
|
|
return await entry.async_unload(self.hass)
|
|
|
|
|
|
|
|
async def async_reload(self, entry_id: str) -> bool:
|
|
|
|
"""Reload an entry.
|
|
|
|
|
|
|
|
If an entry was not loaded, will just load.
|
|
|
|
"""
|
|
|
|
unload_result = await self.async_unload(entry_id)
|
|
|
|
|
|
|
|
if not unload_result:
|
|
|
|
return unload_result
|
|
|
|
|
|
|
|
return await self.async_setup(entry_id)
|
|
|
|
|
2018-09-25 10:21:11 +00:00
|
|
|
@callback
|
2019-08-19 23:45:17 +00:00
|
|
|
def async_update_entry(
|
2019-10-28 20:36:26 +00:00
|
|
|
self,
|
|
|
|
entry: ConfigEntry,
|
|
|
|
*,
|
2020-08-29 05:59:24 +00:00
|
|
|
# pylint: disable=dangerous-default-value # _UNDEFs not modified
|
2019-12-16 11:27:43 +00:00
|
|
|
unique_id: Union[str, dict, None] = _UNDEF,
|
2020-03-09 21:07:50 +00:00
|
|
|
title: Union[str, dict] = _UNDEF,
|
2019-10-28 20:36:26 +00:00
|
|
|
data: dict = _UNDEF,
|
|
|
|
options: dict = _UNDEF,
|
|
|
|
system_options: dict = _UNDEF,
|
2020-08-08 18:23:56 +00:00
|
|
|
) -> bool:
|
|
|
|
"""Update a config entry.
|
|
|
|
|
|
|
|
If the entry was changed, the update_listeners are
|
|
|
|
fired and this function returns True
|
|
|
|
|
|
|
|
If the entry was not changed, the update_listeners are
|
|
|
|
not fired and this function returns False
|
|
|
|
"""
|
|
|
|
changed = False
|
|
|
|
|
|
|
|
if unique_id is not _UNDEF and entry.unique_id != unique_id:
|
|
|
|
changed = True
|
2019-12-16 11:27:43 +00:00
|
|
|
entry.unique_id = cast(Optional[str], unique_id)
|
|
|
|
|
2020-08-08 18:23:56 +00:00
|
|
|
if title is not _UNDEF and entry.title != title:
|
|
|
|
changed = True
|
2020-03-09 21:07:50 +00:00
|
|
|
entry.title = cast(str, title)
|
|
|
|
|
2020-08-08 18:23:56 +00:00
|
|
|
if data is not _UNDEF and entry.data != data: # type: ignore
|
|
|
|
changed = True
|
2020-03-09 21:07:50 +00:00
|
|
|
entry.data = MappingProxyType(data)
|
2019-02-22 16:59:43 +00:00
|
|
|
|
2020-08-08 18:23:56 +00:00
|
|
|
if options is not _UNDEF and entry.options != options: # type: ignore
|
|
|
|
changed = True
|
2020-03-09 21:07:50 +00:00
|
|
|
entry.options = MappingProxyType(options)
|
2019-02-22 16:59:43 +00:00
|
|
|
|
2020-08-08 18:23:56 +00:00
|
|
|
if (
|
|
|
|
system_options is not _UNDEF
|
|
|
|
and entry.system_options.as_dict() != system_options
|
|
|
|
):
|
|
|
|
changed = True
|
2019-08-19 23:45:17 +00:00
|
|
|
entry.system_options.update(**system_options)
|
|
|
|
|
2020-08-08 18:23:56 +00:00
|
|
|
if not changed:
|
|
|
|
return False
|
|
|
|
|
2019-08-19 23:45:17 +00:00
|
|
|
for listener_ref in entry.update_listeners:
|
|
|
|
listener = listener_ref()
|
2020-07-22 15:06:37 +00:00
|
|
|
if listener is not None:
|
|
|
|
self.hass.async_create_task(listener(self.hass, entry))
|
2019-02-22 16:59:43 +00:00
|
|
|
|
2018-09-25 10:21:11 +00:00
|
|
|
self._async_schedule_save()
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2020-08-08 18:23:56 +00:00
|
|
|
return True
|
|
|
|
|
2019-10-28 20:36:26 +00:00
|
|
|
async def async_forward_entry_setup(self, entry: ConfigEntry, domain: str) -> bool:
|
2018-04-09 14:09:08 +00:00
|
|
|
"""Forward the setup of an entry to a different component.
|
|
|
|
|
|
|
|
By default an entry is setup with the component it belongs to. If that
|
|
|
|
component also has related platforms, the component will have to
|
|
|
|
forward the entry to be setup by that component.
|
|
|
|
|
|
|
|
You don't want to await this coroutine if it is called as part of the
|
|
|
|
setup of a component, because it can cause a deadlock.
|
|
|
|
"""
|
|
|
|
# Setup Component if not set up yet
|
2019-04-15 02:07:05 +00:00
|
|
|
if domain not in self.hass.config.components:
|
2019-07-31 19:25:30 +00:00
|
|
|
result = await async_setup_component(self.hass, domain, self._hass_config)
|
2018-04-09 14:09:08 +00:00
|
|
|
|
|
|
|
if not result:
|
|
|
|
return False
|
|
|
|
|
2019-04-15 02:07:05 +00:00
|
|
|
integration = await loader.async_get_integration(self.hass, domain)
|
|
|
|
|
|
|
|
await entry.async_setup(self.hass, integration=integration)
|
2019-10-28 20:36:26 +00:00
|
|
|
return True
|
2018-04-09 14:09:08 +00:00
|
|
|
|
2019-10-28 20:36:26 +00:00
|
|
|
async def async_forward_entry_unload(self, entry: ConfigEntry, domain: str) -> bool:
|
2018-04-12 12:28:54 +00:00
|
|
|
"""Forward the unloading of an entry to a different component."""
|
|
|
|
# It was never loaded.
|
2019-04-15 02:07:05 +00:00
|
|
|
if domain not in self.hass.config.components:
|
2018-04-12 12:28:54 +00:00
|
|
|
return True
|
|
|
|
|
2019-04-15 02:07:05 +00:00
|
|
|
integration = await loader.async_get_integration(self.hass, domain)
|
|
|
|
|
|
|
|
return await entry.async_unload(self.hass, integration=integration)
|
2018-04-12 12:28:54 +00:00
|
|
|
|
2020-02-14 18:00:22 +00:00
|
|
|
@callback
|
2019-02-22 16:59:43 +00:00
|
|
|
def _async_schedule_save(self) -> None:
|
2018-02-16 22:07:38 +00:00
|
|
|
"""Save the entity registry to a file."""
|
2018-08-17 18:18:21 +00:00
|
|
|
self._store.async_delay_save(self._data_to_save, SAVE_DELAY)
|
|
|
|
|
|
|
|
@callback
|
2019-10-28 20:36:26 +00:00
|
|
|
def _data_to_save(self) -> Dict[str, List[Dict[str, Any]]]:
|
2018-08-17 18:18:21 +00:00
|
|
|
"""Return data to save."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return {"entries": [entry.as_dict() for entry in self._entries]}
|
2018-06-25 16:53:49 +00:00
|
|
|
|
2018-02-16 22:07:38 +00:00
|
|
|
|
2019-10-28 20:36:26 +00:00
|
|
|
async def _old_conf_migrator(old_config: Dict[str, Any]) -> Dict[str, Any]:
|
2018-06-25 16:53:49 +00:00
|
|
|
"""Migrate the pre-0.73 config format to the latest version."""
|
2019-07-31 19:25:30 +00:00
|
|
|
return {"entries": old_config}
|
2018-09-14 09:57:31 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ConfigFlow(data_entry_flow.FlowHandler):
|
|
|
|
"""Base class for config flows with some helpers."""
|
|
|
|
|
2019-10-19 18:35:57 +00:00
|
|
|
def __init_subclass__(cls, domain: Optional[str] = None, **kwargs: Any) -> None:
|
2019-08-20 17:46:51 +00:00
|
|
|
"""Initialize a subclass, register if possible."""
|
|
|
|
super().__init_subclass__(**kwargs) # type: ignore
|
|
|
|
if domain is not None:
|
|
|
|
HANDLERS.register(domain)(cls)
|
|
|
|
|
2018-09-17 08:12:46 +00:00
|
|
|
CONNECTION_CLASS = CONN_CLASS_UNKNOWN
|
|
|
|
|
2019-12-16 18:45:09 +00:00
|
|
|
@property
|
|
|
|
def unique_id(self) -> Optional[str]:
|
|
|
|
"""Return unique ID if available."""
|
|
|
|
# pylint: disable=no-member
|
|
|
|
if not self.context:
|
|
|
|
return None
|
|
|
|
|
|
|
|
return cast(Optional[str], self.context.get("unique_id"))
|
|
|
|
|
2019-08-15 21:11:55 +00:00
|
|
|
@staticmethod
|
|
|
|
@callback
|
2019-10-28 20:36:26 +00:00
|
|
|
def async_get_options_flow(config_entry: ConfigEntry) -> "OptionsFlow":
|
2019-08-15 21:11:55 +00:00
|
|
|
"""Get the options flow for this handler."""
|
|
|
|
raise data_entry_flow.UnknownHandler
|
|
|
|
|
2019-12-16 18:45:09 +00:00
|
|
|
@callback
|
2020-03-09 21:07:50 +00:00
|
|
|
def _abort_if_unique_id_configured(
|
2020-08-27 11:56:20 +00:00
|
|
|
self,
|
|
|
|
updates: Optional[Dict[Any, Any]] = None,
|
|
|
|
reload_on_update: bool = True,
|
2020-03-09 21:07:50 +00:00
|
|
|
) -> None:
|
2019-12-16 18:45:09 +00:00
|
|
|
"""Abort if the unique ID is already configured."""
|
2020-01-23 19:21:19 +00:00
|
|
|
assert self.hass
|
2019-12-16 18:45:09 +00:00
|
|
|
if self.unique_id is None:
|
|
|
|
return
|
|
|
|
|
2020-01-23 19:21:19 +00:00
|
|
|
for entry in self._async_current_entries():
|
|
|
|
if entry.unique_id == self.unique_id:
|
2020-08-08 18:23:56 +00:00
|
|
|
if updates is not None:
|
|
|
|
changed = self.hass.config_entries.async_update_entry(
|
2020-01-23 19:21:19 +00:00
|
|
|
entry, data={**entry.data, **updates}
|
|
|
|
)
|
2020-08-24 08:54:26 +00:00
|
|
|
if (
|
|
|
|
changed
|
|
|
|
and reload_on_update
|
|
|
|
and entry.state in (ENTRY_STATE_LOADED, ENTRY_STATE_SETUP_RETRY)
|
|
|
|
):
|
2020-08-08 18:23:56 +00:00
|
|
|
self.hass.async_create_task(
|
|
|
|
self.hass.config_entries.async_reload(entry.entry_id)
|
|
|
|
)
|
2020-01-23 19:21:19 +00:00
|
|
|
raise data_entry_flow.AbortFlow("already_configured")
|
2019-12-16 18:45:09 +00:00
|
|
|
|
2019-12-16 11:27:43 +00:00
|
|
|
async def async_set_unique_id(
|
2020-06-15 11:38:38 +00:00
|
|
|
self, unique_id: Optional[str] = None, *, raise_on_progress: bool = True
|
2019-12-16 11:27:43 +00:00
|
|
|
) -> Optional[ConfigEntry]:
|
|
|
|
"""Set a unique ID for the config flow.
|
|
|
|
|
|
|
|
Returns optionally existing config entry with same ID.
|
|
|
|
"""
|
2020-06-15 11:38:38 +00:00
|
|
|
if unique_id is None:
|
|
|
|
self.context["unique_id"] = None # pylint: disable=no-member
|
|
|
|
return None
|
|
|
|
|
2019-12-16 11:27:43 +00:00
|
|
|
if raise_on_progress:
|
|
|
|
for progress in self._async_in_progress():
|
|
|
|
if progress["context"].get("unique_id") == unique_id:
|
2019-12-16 18:45:09 +00:00
|
|
|
raise data_entry_flow.AbortFlow("already_in_progress")
|
2019-12-16 11:27:43 +00:00
|
|
|
|
2020-05-09 11:08:40 +00:00
|
|
|
self.context["unique_id"] = unique_id # pylint: disable=no-member
|
2019-12-16 11:27:43 +00:00
|
|
|
|
2020-06-15 11:38:38 +00:00
|
|
|
# Abort discoveries done using the default discovery unique id
|
|
|
|
assert self.hass is not None
|
|
|
|
if unique_id != DEFAULT_DISCOVERY_UNIQUE_ID:
|
|
|
|
for progress in self._async_in_progress():
|
|
|
|
if progress["context"].get("unique_id") == DEFAULT_DISCOVERY_UNIQUE_ID:
|
|
|
|
self.hass.config_entries.flow.async_abort(progress["flow_id"])
|
|
|
|
|
2019-12-16 11:27:43 +00:00
|
|
|
for entry in self._async_current_entries():
|
|
|
|
if entry.unique_id == unique_id:
|
|
|
|
return entry
|
|
|
|
|
|
|
|
return None
|
|
|
|
|
2018-09-14 09:57:31 +00:00
|
|
|
@callback
|
2019-10-28 20:36:26 +00:00
|
|
|
def _async_current_entries(self) -> List[ConfigEntry]:
|
2018-09-14 09:57:31 +00:00
|
|
|
"""Return current entries."""
|
2019-10-28 20:36:26 +00:00
|
|
|
assert self.hass is not None
|
2018-09-14 09:57:31 +00:00
|
|
|
return self.hass.config_entries.async_entries(self.handler)
|
|
|
|
|
2019-12-16 18:45:09 +00:00
|
|
|
@callback
|
2019-12-18 06:41:01 +00:00
|
|
|
def _async_current_ids(self, include_ignore: bool = True) -> Set[Optional[str]]:
|
2019-12-16 18:45:09 +00:00
|
|
|
"""Return current unique IDs."""
|
|
|
|
assert self.hass is not None
|
2020-04-04 18:05:15 +00:00
|
|
|
return {
|
2019-12-16 18:45:09 +00:00
|
|
|
entry.unique_id
|
|
|
|
for entry in self.hass.config_entries.async_entries(self.handler)
|
2019-12-18 06:41:01 +00:00
|
|
|
if include_ignore or entry.source != SOURCE_IGNORE
|
2020-04-04 18:05:15 +00:00
|
|
|
}
|
2019-12-16 18:45:09 +00:00
|
|
|
|
2018-09-14 09:57:31 +00:00
|
|
|
@callback
|
2019-10-28 20:36:26 +00:00
|
|
|
def _async_in_progress(self) -> List[Dict]:
|
2018-09-14 09:57:31 +00:00
|
|
|
"""Return other in progress flows for current domain."""
|
2019-10-28 20:36:26 +00:00
|
|
|
assert self.hass is not None
|
2019-07-31 19:25:30 +00:00
|
|
|
return [
|
|
|
|
flw
|
|
|
|
for flw in self.hass.config_entries.flow.async_progress()
|
|
|
|
if flw["handler"] == self.handler and flw["flow_id"] != self.flow_id
|
|
|
|
]
|
2019-02-22 16:59:43 +00:00
|
|
|
|
2019-12-18 06:41:01 +00:00
|
|
|
async def async_step_ignore(self, user_input: Dict[str, Any]) -> Dict[str, Any]:
|
|
|
|
"""Ignore this config flow."""
|
|
|
|
await self.async_set_unique_id(user_input["unique_id"], raise_on_progress=False)
|
|
|
|
return self.async_create_entry(title="Ignored", data={})
|
|
|
|
|
2019-12-21 10:22:07 +00:00
|
|
|
async def async_step_unignore(self, user_input: Dict[str, Any]) -> Dict[str, Any]:
|
|
|
|
"""Rediscover a config entry by it's unique_id."""
|
|
|
|
return self.async_abort(reason="not_implemented")
|
|
|
|
|
2020-06-15 11:38:38 +00:00
|
|
|
async def async_step_user(
|
|
|
|
self, user_input: Optional[Dict[str, Any]] = None
|
|
|
|
) -> Dict[str, Any]:
|
|
|
|
"""Handle a flow initiated by the user."""
|
|
|
|
return self.async_abort(reason="not_implemented")
|
|
|
|
|
|
|
|
async def _async_handle_discovery_without_unique_id(self) -> None:
|
|
|
|
"""Mark this flow discovered, without a unique identifier.
|
|
|
|
|
|
|
|
If a flow initiated by discovery, doesn't have a unique ID, this can
|
|
|
|
be used alternatively. It will ensure only 1 flow is started and only
|
|
|
|
when the handler has no existing config entries.
|
|
|
|
|
|
|
|
It ensures that the discovery can be ignored by the user.
|
|
|
|
"""
|
|
|
|
if self.unique_id is not None:
|
|
|
|
return
|
|
|
|
|
|
|
|
# Abort if the handler has config entries already
|
|
|
|
if self._async_current_entries():
|
|
|
|
raise data_entry_flow.AbortFlow("already_configured")
|
|
|
|
|
|
|
|
# Use an special unique id to differentiate
|
|
|
|
await self.async_set_unique_id(DEFAULT_DISCOVERY_UNIQUE_ID)
|
|
|
|
self._abort_if_unique_id_configured()
|
|
|
|
|
|
|
|
# Abort if any other flow for this handler is already in progress
|
|
|
|
assert self.hass is not None
|
|
|
|
if self._async_in_progress():
|
|
|
|
raise data_entry_flow.AbortFlow("already_in_progress")
|
|
|
|
|
|
|
|
async def async_step_discovery(
|
|
|
|
self, discovery_info: Dict[str, Any]
|
|
|
|
) -> Dict[str, Any]:
|
|
|
|
"""Handle a flow initialized by discovery."""
|
|
|
|
await self._async_handle_discovery_without_unique_id()
|
|
|
|
return await self.async_step_user()
|
|
|
|
|
|
|
|
async_step_hassio = async_step_discovery
|
|
|
|
async_step_homekit = async_step_discovery
|
|
|
|
async_step_ssdp = async_step_discovery
|
|
|
|
async_step_zeroconf = async_step_discovery
|
|
|
|
|
2019-02-22 16:59:43 +00:00
|
|
|
|
2020-01-03 10:52:01 +00:00
|
|
|
class OptionsFlowManager(data_entry_flow.FlowManager):
|
2019-02-22 16:59:43 +00:00
|
|
|
"""Flow to set options for a configuration entry."""
|
|
|
|
|
2020-01-03 10:52:01 +00:00
|
|
|
async def async_create_flow(
|
|
|
|
self,
|
|
|
|
handler_key: Any,
|
|
|
|
*,
|
|
|
|
context: Optional[Dict[str, Any]] = None,
|
|
|
|
data: Optional[Dict[str, Any]] = None,
|
|
|
|
) -> "OptionsFlow":
|
2019-02-22 16:59:43 +00:00
|
|
|
"""Create an options flow for a config entry.
|
|
|
|
|
|
|
|
Entry_id and flow.handler is the same thing to map entry with flow.
|
|
|
|
"""
|
2020-01-03 10:52:01 +00:00
|
|
|
entry = self.hass.config_entries.async_get_entry(handler_key)
|
2019-02-22 16:59:43 +00:00
|
|
|
if entry is None:
|
2020-01-03 10:52:01 +00:00
|
|
|
raise UnknownEntry(handler_key)
|
2019-08-15 21:11:55 +00:00
|
|
|
|
|
|
|
if entry.domain not in HANDLERS:
|
|
|
|
raise data_entry_flow.UnknownHandler
|
|
|
|
|
2020-04-06 10:51:48 +00:00
|
|
|
return cast(OptionsFlow, HANDLERS[entry.domain].async_get_options_flow(entry))
|
2019-02-22 16:59:43 +00:00
|
|
|
|
2020-01-03 10:52:01 +00:00
|
|
|
async def async_finish_flow(
|
|
|
|
self, flow: data_entry_flow.FlowHandler, result: Dict[str, Any]
|
|
|
|
) -> Dict[str, Any]:
|
2019-02-22 16:59:43 +00:00
|
|
|
"""Finish an options flow and update options for configuration entry.
|
|
|
|
|
|
|
|
Flow.handler and entry_id is the same thing to map flow with entry.
|
|
|
|
"""
|
2020-01-03 10:52:01 +00:00
|
|
|
flow = cast(OptionsFlow, flow)
|
|
|
|
|
2019-02-22 16:59:43 +00:00
|
|
|
entry = self.hass.config_entries.async_get_entry(flow.handler)
|
|
|
|
if entry is None:
|
2020-01-03 10:52:01 +00:00
|
|
|
raise UnknownEntry(flow.handler)
|
2020-06-23 00:49:01 +00:00
|
|
|
if result["data"] is not None:
|
|
|
|
self.hass.config_entries.async_update_entry(entry, options=result["data"])
|
2019-02-22 16:59:43 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
result["result"] = True
|
2019-02-22 16:59:43 +00:00
|
|
|
return result
|
2019-08-15 21:11:55 +00:00
|
|
|
|
|
|
|
|
|
|
|
class OptionsFlow(data_entry_flow.FlowHandler):
|
|
|
|
"""Base class for config option flows."""
|
|
|
|
|
2019-10-28 20:36:26 +00:00
|
|
|
handler: str
|
2019-08-18 04:34:11 +00:00
|
|
|
|
|
|
|
|
|
|
|
@attr.s(slots=True)
|
|
|
|
class SystemOptions:
|
|
|
|
"""Config entry system options."""
|
|
|
|
|
2020-07-14 17:30:30 +00:00
|
|
|
disable_new_entities: bool = attr.ib(default=False)
|
2019-08-18 04:34:11 +00:00
|
|
|
|
2019-10-28 20:36:26 +00:00
|
|
|
def update(self, *, disable_new_entities: bool) -> None:
|
2019-08-18 04:34:11 +00:00
|
|
|
"""Update properties."""
|
|
|
|
self.disable_new_entities = disable_new_entities
|
|
|
|
|
2019-10-28 20:36:26 +00:00
|
|
|
def as_dict(self) -> Dict[str, Any]:
|
2020-01-31 16:33:00 +00:00
|
|
|
"""Return dictionary version of this config entries system options."""
|
2019-08-18 04:34:11 +00:00
|
|
|
return {"disable_new_entities": self.disable_new_entities}
|
2019-08-23 00:32:43 +00:00
|
|
|
|
|
|
|
|
|
|
|
class EntityRegistryDisabledHandler:
|
|
|
|
"""Handler to handle when entities related to config entries updating disabled_by."""
|
|
|
|
|
|
|
|
RELOAD_AFTER_UPDATE_DELAY = 30
|
|
|
|
|
|
|
|
def __init__(self, hass: HomeAssistant) -> None:
|
|
|
|
"""Initialize the handler."""
|
|
|
|
self.hass = hass
|
|
|
|
self.registry: Optional[entity_registry.EntityRegistry] = None
|
|
|
|
self.changed: Set[str] = set()
|
|
|
|
self._remove_call_later: Optional[Callable[[], None]] = None
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_setup(self) -> None:
|
|
|
|
"""Set up the disable handler."""
|
|
|
|
self.hass.bus.async_listen(
|
|
|
|
entity_registry.EVENT_ENTITY_REGISTRY_UPDATED, self._handle_entry_updated
|
|
|
|
)
|
|
|
|
|
2019-10-28 20:36:26 +00:00
|
|
|
async def _handle_entry_updated(self, event: Event) -> None:
|
2019-08-23 00:32:43 +00:00
|
|
|
"""Handle entity registry entry update."""
|
|
|
|
if (
|
|
|
|
event.data["action"] != "update"
|
|
|
|
or "disabled_by" not in event.data["changes"]
|
|
|
|
):
|
|
|
|
return
|
|
|
|
|
|
|
|
if self.registry is None:
|
|
|
|
self.registry = await entity_registry.async_get_registry(self.hass)
|
|
|
|
|
|
|
|
entity_entry = self.registry.async_get(event.data["entity_id"])
|
|
|
|
|
|
|
|
if (
|
|
|
|
# Stop if no entry found
|
|
|
|
entity_entry is None
|
|
|
|
# Stop if entry not connected to config entry
|
|
|
|
or entity_entry.config_entry_id is None
|
|
|
|
# Stop if the entry got disabled. In that case the entity handles it
|
|
|
|
# themselves.
|
|
|
|
or entity_entry.disabled_by
|
|
|
|
):
|
|
|
|
return
|
|
|
|
|
|
|
|
config_entry = self.hass.config_entries.async_get_entry(
|
|
|
|
entity_entry.config_entry_id
|
|
|
|
)
|
2019-10-28 20:36:26 +00:00
|
|
|
assert config_entry is not None
|
2019-08-23 00:32:43 +00:00
|
|
|
|
2020-08-25 22:59:22 +00:00
|
|
|
if config_entry.entry_id not in self.changed and config_entry.supports_unload:
|
2019-08-23 00:32:43 +00:00
|
|
|
self.changed.add(config_entry.entry_id)
|
|
|
|
|
|
|
|
if not self.changed:
|
|
|
|
return
|
|
|
|
|
|
|
|
# We are going to delay reloading on *every* entity registry change so that
|
|
|
|
# if a user is happily clicking along, it will only reload at the end.
|
|
|
|
|
|
|
|
if self._remove_call_later:
|
|
|
|
self._remove_call_later()
|
|
|
|
|
|
|
|
self._remove_call_later = self.hass.helpers.event.async_call_later(
|
|
|
|
self.RELOAD_AFTER_UPDATE_DELAY, self._handle_reload
|
|
|
|
)
|
|
|
|
|
2019-10-28 20:36:26 +00:00
|
|
|
async def _handle_reload(self, _now: Any) -> None:
|
2019-08-23 00:32:43 +00:00
|
|
|
"""Handle a reload."""
|
|
|
|
self._remove_call_later = None
|
|
|
|
to_reload = self.changed
|
|
|
|
self.changed = set()
|
|
|
|
|
|
|
|
_LOGGER.info(
|
2020-02-13 16:27:00 +00:00
|
|
|
"Reloading configuration entries because disabled_by changed in entity registry: %s",
|
2019-08-23 00:32:43 +00:00
|
|
|
", ".join(self.changed),
|
|
|
|
)
|
|
|
|
|
|
|
|
await asyncio.gather(
|
|
|
|
*[self.hass.config_entries.async_reload(entry_id) for entry_id in to_reload]
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
async def support_entry_unload(hass: HomeAssistant, domain: str) -> bool:
|
|
|
|
"""Test if a domain supports entry unloading."""
|
|
|
|
integration = await loader.async_get_integration(hass, domain)
|
|
|
|
component = integration.get_component()
|
|
|
|
return hasattr(component, "async_unload_entry")
|