2014-11-05 07:34:19 +00:00
|
|
|
"""
|
2018-01-21 06:35:38 +00:00
|
|
|
The methods for loading Home Assistant components.
|
2014-11-15 07:17:18 +00:00
|
|
|
|
|
|
|
This module has quite some complex parts. I have tried to add as much
|
|
|
|
documentation as possible to keep it understandable.
|
|
|
|
|
2017-10-08 15:17:54 +00:00
|
|
|
Components can be accessed via hass.components.switch from your code.
|
2014-11-15 07:17:18 +00:00
|
|
|
If you want to retrieve a platform that is part of a component, you should
|
2018-05-01 18:57:30 +00:00
|
|
|
call get_component(hass, 'switch.your_platform'). In both cases the config
|
|
|
|
directory is checked to see if it contains a user provided version. If not
|
|
|
|
available it will check the built-in components and platforms.
|
2014-11-05 07:34:19 +00:00
|
|
|
"""
|
2017-07-16 16:23:06 +00:00
|
|
|
import functools as ft
|
2014-11-05 07:34:19 +00:00
|
|
|
import importlib
|
|
|
|
import logging
|
2016-02-19 05:27:50 +00:00
|
|
|
import sys
|
2016-07-28 03:33:49 +00:00
|
|
|
from types import ModuleType
|
2019-02-21 08:41:36 +00:00
|
|
|
from typing import Optional, Set, TYPE_CHECKING, Callable, Any, TypeVar, List # noqa pylint: disable=unused-import
|
2016-07-28 03:33:49 +00:00
|
|
|
|
2016-04-04 19:18:58 +00:00
|
|
|
from homeassistant.const import PLATFORM_FORMAT
|
2014-11-28 23:34:42 +00:00
|
|
|
|
2018-07-13 10:24:51 +00:00
|
|
|
# Typing imports that create a circular dependency
|
|
|
|
# pylint: disable=using-constant-test,unused-import
|
|
|
|
if TYPE_CHECKING:
|
|
|
|
from homeassistant.core import HomeAssistant # NOQA
|
|
|
|
|
2018-07-26 06:55:42 +00:00
|
|
|
CALLABLE_T = TypeVar('CALLABLE_T', bound=Callable) # noqa pylint: disable=invalid-name
|
2018-07-23 08:24:39 +00:00
|
|
|
|
2014-11-28 23:34:42 +00:00
|
|
|
PREPARED = False
|
|
|
|
|
2018-07-13 10:24:51 +00:00
|
|
|
DEPENDENCY_BLACKLIST = {'config'}
|
2017-02-18 19:31:37 +00:00
|
|
|
|
2014-11-05 07:34:19 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2018-05-01 18:57:30 +00:00
|
|
|
DATA_KEY = 'components'
|
2019-02-21 08:41:36 +00:00
|
|
|
PACKAGE_CUSTOM_COMPONENTS = 'custom_components'
|
|
|
|
PACKAGE_BUILTIN = 'homeassistant.components'
|
|
|
|
LOOKUP_PATHS = [PACKAGE_CUSTOM_COMPONENTS, PACKAGE_BUILTIN]
|
2019-03-30 00:04:59 +00:00
|
|
|
COMPONENTS_WITH_BAD_PLATFORMS = ['automation', 'mqtt', 'telegram_bot']
|
2016-10-27 07:16:23 +00:00
|
|
|
|
2014-11-05 15:56:36 +00:00
|
|
|
|
2019-02-07 21:56:40 +00:00
|
|
|
class LoaderError(Exception):
|
|
|
|
"""Loader base error."""
|
|
|
|
|
|
|
|
|
|
|
|
class ComponentNotFound(LoaderError):
|
|
|
|
"""Raised when a component is not found."""
|
|
|
|
|
|
|
|
def __init__(self, domain: str) -> None:
|
|
|
|
"""Initialize a component not found error."""
|
|
|
|
super().__init__("Component {} not found.".format(domain))
|
|
|
|
self.domain = domain
|
|
|
|
|
|
|
|
|
|
|
|
class CircularDependency(LoaderError):
|
|
|
|
"""Raised when a circular dependency is found when resolving components."""
|
|
|
|
|
|
|
|
def __init__(self, from_domain: str, to_domain: str) -> None:
|
|
|
|
"""Initialize circular dependency error."""
|
|
|
|
super().__init__("Circular dependency detected: {} -> {}.".format(
|
|
|
|
from_domain, to_domain))
|
|
|
|
self.from_domain = from_domain
|
|
|
|
self.to_domain = to_domain
|
|
|
|
|
|
|
|
|
2018-07-13 10:24:51 +00:00
|
|
|
def set_component(hass, # type: HomeAssistant
|
|
|
|
comp_name: str, component: Optional[ModuleType]) -> None:
|
2016-10-27 07:16:23 +00:00
|
|
|
"""Set a component in the cache.
|
|
|
|
|
|
|
|
Async friendly.
|
|
|
|
"""
|
2019-02-07 21:33:12 +00:00
|
|
|
cache = hass.data.setdefault(DATA_KEY, {})
|
2018-05-01 18:57:30 +00:00
|
|
|
cache[comp_name] = component
|
2014-11-28 23:34:42 +00:00
|
|
|
|
2014-11-25 08:20:36 +00:00
|
|
|
|
2018-07-23 08:24:39 +00:00
|
|
|
def get_platform(hass, # type: HomeAssistant
|
2019-01-11 19:30:22 +00:00
|
|
|
domain: str, platform_name: str) -> Optional[ModuleType]:
|
2016-10-27 07:16:23 +00:00
|
|
|
"""Try to load specified platform.
|
|
|
|
|
2019-02-21 08:41:36 +00:00
|
|
|
Example invocation: get_platform(hass, 'light', 'hue')
|
|
|
|
|
2016-10-27 07:16:23 +00:00
|
|
|
Async friendly.
|
|
|
|
"""
|
2019-02-21 08:41:36 +00:00
|
|
|
# If the platform has a component, we will limit the platform loading path
|
|
|
|
# to be the same source (custom/built-in).
|
2019-03-30 00:04:59 +00:00
|
|
|
if domain not in COMPONENTS_WITH_BAD_PLATFORMS:
|
2019-03-28 10:09:12 +00:00
|
|
|
component = _load_file(hass, platform_name, LOOKUP_PATHS)
|
|
|
|
else:
|
|
|
|
# avoid load component for legacy platform
|
|
|
|
component = None
|
2019-02-21 08:41:36 +00:00
|
|
|
|
|
|
|
# Until we have moved all platforms under their component/own folder, it
|
|
|
|
# can be that the component is None.
|
|
|
|
if component is not None:
|
|
|
|
base_paths = [component.__name__.rsplit('.', 1)[0]]
|
|
|
|
else:
|
|
|
|
base_paths = LOOKUP_PATHS
|
|
|
|
|
|
|
|
platform = _load_file(
|
|
|
|
hass, PLATFORM_FORMAT.format(domain=domain, platform=platform_name),
|
|
|
|
base_paths)
|
2019-01-11 19:30:22 +00:00
|
|
|
|
2019-02-07 21:33:12 +00:00
|
|
|
if platform is not None:
|
|
|
|
return platform
|
|
|
|
|
2019-03-28 10:09:12 +00:00
|
|
|
# Legacy platform check for automation: components/automation/event.py
|
2019-03-30 00:04:59 +00:00
|
|
|
if component is None and domain in COMPONENTS_WITH_BAD_PLATFORMS:
|
2019-03-28 10:09:12 +00:00
|
|
|
platform = _load_file(
|
|
|
|
hass,
|
|
|
|
PLATFORM_FORMAT.format(domain=platform_name, platform=domain),
|
|
|
|
base_paths
|
|
|
|
)
|
|
|
|
|
2019-03-28 03:52:28 +00:00
|
|
|
# Legacy platform check for custom: custom_components/light/hue.py
|
|
|
|
# Only check if the component was also in custom components.
|
|
|
|
if component is None or base_paths[0] == PACKAGE_CUSTOM_COMPONENTS:
|
|
|
|
platform = _load_file(
|
|
|
|
hass,
|
|
|
|
PLATFORM_FORMAT.format(domain=platform_name, platform=domain),
|
|
|
|
[PACKAGE_CUSTOM_COMPONENTS]
|
|
|
|
)
|
2019-01-11 19:30:22 +00:00
|
|
|
|
|
|
|
if platform is None:
|
2019-02-21 08:41:36 +00:00
|
|
|
if component is None:
|
|
|
|
extra = ""
|
|
|
|
else:
|
|
|
|
extra = " Search path was limited to path of component: {}".format(
|
|
|
|
base_paths[0])
|
|
|
|
_LOGGER.error("Unable to find platform %s.%s", platform_name, extra)
|
2019-02-07 21:33:12 +00:00
|
|
|
return None
|
|
|
|
|
2019-03-30 00:04:59 +00:00
|
|
|
if domain not in COMPONENTS_WITH_BAD_PLATFORMS:
|
|
|
|
_LOGGER.error(
|
|
|
|
"Integrations need to be in their own folder. Change %s/%s.py to "
|
|
|
|
"%s/%s.py. This will stop working soon.",
|
|
|
|
domain, platform_name, platform_name, domain)
|
2019-01-11 19:30:22 +00:00
|
|
|
|
|
|
|
return platform
|
2016-04-04 19:18:58 +00:00
|
|
|
|
2014-11-05 07:34:19 +00:00
|
|
|
|
2018-07-23 08:24:39 +00:00
|
|
|
def get_component(hass, # type: HomeAssistant
|
|
|
|
comp_or_platform: str) -> Optional[ModuleType]:
|
2018-05-07 09:25:48 +00:00
|
|
|
"""Try to load specified component.
|
|
|
|
|
2019-01-11 19:30:22 +00:00
|
|
|
Async friendly.
|
|
|
|
"""
|
2019-02-21 08:41:36 +00:00
|
|
|
comp = _load_file(hass, comp_or_platform, LOOKUP_PATHS)
|
2019-01-11 19:30:22 +00:00
|
|
|
|
|
|
|
if comp is None:
|
|
|
|
_LOGGER.error("Unable to find component %s", comp_or_platform)
|
|
|
|
|
|
|
|
return comp
|
|
|
|
|
|
|
|
|
|
|
|
def _load_file(hass, # type: HomeAssistant
|
2019-02-21 08:41:36 +00:00
|
|
|
comp_or_platform: str,
|
|
|
|
base_paths: List[str]) -> Optional[ModuleType]:
|
2019-01-11 19:30:22 +00:00
|
|
|
"""Try to load specified file.
|
|
|
|
|
2018-05-07 09:25:48 +00:00
|
|
|
Looks in config dir first, then built-in components.
|
|
|
|
Only returns it if also found to be valid.
|
|
|
|
Async friendly.
|
|
|
|
"""
|
2018-05-01 18:57:30 +00:00
|
|
|
try:
|
2018-07-13 17:14:45 +00:00
|
|
|
return hass.data[DATA_KEY][comp_or_platform] # type: ignore
|
2018-05-01 18:57:30 +00:00
|
|
|
except KeyError:
|
|
|
|
pass
|
2016-10-27 07:16:23 +00:00
|
|
|
|
2018-05-01 18:57:30 +00:00
|
|
|
cache = hass.data.get(DATA_KEY)
|
|
|
|
if cache is None:
|
2018-07-23 08:24:39 +00:00
|
|
|
if hass.config.config_dir is None:
|
|
|
|
_LOGGER.error("Can't load components - config dir is not set")
|
|
|
|
return None
|
2018-05-07 09:25:48 +00:00
|
|
|
# Only insert if it's not there (happens during tests)
|
|
|
|
if sys.path[0] != hass.config.config_dir:
|
|
|
|
sys.path.insert(0, hass.config.config_dir)
|
2018-05-01 18:57:30 +00:00
|
|
|
cache = hass.data[DATA_KEY] = {}
|
2014-11-12 05:39:17 +00:00
|
|
|
|
2019-02-22 13:11:07 +00:00
|
|
|
for path in ('{}.{}'.format(base, comp_or_platform)
|
|
|
|
for base in base_paths):
|
2018-05-01 18:57:30 +00:00
|
|
|
try:
|
2018-05-07 09:25:48 +00:00
|
|
|
module = importlib.import_module(path)
|
2014-11-15 07:17:18 +00:00
|
|
|
|
2018-05-07 09:25:48 +00:00
|
|
|
# In Python 3 you can import files from directories that do not
|
|
|
|
# contain the file __init__.py. A directory is a valid module if
|
|
|
|
# it contains a file with the .py extension. In this case Python
|
|
|
|
# will succeed in importing the directory as a module and call it
|
|
|
|
# a namespace. We do not care about namespaces.
|
|
|
|
# This prevents that when only
|
|
|
|
# custom_components/switch/some_platform.py exists,
|
|
|
|
# the import custom_components.switch would succeed.
|
2018-07-07 14:48:02 +00:00
|
|
|
# __file__ was unset for namespaces before Python 3.7
|
|
|
|
if getattr(module, '__file__', None) is None:
|
2018-05-07 09:25:48 +00:00
|
|
|
continue
|
2014-11-05 07:34:19 +00:00
|
|
|
|
2018-05-07 09:25:48 +00:00
|
|
|
_LOGGER.info("Loaded %s from %s", comp_or_platform, path)
|
2014-11-05 07:34:19 +00:00
|
|
|
|
2018-05-07 09:25:48 +00:00
|
|
|
cache[comp_or_platform] = module
|
2018-05-01 18:57:30 +00:00
|
|
|
|
2019-02-21 22:57:38 +00:00
|
|
|
if module.__name__.startswith(PACKAGE_CUSTOM_COMPONENTS):
|
2018-06-27 19:21:32 +00:00
|
|
|
_LOGGER.warning(
|
|
|
|
'You are using a custom component for %s which has not '
|
|
|
|
'been tested by Home Assistant. This component might '
|
|
|
|
'cause stability problems, be sure to disable it if you '
|
|
|
|
'do experience issues with Home Assistant.',
|
|
|
|
comp_or_platform)
|
|
|
|
|
2018-05-07 09:25:48 +00:00
|
|
|
return module
|
2018-05-01 18:57:30 +00:00
|
|
|
|
2018-05-07 09:25:48 +00:00
|
|
|
except ImportError as err:
|
|
|
|
# This error happens if for example custom_components/switch
|
|
|
|
# exists and we try to load switch.demo.
|
2018-05-07 17:12:12 +00:00
|
|
|
# Ignore errors for custom_components, custom_components.switch
|
|
|
|
# and custom_components.switch.demo.
|
|
|
|
white_listed_errors = []
|
|
|
|
parts = []
|
|
|
|
for part in path.split('.'):
|
|
|
|
parts.append(part)
|
|
|
|
white_listed_errors.append(
|
|
|
|
"No module named '{}'".format('.'.join(parts)))
|
|
|
|
|
|
|
|
if str(err) not in white_listed_errors:
|
2018-05-07 09:25:48 +00:00
|
|
|
_LOGGER.exception(
|
|
|
|
("Error loading %s. Make sure all "
|
|
|
|
"dependencies are installed"), path)
|
2018-05-01 18:57:30 +00:00
|
|
|
|
2018-05-07 09:25:48 +00:00
|
|
|
return None
|
2014-11-28 23:34:42 +00:00
|
|
|
|
|
|
|
|
2018-07-23 08:24:39 +00:00
|
|
|
class ModuleWrapper:
|
|
|
|
"""Class to wrap a Python module and auto fill in hass argument."""
|
|
|
|
|
|
|
|
def __init__(self,
|
|
|
|
hass, # type: HomeAssistant
|
|
|
|
module: ModuleType) -> None:
|
|
|
|
"""Initialize the module wrapper."""
|
|
|
|
self._hass = hass
|
|
|
|
self._module = module
|
|
|
|
|
|
|
|
def __getattr__(self, attr: str) -> Any:
|
|
|
|
"""Fetch an attribute."""
|
|
|
|
value = getattr(self._module, attr)
|
|
|
|
|
|
|
|
if hasattr(value, '__bind_hass'):
|
|
|
|
value = ft.partial(value, self._hass)
|
|
|
|
|
|
|
|
setattr(self, attr, value)
|
|
|
|
return value
|
|
|
|
|
|
|
|
|
2017-07-16 16:23:06 +00:00
|
|
|
class Components:
|
|
|
|
"""Helper to load components."""
|
|
|
|
|
2018-07-23 08:24:39 +00:00
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
hass # type: HomeAssistant
|
|
|
|
) -> None:
|
2017-07-16 16:23:06 +00:00
|
|
|
"""Initialize the Components class."""
|
|
|
|
self._hass = hass
|
|
|
|
|
2018-07-23 08:24:39 +00:00
|
|
|
def __getattr__(self, comp_name: str) -> ModuleWrapper:
|
2017-07-16 16:23:06 +00:00
|
|
|
"""Fetch a component."""
|
2018-05-01 18:57:30 +00:00
|
|
|
component = get_component(self._hass, comp_name)
|
2017-07-16 16:23:06 +00:00
|
|
|
if component is None:
|
|
|
|
raise ImportError('Unable to load {}'.format(comp_name))
|
2017-10-08 15:17:54 +00:00
|
|
|
wrapped = ModuleWrapper(self._hass, component)
|
2017-07-16 16:23:06 +00:00
|
|
|
setattr(self, comp_name, wrapped)
|
|
|
|
return wrapped
|
|
|
|
|
|
|
|
|
2017-10-08 15:17:54 +00:00
|
|
|
class Helpers:
|
|
|
|
"""Helper to load helpers."""
|
2017-07-16 16:23:06 +00:00
|
|
|
|
2018-07-23 08:24:39 +00:00
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
hass # type: HomeAssistant
|
|
|
|
) -> None:
|
2017-10-08 15:17:54 +00:00
|
|
|
"""Initialize the Helpers class."""
|
|
|
|
self._hass = hass
|
|
|
|
|
2018-07-23 08:24:39 +00:00
|
|
|
def __getattr__(self, helper_name: str) -> ModuleWrapper:
|
2017-10-08 15:17:54 +00:00
|
|
|
"""Fetch a helper."""
|
|
|
|
helper = importlib.import_module(
|
|
|
|
'homeassistant.helpers.{}'.format(helper_name))
|
|
|
|
wrapped = ModuleWrapper(self._hass, helper)
|
|
|
|
setattr(self, helper_name, wrapped)
|
|
|
|
return wrapped
|
|
|
|
|
|
|
|
|
2018-07-23 08:24:39 +00:00
|
|
|
def bind_hass(func: CALLABLE_T) -> CALLABLE_T:
|
2018-01-21 06:35:38 +00:00
|
|
|
"""Decorate function to indicate that first argument is hass."""
|
2018-07-23 08:24:39 +00:00
|
|
|
setattr(func, '__bind_hass', True)
|
2017-07-16 16:23:06 +00:00
|
|
|
return func
|
|
|
|
|
|
|
|
|
2019-02-07 21:56:40 +00:00
|
|
|
def component_dependencies(hass, # type: HomeAssistant
|
|
|
|
comp_name: str) -> Set[str]:
|
|
|
|
"""Return all dependencies and subdependencies of components.
|
2016-03-07 23:06:04 +00:00
|
|
|
|
2019-02-07 21:56:40 +00:00
|
|
|
Raises CircularDependency if a circular dependency is found.
|
2016-10-27 07:16:23 +00:00
|
|
|
|
|
|
|
Async friendly.
|
2014-11-28 23:34:42 +00:00
|
|
|
"""
|
2019-02-07 21:56:40 +00:00
|
|
|
return _component_dependencies(hass, comp_name, set(), set())
|
2014-11-28 23:34:42 +00:00
|
|
|
|
|
|
|
|
2019-02-07 21:56:40 +00:00
|
|
|
def _component_dependencies(hass, # type: HomeAssistant
|
|
|
|
comp_name: str, loaded: Set[str],
|
|
|
|
loading: Set) -> Set[str]:
|
|
|
|
"""Recursive function to get component dependencies.
|
2016-10-27 07:16:23 +00:00
|
|
|
|
|
|
|
Async friendly.
|
|
|
|
"""
|
2018-05-01 18:57:30 +00:00
|
|
|
component = get_component(hass, comp_name)
|
2014-11-28 23:34:42 +00:00
|
|
|
|
|
|
|
if component is None:
|
2019-02-07 21:56:40 +00:00
|
|
|
raise ComponentNotFound(comp_name)
|
2014-11-28 23:34:42 +00:00
|
|
|
|
|
|
|
loading.add(comp_name)
|
|
|
|
|
2015-11-26 21:11:59 +00:00
|
|
|
for dependency in getattr(component, 'DEPENDENCIES', []):
|
2014-11-28 23:34:42 +00:00
|
|
|
# Check not already loaded
|
2019-02-07 21:56:40 +00:00
|
|
|
if dependency in loaded:
|
2015-08-03 15:05:33 +00:00
|
|
|
continue
|
2014-11-28 23:34:42 +00:00
|
|
|
|
2016-03-07 23:06:04 +00:00
|
|
|
# If we are already loading it, we have a circular dependency.
|
2015-08-03 15:05:33 +00:00
|
|
|
if dependency in loading:
|
2019-02-07 21:56:40 +00:00
|
|
|
raise CircularDependency(comp_name, dependency)
|
2014-11-28 23:34:42 +00:00
|
|
|
|
2019-02-07 21:56:40 +00:00
|
|
|
dep_loaded = _component_dependencies(
|
|
|
|
hass, dependency, loaded, loading)
|
2014-11-28 23:34:42 +00:00
|
|
|
|
2019-02-07 21:56:40 +00:00
|
|
|
loaded.update(dep_loaded)
|
2014-11-28 23:34:42 +00:00
|
|
|
|
2019-02-07 21:56:40 +00:00
|
|
|
loaded.add(comp_name)
|
2014-11-28 23:34:42 +00:00
|
|
|
loading.remove(comp_name)
|
|
|
|
|
2019-02-07 21:56:40 +00:00
|
|
|
return loaded
|