core/homeassistant/requirements.py

146 lines
4.7 KiB
Python
Raw Normal View History

2018-01-30 11:30:47 +00:00
"""Module to handle installing requirements."""
import asyncio
import os
from typing import Any, Dict, Iterable, List, Optional, Set, Union, cast
2018-01-30 11:30:47 +00:00
from homeassistant.core import HomeAssistant
from homeassistant.exceptions import HomeAssistantError
from homeassistant.loader import Integration, IntegrationNotFound, async_get_integration
2018-01-30 11:30:47 +00:00
import homeassistant.util.package as pkg_util
2019-07-31 19:25:30 +00:00
DATA_PIP_LOCK = "pip_lock"
DATA_PKG_CACHE = "pkg_cache"
DATA_INTEGRATIONS_WITH_REQS = "integrations_with_reqs"
2019-07-31 19:25:30 +00:00
CONSTRAINT_FILE = "package_constraints.txt"
DISCOVERY_INTEGRATIONS: Dict[str, Iterable[str]] = {
"mqtt": ("mqtt",),
"ssdp": ("ssdp",),
"zeroconf": ("zeroconf", "homekit"),
}
_UNDEF = object()
2018-01-30 11:30:47 +00:00
class RequirementsNotFound(HomeAssistantError):
"""Raised when a component is not found."""
def __init__(self, domain: str, requirements: List) -> None:
"""Initialize a component not found error."""
super().__init__(f"Requirements for {domain} not found: {requirements}.")
self.domain = domain
self.requirements = requirements
async def async_get_integration_with_requirements(
hass: HomeAssistant, domain: str, done: Optional[Set[str]] = None
) -> Integration:
"""Get an integration with all requirements installed, including the dependencies.
This can raise IntegrationNotFound if manifest or integration
is invalid, RequirementNotFound if there was some type of
failure to install requirements.
"""
2019-12-05 18:40:05 +00:00
if done is None:
done = {domain}
else:
done.add(domain)
integration = await async_get_integration(hass, domain)
2019-10-31 18:39:26 +00:00
if hass.config.skip_pip:
return integration
cache = hass.data.get(DATA_INTEGRATIONS_WITH_REQS)
if cache is None:
cache = hass.data[DATA_INTEGRATIONS_WITH_REQS] = {}
int_or_evt: Union[Integration, asyncio.Event, None] = cache.get(domain, _UNDEF)
if isinstance(int_or_evt, asyncio.Event):
await int_or_evt.wait()
int_or_evt = cache.get(domain, _UNDEF)
# When we have waited and it's _UNDEF, it doesn't exist
# We don't cache that it doesn't exist, or else people can't fix it
# and then restart, because their config will never be valid.
if int_or_evt is _UNDEF:
raise IntegrationNotFound(domain)
if int_or_evt is not _UNDEF:
return cast(Integration, int_or_evt)
event = cache[domain] = asyncio.Event()
2019-10-31 18:39:26 +00:00
if integration.requirements:
await async_process_requirements(
hass, integration.domain, integration.requirements
)
2019-12-05 18:40:05 +00:00
deps_to_check = [
dep
for dep in integration.dependencies + integration.after_dependencies
2019-12-05 18:40:05 +00:00
if dep not in done
]
2019-11-02 00:21:50 +00:00
for check_domain, to_check in DISCOVERY_INTEGRATIONS.items():
if (
check_domain not in done
and check_domain not in deps_to_check
and any(check in integration.manifest for check in to_check)
):
deps_to_check.append(check_domain)
2019-12-05 18:40:05 +00:00
if deps_to_check:
2019-11-02 00:21:50 +00:00
await asyncio.gather(
2019-12-05 18:40:05 +00:00
*[
async_get_integration_with_requirements(hass, dep, done)
for dep in deps_to_check
]
2019-11-02 00:21:50 +00:00
)
cache[domain] = integration
event.set()
return integration
2019-07-31 19:25:30 +00:00
async def async_process_requirements(
hass: HomeAssistant, name: str, requirements: List[str]
) -> None:
2018-01-30 11:30:47 +00:00
"""Install the requirements for a component or platform.
This method is a coroutine. It will raise RequirementsNotFound
if an requirement can't be satisfied.
2018-01-30 11:30:47 +00:00
"""
pip_lock = hass.data.get(DATA_PIP_LOCK)
if pip_lock is None:
pip_lock = hass.data[DATA_PIP_LOCK] = asyncio.Lock()
2018-01-30 11:30:47 +00:00
kwargs = pip_kwargs(hass.config.config_dir)
2018-01-30 11:30:47 +00:00
async with pip_lock:
2018-01-30 11:30:47 +00:00
for req in requirements:
if pkg_util.is_installed(req):
continue
def _install(req: str, kwargs: Dict) -> bool:
"""Install requirement."""
return pkg_util.install_package(req, **kwargs)
ret = await hass.async_add_executor_job(_install, req, kwargs)
2018-01-30 11:30:47 +00:00
if not ret:
raise RequirementsNotFound(name, [req])
2018-01-30 11:30:47 +00:00
def pip_kwargs(config_dir: Optional[str]) -> Dict[str, Any]:
2018-01-30 11:30:47 +00:00
"""Return keyword arguments for PIP install."""
is_docker = pkg_util.is_docker_env()
2018-01-30 11:30:47 +00:00
kwargs = {
2019-07-31 19:25:30 +00:00
"constraints": os.path.join(os.path.dirname(__file__), CONSTRAINT_FILE),
"no_cache_dir": is_docker,
2018-01-30 11:30:47 +00:00
}
2019-07-31 19:25:30 +00:00
if "WHEELS_LINKS" in os.environ:
kwargs["find_links"] = os.environ["WHEELS_LINKS"]
if not (config_dir is None or pkg_util.is_virtual_env()) and not is_docker:
kwargs["target"] = os.path.join(config_dir, "deps")
2018-01-30 11:30:47 +00:00
return kwargs