core/homeassistant/helpers/state.py

135 lines
3.7 KiB
Python
Raw Normal View History

"""Helpers that help with state related things."""
import asyncio
from collections import defaultdict
import datetime as dt
2015-03-16 06:36:42 +00:00
import logging
from types import ModuleType, TracebackType
from typing import Dict, Iterable, List, Optional, Type, Union
2015-03-16 06:36:42 +00:00
2019-07-31 19:25:30 +00:00
from homeassistant.components.sun import STATE_ABOVE_HORIZON, STATE_BELOW_HORIZON
2015-03-17 06:32:18 +00:00
from homeassistant.const import (
2019-07-31 19:25:30 +00:00
STATE_CLOSED,
STATE_HOME,
STATE_LOCKED,
STATE_NOT_HOME,
STATE_OFF,
STATE_ON,
STATE_OPEN,
STATE_UNKNOWN,
STATE_UNLOCKED,
)
from homeassistant.core import Context, State
from homeassistant.loader import IntegrationNotFound, async_get_integration, bind_hass
import homeassistant.util.dt as dt_util
from .typing import HomeAssistantType
2015-03-16 06:36:42 +00:00
_LOGGER = logging.getLogger(__name__)
class AsyncTrackStates:
"""
Record the time when the with-block is entered.
2016-03-07 22:39:52 +00:00
Add all states that have changed since the start time to the return list
when with-block is exited.
Must be run within the event loop.
2015-03-16 06:36:42 +00:00
"""
def __init__(self, hass: HomeAssistantType) -> None:
"""Initialize a TrackStates block."""
2015-03-16 06:36:42 +00:00
self.hass = hass
self.states: List[State] = []
2015-03-16 06:36:42 +00:00
# pylint: disable=attribute-defined-outside-init
def __enter__(self) -> List[State]:
"""Record time from which to track changes."""
self.now = dt_util.utcnow()
2015-03-16 06:36:42 +00:00
return self.states
2019-07-31 19:25:30 +00:00
def __exit__(
self,
exc_type: Optional[Type[BaseException]],
exc_value: Optional[BaseException],
traceback: Optional[TracebackType],
) -> None:
"""Add changes states to changes list."""
2019-07-31 19:25:30 +00:00
self.states.extend(get_changed_since(self.hass.states.async_all(), self.now))
2019-07-31 19:25:30 +00:00
def get_changed_since(
states: Iterable[State], utc_point_in_time: dt.datetime
) -> List[State]:
"""Return list of states that have been changed since utc_point_in_time."""
2019-07-31 19:25:30 +00:00
return [state for state in states if state.last_updated >= utc_point_in_time]
2015-03-16 06:36:42 +00:00
@bind_hass
async def async_reproduce_state(
2019-07-31 19:25:30 +00:00
hass: HomeAssistantType,
states: Union[State, Iterable[State]],
blocking: bool = False,
context: Optional[Context] = None,
) -> None:
"""Reproduce a list of states on multiple domains."""
2015-03-16 06:36:42 +00:00
if isinstance(states, State):
states = [states]
to_call: Dict[str, List[State]] = defaultdict(list)
for state in states:
to_call[state.domain].append(state)
async def worker(domain: str, states_by_domain: List[State]) -> None:
try:
integration = await async_get_integration(hass, domain)
except IntegrationNotFound:
_LOGGER.warning(
"Trying to reproduce state for unknown integration: %s", domain
)
return
try:
platform: Optional[ModuleType] = integration.get_platform("reproduce_state")
except ImportError:
_LOGGER.warning("Integration %s does not support reproduce state", domain)
return
await platform.async_reproduce_states( # type: ignore
hass, states_by_domain, context=context
)
if to_call:
# run all domains in parallel
2019-07-31 19:25:30 +00:00
await asyncio.gather(
*(worker(domain, data) for domain, data in to_call.items())
)
def state_as_number(state: State) -> float:
"""
Try to coerce our state to a number.
Raises ValueError if this is not possible.
"""
2019-07-31 19:25:30 +00:00
if state.state in (
STATE_ON,
STATE_LOCKED,
STATE_ABOVE_HORIZON,
STATE_OPEN,
STATE_HOME,
):
return 1
2019-07-31 19:25:30 +00:00
if state.state in (
STATE_OFF,
STATE_UNLOCKED,
STATE_UNKNOWN,
STATE_BELOW_HORIZON,
STATE_CLOSED,
STATE_NOT_HOME,
):
return 0
return float(state.state)