2018-02-08 11:16:51 +00:00
|
|
|
"""Class to manage the entities for a single platform."""
|
|
|
|
import asyncio
|
2019-08-05 21:04:20 +00:00
|
|
|
from contextvars import ContextVar
|
2020-03-18 17:27:25 +00:00
|
|
|
from datetime import datetime, timedelta
|
|
|
|
from logging import Logger
|
|
|
|
from types import ModuleType
|
2020-08-12 21:01:10 +00:00
|
|
|
from typing import TYPE_CHECKING, Callable, Coroutine, Dict, Iterable, List, Optional
|
2018-02-08 11:16:51 +00:00
|
|
|
|
2020-08-19 12:57:38 +00:00
|
|
|
from homeassistant import config_entries
|
2018-02-08 11:16:51 +00:00
|
|
|
from homeassistant.const import DEVICE_DEFAULT_NAME
|
2020-08-12 21:01:10 +00:00
|
|
|
from homeassistant.core import (
|
|
|
|
CALLBACK_TYPE,
|
|
|
|
ServiceCall,
|
|
|
|
callback,
|
|
|
|
split_entity_id,
|
|
|
|
valid_entity_id,
|
|
|
|
)
|
2018-02-08 11:16:51 +00:00
|
|
|
from homeassistant.exceptions import HomeAssistantError, PlatformNotReady
|
2020-01-20 01:55:18 +00:00
|
|
|
from homeassistant.helpers import config_validation as cv, service
|
2020-03-18 17:27:25 +00:00
|
|
|
from homeassistant.helpers.typing import HomeAssistantType
|
2019-10-01 14:59:06 +00:00
|
|
|
from homeassistant.util.async_ import run_callback_threadsafe
|
2018-02-08 11:16:51 +00:00
|
|
|
|
2019-08-16 23:17:16 +00:00
|
|
|
from .entity_registry import DISABLED_INTEGRATION
|
2019-12-09 15:42:10 +00:00
|
|
|
from .event import async_call_later, async_track_time_interval
|
2019-07-21 16:59:02 +00:00
|
|
|
|
2020-03-18 17:27:25 +00:00
|
|
|
if TYPE_CHECKING:
|
|
|
|
from .entity import Entity
|
|
|
|
|
2020-08-12 21:01:10 +00:00
|
|
|
# mypy: allow-untyped-defs
|
2019-07-21 16:59:02 +00:00
|
|
|
|
2018-02-08 11:16:51 +00:00
|
|
|
SLOW_SETUP_WARNING = 10
|
|
|
|
SLOW_SETUP_MAX_WAIT = 60
|
2020-08-15 05:14:02 +00:00
|
|
|
SLOW_ADD_ENTITY_MAX_WAIT = 15 # Per Entity
|
|
|
|
SLOW_ADD_MIN_TIMEOUT = 500
|
2020-08-07 06:36:38 +00:00
|
|
|
|
2018-02-08 11:16:51 +00:00
|
|
|
PLATFORM_NOT_READY_RETRIES = 10
|
2020-03-23 19:59:36 +00:00
|
|
|
DATA_ENTITY_PLATFORM = "entity_platform"
|
2020-07-01 15:42:57 +00:00
|
|
|
PLATFORM_NOT_READY_BASE_WAIT_TIME = 30 # seconds
|
2018-02-08 11:16:51 +00:00
|
|
|
|
|
|
|
|
2018-07-20 08:45:20 +00:00
|
|
|
class EntityPlatform:
|
2018-02-08 11:16:51 +00:00
|
|
|
"""Manage the entities for a single platform."""
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
*,
|
2020-03-18 17:27:25 +00:00
|
|
|
hass: HomeAssistantType,
|
|
|
|
logger: Logger,
|
|
|
|
domain: str,
|
|
|
|
platform_name: str,
|
|
|
|
platform: Optional[ModuleType],
|
|
|
|
scan_interval: timedelta,
|
|
|
|
entity_namespace: Optional[str],
|
2019-07-31 19:25:30 +00:00
|
|
|
):
|
2020-03-18 17:27:25 +00:00
|
|
|
"""Initialize the entity platform."""
|
2018-02-08 11:16:51 +00:00
|
|
|
self.hass = hass
|
|
|
|
self.logger = logger
|
|
|
|
self.domain = domain
|
|
|
|
self.platform_name = platform_name
|
2018-04-08 03:04:50 +00:00
|
|
|
self.platform = platform
|
2018-02-08 11:16:51 +00:00
|
|
|
self.scan_interval = scan_interval
|
|
|
|
self.entity_namespace = entity_namespace
|
2020-08-19 12:57:38 +00:00
|
|
|
self.config_entry: Optional[config_entries.ConfigEntry] = None
|
2020-03-18 17:27:25 +00:00
|
|
|
self.entities: Dict[str, Entity] = {} # pylint: disable=used-before-assignment
|
|
|
|
self._tasks: List[asyncio.Future] = []
|
2018-04-12 12:28:54 +00:00
|
|
|
# Method to cancel the state change listener
|
2020-03-18 17:27:25 +00:00
|
|
|
self._async_unsub_polling: Optional[CALLBACK_TYPE] = None
|
2018-04-12 12:28:54 +00:00
|
|
|
# Method to cancel the retry of setup
|
2020-03-18 17:27:25 +00:00
|
|
|
self._async_cancel_retry_setup: Optional[CALLBACK_TYPE] = None
|
|
|
|
self._process_updates: Optional[asyncio.Lock] = None
|
2018-02-08 11:16:51 +00:00
|
|
|
|
2020-03-23 19:59:36 +00:00
|
|
|
self.parallel_updates: Optional[asyncio.Semaphore] = None
|
|
|
|
|
2018-04-08 03:04:50 +00:00
|
|
|
# Platform is None for the EntityComponent "catch-all" EntityPlatform
|
|
|
|
# which powers entity_component.add_entities
|
2020-03-23 19:59:36 +00:00
|
|
|
self.parallel_updates_created = platform is None
|
2018-04-08 03:04:50 +00:00
|
|
|
|
2020-03-23 19:59:36 +00:00
|
|
|
hass.data.setdefault(DATA_ENTITY_PLATFORM, {}).setdefault(
|
|
|
|
self.platform_name, []
|
|
|
|
).append(self)
|
2018-04-08 03:04:50 +00:00
|
|
|
|
2020-02-04 23:30:15 +00:00
|
|
|
@callback
|
|
|
|
def _get_parallel_updates_semaphore(
|
|
|
|
self, entity_has_async_update: bool
|
|
|
|
) -> Optional[asyncio.Semaphore]:
|
|
|
|
"""Get or create a semaphore for parallel updates.
|
|
|
|
|
|
|
|
Semaphore will be created on demand because we base it off if update method is async or not.
|
|
|
|
|
|
|
|
If parallel updates is set to 0, we skip the semaphore.
|
|
|
|
If parallel updates is set to a number, we initialize the semaphore to that number.
|
2020-07-20 00:40:08 +00:00
|
|
|
The default value for parallel requests is decided based on the first entity that is added to Home Assistant.
|
|
|
|
It's 0 if the entity defines the async_update method, else it's 1.
|
2020-02-04 23:30:15 +00:00
|
|
|
"""
|
|
|
|
if self.parallel_updates_created:
|
|
|
|
return self.parallel_updates
|
|
|
|
|
|
|
|
self.parallel_updates_created = True
|
|
|
|
|
|
|
|
parallel_updates = getattr(self.platform, "PARALLEL_UPDATES", None)
|
|
|
|
|
|
|
|
if parallel_updates is None and not entity_has_async_update:
|
|
|
|
parallel_updates = 1
|
|
|
|
|
|
|
|
if parallel_updates == 0:
|
|
|
|
parallel_updates = None
|
|
|
|
|
|
|
|
if parallel_updates is not None:
|
|
|
|
self.parallel_updates = asyncio.Semaphore(parallel_updates)
|
|
|
|
|
|
|
|
return self.parallel_updates
|
2018-02-08 11:16:51 +00:00
|
|
|
|
2018-04-09 14:09:08 +00:00
|
|
|
async def async_setup(self, platform_config, discovery_info=None):
|
2018-08-19 20:29:08 +00:00
|
|
|
"""Set up the platform from a config file."""
|
2018-04-08 03:04:50 +00:00
|
|
|
platform = self.platform
|
2018-04-09 14:09:08 +00:00
|
|
|
hass = self.hass
|
|
|
|
|
2020-01-12 02:21:57 +00:00
|
|
|
if not hasattr(platform, "async_setup_platform") and not hasattr(
|
|
|
|
platform, "setup_platform"
|
|
|
|
):
|
|
|
|
self.logger.error(
|
|
|
|
"The %s platform for the %s integration does not support platform setup. Please remove it from your config.",
|
|
|
|
self.platform_name,
|
|
|
|
self.domain,
|
|
|
|
)
|
|
|
|
return
|
|
|
|
|
2018-04-09 14:09:08 +00:00
|
|
|
@callback
|
2020-08-12 21:01:10 +00:00
|
|
|
def async_create_setup_task() -> Coroutine:
|
2018-08-19 20:29:08 +00:00
|
|
|
"""Get task to set up platform."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if getattr(platform, "async_setup_platform", None):
|
2020-08-12 21:01:10 +00:00
|
|
|
return platform.async_setup_platform( # type: ignore
|
2019-07-31 19:25:30 +00:00
|
|
|
hass,
|
|
|
|
platform_config,
|
|
|
|
self._async_schedule_add_entities,
|
|
|
|
discovery_info,
|
2018-04-09 14:09:08 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
# This should not be replaced with hass.async_add_job because
|
|
|
|
# we don't want to track this task in case it blocks startup.
|
|
|
|
return hass.loop.run_in_executor(
|
2019-07-31 19:25:30 +00:00
|
|
|
None,
|
2020-08-12 21:01:10 +00:00
|
|
|
platform.setup_platform, # type: ignore
|
2019-07-31 19:25:30 +00:00
|
|
|
hass,
|
|
|
|
platform_config,
|
|
|
|
self._schedule_add_entities,
|
|
|
|
discovery_info,
|
2018-04-09 14:09:08 +00:00
|
|
|
)
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2018-04-09 14:09:08 +00:00
|
|
|
await self._async_setup_platform(async_create_setup_task)
|
|
|
|
|
2020-08-19 12:57:38 +00:00
|
|
|
async def async_setup_entry(self, config_entry: config_entries.ConfigEntry) -> bool:
|
2018-08-19 20:29:08 +00:00
|
|
|
"""Set up the platform from a config entry."""
|
2018-04-09 14:09:08 +00:00
|
|
|
# Store it so that we can save config entry ID in entity registry
|
|
|
|
self.config_entry = config_entry
|
|
|
|
platform = self.platform
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_create_setup_task():
|
2018-08-19 20:29:08 +00:00
|
|
|
"""Get task to set up platform."""
|
2020-08-12 21:01:10 +00:00
|
|
|
return platform.async_setup_entry( # type: ignore
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass, config_entry, self._async_schedule_add_entities
|
|
|
|
)
|
2018-04-09 14:09:08 +00:00
|
|
|
|
|
|
|
return await self._async_setup_platform(async_create_setup_task)
|
|
|
|
|
2020-08-12 21:01:10 +00:00
|
|
|
async def _async_setup_platform(
|
|
|
|
self, async_create_setup_task: Callable[[], Coroutine], tries: int = 0
|
|
|
|
) -> bool:
|
2018-08-24 08:28:43 +00:00
|
|
|
"""Set up a platform via config file or config entry.
|
2018-04-09 14:09:08 +00:00
|
|
|
|
|
|
|
async_create_setup_task creates a coroutine that sets up platform.
|
|
|
|
"""
|
2019-08-05 21:04:20 +00:00
|
|
|
current_platform.set(self)
|
2018-02-08 11:16:51 +00:00
|
|
|
logger = self.logger
|
|
|
|
hass = self.hass
|
2019-08-23 16:53:33 +00:00
|
|
|
full_name = f"{self.domain}.{self.platform_name}"
|
2018-02-08 11:16:51 +00:00
|
|
|
|
|
|
|
logger.info("Setting up %s", full_name)
|
|
|
|
warn_task = hass.loop.call_later(
|
2019-07-31 19:25:30 +00:00
|
|
|
SLOW_SETUP_WARNING,
|
|
|
|
logger.warning,
|
2020-01-14 20:54:45 +00:00
|
|
|
"Setup of %s platform %s is taking over %s seconds.",
|
|
|
|
self.domain,
|
2019-07-31 19:25:30 +00:00
|
|
|
self.platform_name,
|
|
|
|
SLOW_SETUP_WARNING,
|
|
|
|
)
|
2018-02-08 11:16:51 +00:00
|
|
|
|
|
|
|
try:
|
2018-04-09 14:09:08 +00:00
|
|
|
task = async_create_setup_task()
|
|
|
|
|
2020-08-05 12:58:19 +00:00
|
|
|
async with hass.timeout.async_timeout(SLOW_SETUP_MAX_WAIT, self.domain):
|
|
|
|
await asyncio.shield(task)
|
2018-02-08 11:16:51 +00:00
|
|
|
|
|
|
|
# Block till all entities are done
|
|
|
|
if self._tasks:
|
|
|
|
pending = [task for task in self._tasks if not task.done()]
|
|
|
|
self._tasks.clear()
|
|
|
|
|
|
|
|
if pending:
|
2020-04-01 14:09:13 +00:00
|
|
|
await asyncio.gather(*pending)
|
2018-02-08 11:16:51 +00:00
|
|
|
|
|
|
|
hass.config.components.add(full_name)
|
2018-04-09 14:09:08 +00:00
|
|
|
return True
|
2018-02-08 11:16:51 +00:00
|
|
|
except PlatformNotReady:
|
|
|
|
tries += 1
|
2020-07-01 15:42:57 +00:00
|
|
|
wait_time = min(tries, 6) * PLATFORM_NOT_READY_BASE_WAIT_TIME
|
2018-02-08 11:16:51 +00:00
|
|
|
logger.warning(
|
2019-07-31 19:25:30 +00:00
|
|
|
"Platform %s not ready yet. Retrying in %d seconds.",
|
|
|
|
self.platform_name,
|
|
|
|
wait_time,
|
|
|
|
)
|
2018-04-09 14:09:08 +00:00
|
|
|
|
|
|
|
async def setup_again(now):
|
|
|
|
"""Run setup again."""
|
2018-04-12 12:28:54 +00:00
|
|
|
self._async_cancel_retry_setup = None
|
2019-07-31 19:25:30 +00:00
|
|
|
await self._async_setup_platform(async_create_setup_task, tries)
|
2018-04-09 14:09:08 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
self._async_cancel_retry_setup = async_call_later(
|
|
|
|
hass, wait_time, setup_again
|
|
|
|
)
|
2018-04-09 14:09:08 +00:00
|
|
|
return False
|
2018-02-08 11:16:51 +00:00
|
|
|
except asyncio.TimeoutError:
|
|
|
|
logger.error(
|
|
|
|
"Setup of platform %s is taking longer than %s seconds."
|
|
|
|
" Startup will proceed without waiting any longer.",
|
2019-07-31 19:25:30 +00:00
|
|
|
self.platform_name,
|
|
|
|
SLOW_SETUP_MAX_WAIT,
|
|
|
|
)
|
2018-04-09 14:09:08 +00:00
|
|
|
return False
|
2018-02-08 11:16:51 +00:00
|
|
|
except Exception: # pylint: disable=broad-except
|
2020-01-20 01:55:18 +00:00
|
|
|
logger.exception(
|
|
|
|
"Error while setting up %s platform for %s",
|
|
|
|
self.platform_name,
|
|
|
|
self.domain,
|
|
|
|
)
|
2018-04-09 14:09:08 +00:00
|
|
|
return False
|
2018-02-08 11:16:51 +00:00
|
|
|
finally:
|
|
|
|
warn_task.cancel()
|
|
|
|
|
2020-03-18 17:27:25 +00:00
|
|
|
def _schedule_add_entities(
|
|
|
|
self, new_entities: Iterable["Entity"], update_before_add: bool = False
|
|
|
|
) -> None:
|
2018-08-24 08:28:43 +00:00
|
|
|
"""Schedule adding entities for a single platform, synchronously."""
|
2018-02-08 11:16:51 +00:00
|
|
|
run_callback_threadsafe(
|
|
|
|
self.hass.loop,
|
2019-07-31 19:25:30 +00:00
|
|
|
self._async_schedule_add_entities,
|
|
|
|
list(new_entities),
|
|
|
|
update_before_add,
|
2018-02-08 11:16:51 +00:00
|
|
|
).result()
|
|
|
|
|
|
|
|
@callback
|
2020-03-18 17:27:25 +00:00
|
|
|
def _async_schedule_add_entities(
|
|
|
|
self, new_entities: Iterable["Entity"], update_before_add: bool = False
|
|
|
|
) -> None:
|
2018-02-08 11:16:51 +00:00
|
|
|
"""Schedule adding entities for a single platform async."""
|
2019-07-31 19:25:30 +00:00
|
|
|
self._tasks.append(
|
2020-06-01 05:18:30 +00:00
|
|
|
self.hass.async_create_task(
|
|
|
|
self.async_add_entities(
|
|
|
|
new_entities, update_before_add=update_before_add
|
2020-03-18 17:27:25 +00:00
|
|
|
),
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
)
|
2018-02-08 11:16:51 +00:00
|
|
|
|
2020-03-18 17:27:25 +00:00
|
|
|
def add_entities(
|
|
|
|
self, new_entities: Iterable["Entity"], update_before_add: bool = False
|
|
|
|
) -> None:
|
2018-02-08 11:16:51 +00:00
|
|
|
"""Add entities for a single platform."""
|
|
|
|
# That avoid deadlocks
|
|
|
|
if update_before_add:
|
|
|
|
self.logger.warning(
|
|
|
|
"Call 'add_entities' with update_before_add=True "
|
2019-07-31 19:25:30 +00:00
|
|
|
"only inside tests or you can run into a deadlock!"
|
|
|
|
)
|
2018-02-08 11:16:51 +00:00
|
|
|
|
2019-10-01 14:59:06 +00:00
|
|
|
asyncio.run_coroutine_threadsafe(
|
2018-02-08 11:16:51 +00:00
|
|
|
self.async_add_entities(list(new_entities), update_before_add),
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass.loop,
|
|
|
|
).result()
|
2018-02-08 11:16:51 +00:00
|
|
|
|
2020-03-18 17:27:25 +00:00
|
|
|
async def async_add_entities(
|
|
|
|
self, new_entities: Iterable["Entity"], update_before_add: bool = False
|
|
|
|
) -> None:
|
2018-02-08 11:16:51 +00:00
|
|
|
"""Add entities for a single platform async.
|
|
|
|
|
|
|
|
This method must be run in the event loop.
|
|
|
|
"""
|
|
|
|
# handle empty list from component/platform
|
|
|
|
if not new_entities:
|
|
|
|
return
|
|
|
|
|
|
|
|
hass = self.hass
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
device_registry = await hass.helpers.device_registry.async_get_registry()
|
|
|
|
entity_registry = await hass.helpers.entity_registry.async_get_registry()
|
2018-02-08 11:16:51 +00:00
|
|
|
tasks = [
|
2020-08-06 07:32:42 +00:00
|
|
|
self._async_add_entity( # type: ignore
|
|
|
|
entity, update_before_add, entity_registry, device_registry
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
|
|
|
for entity in new_entities
|
|
|
|
]
|
2018-02-08 11:16:51 +00:00
|
|
|
|
2018-06-18 13:22:52 +00:00
|
|
|
# No entities for processing
|
|
|
|
if not tasks:
|
|
|
|
return
|
|
|
|
|
2020-08-07 06:36:38 +00:00
|
|
|
timeout = max(SLOW_ADD_ENTITY_MAX_WAIT * len(tasks), SLOW_ADD_MIN_TIMEOUT)
|
|
|
|
try:
|
|
|
|
async with self.hass.timeout.async_timeout(timeout, self.domain):
|
|
|
|
await asyncio.gather(*tasks)
|
|
|
|
except asyncio.TimeoutError:
|
|
|
|
self.logger.warning(
|
|
|
|
"Timed out adding entities for domain %s with platform %s after %ds",
|
|
|
|
self.domain,
|
|
|
|
self.platform_name,
|
|
|
|
timeout,
|
|
|
|
)
|
2018-02-08 11:16:51 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
if self._async_unsub_polling is not None or not any(
|
|
|
|
entity.should_poll for entity in self.entities.values()
|
|
|
|
):
|
2018-02-08 11:16:51 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
self._async_unsub_polling = async_track_time_interval(
|
2020-04-14 00:41:01 +00:00
|
|
|
self.hass, self._update_entity_states, self.scan_interval,
|
2018-02-08 11:16:51 +00:00
|
|
|
)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
async def _async_add_entity(
|
|
|
|
self, entity, update_before_add, entity_registry, device_registry
|
|
|
|
):
|
2018-08-24 08:28:43 +00:00
|
|
|
"""Add an entity to the platform."""
|
2018-02-08 11:16:51 +00:00
|
|
|
if entity is None:
|
2019-07-31 19:25:30 +00:00
|
|
|
raise ValueError("Entity cannot be None")
|
2018-02-08 11:16:51 +00:00
|
|
|
|
2020-08-19 12:57:38 +00:00
|
|
|
entity.add_to_platform_start(
|
|
|
|
self.hass,
|
|
|
|
self,
|
|
|
|
self._get_parallel_updates_semaphore(hasattr(entity, "async_update")),
|
2020-02-04 23:30:15 +00:00
|
|
|
)
|
2018-02-08 11:16:51 +00:00
|
|
|
|
|
|
|
# Update properties before we generate the entity_id
|
|
|
|
if update_before_add:
|
|
|
|
try:
|
2018-02-25 11:38:46 +00:00
|
|
|
await entity.async_device_update(warning=False)
|
2018-02-08 11:16:51 +00:00
|
|
|
except Exception: # pylint: disable=broad-except
|
2019-07-31 19:25:30 +00:00
|
|
|
self.logger.exception("%s: Error on device update!", self.platform_name)
|
2020-08-19 12:57:38 +00:00
|
|
|
entity.add_to_platform_abort()
|
2018-02-08 11:16:51 +00:00
|
|
|
return
|
|
|
|
|
2020-08-01 09:20:37 +00:00
|
|
|
requested_entity_id = None
|
2020-08-12 21:01:10 +00:00
|
|
|
suggested_object_id: Optional[str] = None
|
2018-02-08 11:16:51 +00:00
|
|
|
|
|
|
|
# Get entity_id from unique ID registration
|
|
|
|
if entity.unique_id is not None:
|
|
|
|
if entity.entity_id is not None:
|
2020-08-01 09:20:37 +00:00
|
|
|
requested_entity_id = entity.entity_id
|
2018-02-08 11:16:51 +00:00
|
|
|
suggested_object_id = split_entity_id(entity.entity_id)[1]
|
|
|
|
else:
|
|
|
|
suggested_object_id = entity.name
|
|
|
|
|
2018-02-12 04:55:38 +00:00
|
|
|
if self.entity_namespace is not None:
|
2020-01-03 13:47:06 +00:00
|
|
|
suggested_object_id = f"{self.entity_namespace} {suggested_object_id}"
|
2018-02-12 04:55:38 +00:00
|
|
|
|
2018-06-07 18:23:09 +00:00
|
|
|
if self.config_entry is not None:
|
2020-08-12 21:01:10 +00:00
|
|
|
config_entry_id: Optional[str] = self.config_entry.entry_id
|
2018-06-07 18:23:09 +00:00
|
|
|
else:
|
|
|
|
config_entry_id = None
|
|
|
|
|
2018-08-25 08:59:28 +00:00
|
|
|
device_info = entity.device_info
|
2018-10-08 10:53:51 +00:00
|
|
|
device_id = None
|
2018-09-17 11:39:30 +00:00
|
|
|
|
2018-08-25 08:59:28 +00:00
|
|
|
if config_entry_id is not None and device_info is not None:
|
2019-07-31 19:25:30 +00:00
|
|
|
processed_dev_info = {"config_entry_id": config_entry_id}
|
2018-10-08 07:30:40 +00:00
|
|
|
for key in (
|
2019-07-31 19:25:30 +00:00
|
|
|
"connections",
|
|
|
|
"identifiers",
|
|
|
|
"manufacturer",
|
|
|
|
"model",
|
|
|
|
"name",
|
2020-08-13 08:38:56 +00:00
|
|
|
"default_manufacturer",
|
|
|
|
"default_model",
|
|
|
|
"default_name",
|
2019-07-31 19:25:30 +00:00
|
|
|
"sw_version",
|
2020-05-03 20:56:58 +00:00
|
|
|
"entry_type",
|
2019-07-31 19:25:30 +00:00
|
|
|
"via_device",
|
2018-10-08 07:30:40 +00:00
|
|
|
):
|
|
|
|
if key in device_info:
|
|
|
|
processed_dev_info[key] = device_info[key]
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
device = device_registry.async_get_or_create(**processed_dev_info)
|
2018-10-08 10:53:51 +00:00
|
|
|
if device:
|
|
|
|
device_id = device.id
|
2018-08-22 08:46:37 +00:00
|
|
|
|
2019-08-16 23:17:16 +00:00
|
|
|
disabled_by: Optional[str] = None
|
|
|
|
if not entity.entity_registry_enabled_default:
|
|
|
|
disabled_by = DISABLED_INTEGRATION
|
|
|
|
|
2018-08-22 08:46:37 +00:00
|
|
|
entry = entity_registry.async_get_or_create(
|
2019-07-31 19:25:30 +00:00
|
|
|
self.domain,
|
|
|
|
self.platform_name,
|
|
|
|
entity.unique_id,
|
2018-06-07 18:23:09 +00:00
|
|
|
suggested_object_id=suggested_object_id,
|
2019-08-18 04:34:11 +00:00
|
|
|
config_entry=self.config_entry,
|
2018-12-14 09:33:37 +00:00
|
|
|
device_id=device_id,
|
2019-07-31 19:25:30 +00:00
|
|
|
known_object_ids=self.entities.keys(),
|
2019-08-16 23:17:16 +00:00
|
|
|
disabled_by=disabled_by,
|
2019-12-31 13:29:43 +00:00
|
|
|
capabilities=entity.capability_attributes,
|
|
|
|
supported_features=entity.supported_features,
|
|
|
|
device_class=entity.device_class,
|
2020-01-15 16:09:05 +00:00
|
|
|
unit_of_measurement=entity.unit_of_measurement,
|
2020-02-11 17:40:50 +00:00
|
|
|
original_name=entity.name,
|
|
|
|
original_icon=entity.icon,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-02-13 12:33:15 +00:00
|
|
|
|
2019-08-22 21:12:24 +00:00
|
|
|
entity.registry_entry = entry
|
|
|
|
entity.entity_id = entry.entity_id
|
|
|
|
|
2018-02-13 12:33:15 +00:00
|
|
|
if entry.disabled:
|
|
|
|
self.logger.info(
|
|
|
|
"Not adding entity %s because it's disabled",
|
2019-07-31 19:25:30 +00:00
|
|
|
entry.name
|
|
|
|
or entity.name
|
2019-08-23 16:53:33 +00:00
|
|
|
or f'"{self.platform_name} {entity.unique_id}"',
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2020-08-19 12:57:38 +00:00
|
|
|
entity.add_to_platform_abort()
|
2018-02-13 12:33:15 +00:00
|
|
|
return
|
|
|
|
|
2018-02-08 11:16:51 +00:00
|
|
|
# We won't generate an entity ID if the platform has already set one
|
|
|
|
# We will however make sure that platform cannot pick a registered ID
|
2019-07-31 19:25:30 +00:00
|
|
|
elif entity.entity_id is not None and entity_registry.async_is_registered(
|
|
|
|
entity.entity_id
|
|
|
|
):
|
2018-02-08 11:16:51 +00:00
|
|
|
# If entity already registered, convert entity id to suggestion
|
|
|
|
suggested_object_id = split_entity_id(entity.entity_id)[1]
|
|
|
|
entity.entity_id = None
|
|
|
|
|
|
|
|
# Generate entity ID
|
|
|
|
if entity.entity_id is None:
|
2019-07-31 19:25:30 +00:00
|
|
|
suggested_object_id = (
|
2018-02-08 11:16:51 +00:00
|
|
|
suggested_object_id or entity.name or DEVICE_DEFAULT_NAME
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2018-02-08 11:16:51 +00:00
|
|
|
|
|
|
|
if self.entity_namespace is not None:
|
2020-01-03 13:47:06 +00:00
|
|
|
suggested_object_id = f"{self.entity_namespace} {suggested_object_id}"
|
2018-08-22 08:46:37 +00:00
|
|
|
entity.entity_id = entity_registry.async_generate_entity_id(
|
2019-07-31 19:25:30 +00:00
|
|
|
self.domain, suggested_object_id, self.entities.keys()
|
|
|
|
)
|
2018-02-08 11:16:51 +00:00
|
|
|
|
|
|
|
# Make sure it is valid in case an entity set the value themselves
|
|
|
|
if not valid_entity_id(entity.entity_id):
|
2020-08-19 12:57:38 +00:00
|
|
|
entity.add_to_platform_abort()
|
2019-08-23 16:53:33 +00:00
|
|
|
raise HomeAssistantError(f"Invalid entity id: {entity.entity_id}")
|
2019-12-31 13:29:43 +00:00
|
|
|
|
|
|
|
already_exists = entity.entity_id in self.entities
|
|
|
|
|
|
|
|
if not already_exists:
|
|
|
|
existing = self.hass.states.get(entity.entity_id)
|
|
|
|
|
|
|
|
if existing and not existing.attributes.get("restored"):
|
|
|
|
already_exists = True
|
|
|
|
|
|
|
|
if already_exists:
|
2018-02-12 03:33:37 +00:00
|
|
|
if entity.unique_id is not None:
|
2020-08-01 09:20:37 +00:00
|
|
|
msg = f"Platform {self.platform_name} does not generate unique IDs. "
|
|
|
|
if requested_entity_id:
|
|
|
|
msg += f"ID {entity.unique_id} is already used by {entity.entity_id} - ignoring {requested_entity_id}"
|
|
|
|
else:
|
|
|
|
msg += f"ID {entity.unique_id} already exists - ignoring {entity.entity_id}"
|
|
|
|
else:
|
|
|
|
msg = f"Entity id already exists - ignoring: {entity.entity_id}"
|
2020-04-01 14:09:13 +00:00
|
|
|
self.logger.error(msg)
|
2020-08-19 12:57:38 +00:00
|
|
|
entity.add_to_platform_abort()
|
2020-04-01 14:09:13 +00:00
|
|
|
return
|
2018-02-08 11:16:51 +00:00
|
|
|
|
2018-10-25 17:57:36 +00:00
|
|
|
entity_id = entity.entity_id
|
|
|
|
self.entities[entity_id] = entity
|
|
|
|
entity.async_on_remove(lambda: self.entities.pop(entity_id))
|
2018-02-08 11:16:51 +00:00
|
|
|
|
2020-08-19 12:57:38 +00:00
|
|
|
await entity.add_to_platform_finish()
|
2018-02-08 11:16:51 +00:00
|
|
|
|
2019-12-21 07:23:48 +00:00
|
|
|
async def async_reset(self) -> None:
|
2018-02-08 11:16:51 +00:00
|
|
|
"""Remove all entities and reset data.
|
|
|
|
|
|
|
|
This method must be run in the event loop.
|
|
|
|
"""
|
2018-04-12 12:28:54 +00:00
|
|
|
if self._async_cancel_retry_setup is not None:
|
|
|
|
self._async_cancel_retry_setup()
|
|
|
|
self._async_cancel_retry_setup = None
|
|
|
|
|
2018-02-08 11:16:51 +00:00
|
|
|
if not self.entities:
|
|
|
|
return
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
tasks = [self.async_remove_entity(entity_id) for entity_id in self.entities]
|
2018-02-08 11:16:51 +00:00
|
|
|
|
2020-04-01 14:09:13 +00:00
|
|
|
await asyncio.gather(*tasks)
|
2018-02-08 11:16:51 +00:00
|
|
|
|
|
|
|
if self._async_unsub_polling is not None:
|
|
|
|
self._async_unsub_polling()
|
|
|
|
self._async_unsub_polling = None
|
|
|
|
|
2020-03-23 19:59:36 +00:00
|
|
|
async def async_destroy(self) -> None:
|
|
|
|
"""Destroy an entity platform.
|
|
|
|
|
|
|
|
Call before discarding the object.
|
|
|
|
"""
|
|
|
|
await self.async_reset()
|
|
|
|
self.hass.data[DATA_ENTITY_PLATFORM][self.platform_name].remove(self)
|
|
|
|
|
2019-12-21 07:23:48 +00:00
|
|
|
async def async_remove_entity(self, entity_id: str) -> None:
|
2018-02-08 11:16:51 +00:00
|
|
|
"""Remove entity id from platform."""
|
2018-10-25 17:57:36 +00:00
|
|
|
await self.entities[entity_id].async_remove()
|
2018-02-08 11:16:51 +00:00
|
|
|
|
|
|
|
# Clean up polling job if no longer needed
|
2019-07-31 19:25:30 +00:00
|
|
|
if self._async_unsub_polling is not None and not any(
|
|
|
|
entity.should_poll for entity in self.entities.values()
|
|
|
|
):
|
2018-02-08 11:16:51 +00:00
|
|
|
self._async_unsub_polling()
|
|
|
|
self._async_unsub_polling = None
|
|
|
|
|
2020-08-12 21:01:10 +00:00
|
|
|
async def async_extract_from_service(
|
|
|
|
self, service_call: ServiceCall, expand_group: bool = True
|
|
|
|
) -> List["Entity"]:
|
2020-01-20 01:55:18 +00:00
|
|
|
"""Extract all known and available entities from a service call.
|
|
|
|
|
|
|
|
Will return an empty list if entities specified but unknown.
|
|
|
|
|
|
|
|
This method must be run in the event loop.
|
|
|
|
"""
|
|
|
|
return await service.async_extract_entities(
|
|
|
|
self.hass, self.entities.values(), service_call, expand_group
|
|
|
|
)
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_register_entity_service(self, name, schema, func, required_features=None):
|
2020-03-23 19:59:36 +00:00
|
|
|
"""Register an entity service.
|
|
|
|
|
|
|
|
Services will automatically be shared by all platforms of the same domain.
|
|
|
|
"""
|
|
|
|
if self.hass.services.has_service(self.platform_name, name):
|
|
|
|
return
|
|
|
|
|
2020-01-20 01:55:18 +00:00
|
|
|
if isinstance(schema, dict):
|
|
|
|
schema = cv.make_entity_service_schema(schema)
|
|
|
|
|
2020-08-12 21:01:10 +00:00
|
|
|
async def handle_service(call: ServiceCall) -> None:
|
2020-01-20 01:55:18 +00:00
|
|
|
"""Handle the service."""
|
2020-08-12 21:01:10 +00:00
|
|
|
await service.entity_service_call( # type: ignore
|
2020-03-23 19:59:36 +00:00
|
|
|
self.hass,
|
2020-07-02 09:39:53 +00:00
|
|
|
[
|
|
|
|
plf
|
|
|
|
for plf in self.hass.data[DATA_ENTITY_PLATFORM][self.platform_name]
|
|
|
|
if plf.domain == self.domain
|
|
|
|
],
|
2020-03-23 19:59:36 +00:00
|
|
|
func,
|
|
|
|
call,
|
|
|
|
required_features,
|
2020-01-20 01:55:18 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
self.hass.services.async_register(
|
|
|
|
self.platform_name, name, handle_service, schema
|
|
|
|
)
|
|
|
|
|
2019-12-21 07:23:48 +00:00
|
|
|
async def _update_entity_states(self, now: datetime) -> None:
|
2018-02-08 11:16:51 +00:00
|
|
|
"""Update the states of all the polling entities.
|
|
|
|
|
|
|
|
To protect from flooding the executor, we will update async entities
|
|
|
|
in parallel and other entities sequential.
|
|
|
|
|
|
|
|
This method must be run in the event loop.
|
|
|
|
"""
|
2019-05-31 18:26:05 +00:00
|
|
|
if self._process_updates is None:
|
|
|
|
self._process_updates = asyncio.Lock()
|
2018-02-08 11:16:51 +00:00
|
|
|
if self._process_updates.locked():
|
|
|
|
self.logger.warning(
|
2020-01-02 19:17:10 +00:00
|
|
|
"Updating %s %s took longer than the scheduled update interval %s",
|
2019-07-31 19:25:30 +00:00
|
|
|
self.platform_name,
|
|
|
|
self.domain,
|
|
|
|
self.scan_interval,
|
|
|
|
)
|
2018-02-08 11:16:51 +00:00
|
|
|
return
|
|
|
|
|
2018-03-17 11:27:21 +00:00
|
|
|
async with self._process_updates:
|
2018-02-08 11:16:51 +00:00
|
|
|
tasks = []
|
|
|
|
for entity in self.entities.values():
|
|
|
|
if not entity.should_poll:
|
|
|
|
continue
|
2020-06-06 18:34:56 +00:00
|
|
|
tasks.append(entity.async_update_ha_state(True))
|
2018-02-08 11:16:51 +00:00
|
|
|
|
|
|
|
if tasks:
|
2020-04-01 14:09:13 +00:00
|
|
|
await asyncio.gather(*tasks)
|
2019-08-05 21:04:20 +00:00
|
|
|
|
|
|
|
|
|
|
|
current_platform: ContextVar[Optional[EntityPlatform]] = ContextVar(
|
|
|
|
"current_platform", default=None
|
|
|
|
)
|