2021-05-31 23:35:31 +00:00
|
|
|
"""Helpers to help during startup."""
|
2021-09-29 14:32:11 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2022-08-09 20:12:33 +00:00
|
|
|
from collections.abc import Callable, Coroutine
|
|
|
|
from typing import Any
|
2021-05-31 23:35:31 +00:00
|
|
|
|
2022-10-05 10:24:51 +00:00
|
|
|
from homeassistant.const import EVENT_HOMEASSISTANT_START, EVENT_HOMEASSISTANT_STARTED
|
|
|
|
from homeassistant.core import (
|
|
|
|
CALLBACK_TYPE,
|
|
|
|
CoreState,
|
|
|
|
Event,
|
|
|
|
HassJob,
|
|
|
|
HomeAssistant,
|
|
|
|
callback,
|
|
|
|
)
|
2021-05-31 23:35:31 +00:00
|
|
|
|
|
|
|
|
|
|
|
@callback
|
2022-10-05 10:24:51 +00:00
|
|
|
def _async_at_core_state(
|
2022-08-09 20:12:33 +00:00
|
|
|
hass: HomeAssistant,
|
|
|
|
at_start_cb: Callable[[HomeAssistant], Coroutine[Any, Any, None] | None],
|
2022-10-05 10:24:51 +00:00
|
|
|
event_type: str,
|
|
|
|
check_state: Callable[[HomeAssistant], bool],
|
2022-04-01 21:34:36 +00:00
|
|
|
) -> CALLBACK_TYPE:
|
2022-10-05 10:24:51 +00:00
|
|
|
"""Execute a job at_start_cb when Home Assistant has the wanted state.
|
2021-05-31 23:35:31 +00:00
|
|
|
|
2022-10-05 10:24:51 +00:00
|
|
|
The job is executed immediately if Home Assistant is in the wanted state.
|
|
|
|
Will wait for event specified by event_type if it isn't.
|
2021-05-31 23:35:31 +00:00
|
|
|
"""
|
2022-01-05 17:04:09 +00:00
|
|
|
at_start_job = HassJob(at_start_cb)
|
2022-10-05 10:24:51 +00:00
|
|
|
if check_state(hass):
|
2022-01-05 17:04:09 +00:00
|
|
|
hass.async_run_hass_job(at_start_job, hass)
|
2022-04-01 21:34:36 +00:00
|
|
|
return lambda: None
|
2021-05-31 23:35:31 +00:00
|
|
|
|
2022-05-02 14:41:14 +00:00
|
|
|
unsub: None | CALLBACK_TYPE = None
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _matched_event(event: Event) -> None:
|
2021-05-31 23:35:31 +00:00
|
|
|
"""Call the callback when Home Assistant started."""
|
2022-01-05 17:04:09 +00:00
|
|
|
hass.async_run_hass_job(at_start_job, hass)
|
2022-05-02 14:41:14 +00:00
|
|
|
nonlocal unsub
|
|
|
|
unsub = None
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def cancel() -> None:
|
|
|
|
if unsub:
|
|
|
|
unsub()
|
2021-05-31 23:35:31 +00:00
|
|
|
|
2022-10-05 10:24:51 +00:00
|
|
|
unsub = hass.bus.async_listen_once(event_type, _matched_event)
|
2022-05-02 14:41:14 +00:00
|
|
|
return cancel
|
2022-10-05 10:24:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_at_start(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
at_start_cb: Callable[[HomeAssistant], Coroutine[Any, Any, None] | None],
|
|
|
|
) -> CALLBACK_TYPE:
|
|
|
|
"""Execute a job at_start_cb when Home Assistant is starting.
|
|
|
|
|
|
|
|
The job is executed immediately if Home Assistant is already starting or started.
|
|
|
|
Will wait for EVENT_HOMEASSISTANT_START if it isn't.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def _is_running(hass: HomeAssistant) -> bool:
|
|
|
|
return hass.is_running
|
|
|
|
|
|
|
|
return _async_at_core_state(
|
|
|
|
hass, at_start_cb, EVENT_HOMEASSISTANT_START, _is_running
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_at_started(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
at_start_cb: Callable[[HomeAssistant], Coroutine[Any, Any, None] | None],
|
|
|
|
) -> CALLBACK_TYPE:
|
|
|
|
"""Execute a job at_start_cb when Home Assistant has started.
|
|
|
|
|
|
|
|
The job is executed immediately if Home Assistant is already started.
|
|
|
|
Will wait for EVENT_HOMEASSISTANT_STARTED if it isn't.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def _is_started(hass: HomeAssistant) -> bool:
|
|
|
|
return hass.state == CoreState.running
|
|
|
|
|
|
|
|
return _async_at_core_state(
|
|
|
|
hass, at_start_cb, EVENT_HOMEASSISTANT_STARTED, _is_started
|
|
|
|
)
|