core/homeassistant/helpers/debounce.py

131 lines
3.9 KiB
Python
Raw Normal View History

"""Debounce helper."""
2021-03-17 17:34:19 +00:00
from __future__ import annotations
import asyncio
2022-07-19 16:35:04 +00:00
from collections.abc import Callable
from logging import Logger
2022-07-19 16:35:04 +00:00
from typing import Generic, TypeVar
from homeassistant.core import HassJob, HomeAssistant, callback
2022-07-19 16:35:04 +00:00
_R_co = TypeVar("_R_co", covariant=True)
2022-07-19 16:35:04 +00:00
class Debouncer(Generic[_R_co]):
"""Class to rate limit calls to a specific command."""
def __init__(
self,
hass: HomeAssistant,
logger: Logger,
*,
cooldown: float,
immediate: bool,
2022-07-19 16:35:04 +00:00
function: Callable[[], _R_co] | None = None,
) -> None:
"""Initialize debounce.
immediate: indicate if the function needs to be called right away and
wait <cooldown> until executing next invocation.
function: optional and can be instantiated later.
"""
self.hass = hass
self.logger = logger
self._function = function
self.cooldown = cooldown
self.immediate = immediate
2021-03-17 17:34:19 +00:00
self._timer_task: asyncio.TimerHandle | None = None
self._execute_at_end_of_timer: bool = False
2020-02-26 19:27:37 +00:00
self._execute_lock = asyncio.Lock()
2022-07-19 16:35:04 +00:00
self._job: HassJob[[], _R_co] | None = (
None if function is None else HassJob(function)
)
@property
2022-07-19 16:35:04 +00:00
def function(self) -> Callable[[], _R_co] | None:
"""Return the function being wrapped by the Debouncer."""
return self._function
@function.setter
2022-07-19 16:35:04 +00:00
def function(self, function: Callable[[], _R_co]) -> None:
"""Update the function being wrapped by the Debouncer."""
self._function = function
if self._job is None or function != self._job.target:
self._job = HassJob(function)
async def async_call(self) -> None:
"""Call the function."""
assert self._job is not None
if self._timer_task:
if not self._execute_at_end_of_timer:
self._execute_at_end_of_timer = True
return
2020-02-26 19:27:37 +00:00
# Locked means a call is in progress. Any call is good, so abort.
if self._execute_lock.locked():
return
if not self.immediate:
self._execute_at_end_of_timer = True
2020-02-26 19:27:37 +00:00
self._schedule_timer()
return
2020-02-26 19:27:37 +00:00
async with self._execute_lock:
# Abort if timer got set while we're waiting for the lock.
if self._timer_task:
return
task = self.hass.async_run_hass_job(self._job)
if task:
await task
2020-02-26 19:27:37 +00:00
self._schedule_timer()
async def _handle_timer_finish(self) -> None:
"""Handle a finished timer."""
assert self._job is not None
self._timer_task = None
if not self._execute_at_end_of_timer:
return
self._execute_at_end_of_timer = False
2020-02-26 19:27:37 +00:00
# Locked means a call is in progress. Any call is good, so abort.
if self._execute_lock.locked():
return
async with self._execute_lock:
# Abort if timer got set while we're waiting for the lock.
if self._timer_task:
return # type: ignore[unreachable]
2020-02-26 19:27:37 +00:00
try:
task = self.hass.async_run_hass_job(self._job)
if task:
await task
2020-02-26 19:27:37 +00:00
except Exception: # pylint: disable=broad-except
self.logger.exception("Unexpected exception from %s", self.function)
self._schedule_timer()
@callback
def async_cancel(self) -> None:
"""Cancel any scheduled call."""
if self._timer_task:
self._timer_task.cancel()
self._timer_task = None
self._execute_at_end_of_timer = False
2020-02-26 19:27:37 +00:00
@callback
def _schedule_timer(self) -> None:
"""Schedule a timer."""
self._timer_task = self.hass.loop.call_later(
self.cooldown,
lambda: self.hass.async_create_task(self._handle_timer_finish()),
)