core/homeassistant/components/automation/numeric_state.py

111 lines
3.4 KiB
Python
Raw Normal View History

"""
homeassistant.components.automation.numeric_state
2015-11-09 12:12:18 +00:00
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Offers numeric state listening automation rules.
2015-10-13 19:07:53 +00:00
For more details about this automation rule, please refer to the documentation
2015-11-09 12:12:18 +00:00
at https://home-assistant.io/components/automation/#numeric-state-trigger
"""
import logging
2015-12-14 18:08:31 +00:00
from homeassistant.const import CONF_VALUE_TEMPLATE
from homeassistant.helpers.event import track_state_change
2015-12-14 18:08:31 +00:00
from homeassistant.util import template
2015-09-15 05:05:40 +00:00
CONF_ENTITY_ID = "entity_id"
CONF_BELOW = "below"
CONF_ABOVE = "above"
_LOGGER = logging.getLogger(__name__)
2015-09-13 11:05:36 +00:00
2015-09-14 05:25:42 +00:00
def trigger(hass, config, action):
""" Listen for state changes based on `config`. """
entity_id = config.get(CONF_ENTITY_ID)
if entity_id is None:
_LOGGER.error("Missing configuration key %s", CONF_ENTITY_ID)
return False
below = config.get(CONF_BELOW)
above = config.get(CONF_ABOVE)
2015-12-14 18:08:31 +00:00
value_template = config.get(CONF_VALUE_TEMPLATE)
if below is None and above is None:
_LOGGER.error("Missing configuration key."
" One of %s or %s is required",
2015-09-13 11:05:36 +00:00
CONF_BELOW, CONF_ABOVE)
return False
2015-12-14 18:08:31 +00:00
if value_template is not None:
2015-12-15 15:57:30 +00:00
renderer = lambda value: template.render(hass,
value_template,
2015-12-15 17:12:43 +00:00
{'state': value})
2015-12-14 18:08:31 +00:00
else:
2015-12-15 15:57:30 +00:00
renderer = lambda value: value.state
2015-12-14 18:08:31 +00:00
# pylint: disable=unused-argument
def state_automation_listener(entity, from_s, to_s):
""" Listens for state changes and calls action. """
# Fire action if we go from outside range into range
2015-12-15 15:57:30 +00:00
if _in_range(above, below, renderer(to_s)) and \
(from_s is None or not _in_range(above, below, renderer(from_s))):
action()
track_state_change(
hass, entity_id, state_automation_listener)
return True
def if_action(hass, config):
""" Wraps action method with state based condition. """
entity_id = config.get(CONF_ENTITY_ID)
if entity_id is None:
_LOGGER.error("Missing configuration key %s", CONF_ENTITY_ID)
return None
below = config.get(CONF_BELOW)
above = config.get(CONF_ABOVE)
2015-12-14 18:08:31 +00:00
value_template = config.get(CONF_VALUE_TEMPLATE)
if below is None and above is None:
_LOGGER.error("Missing configuration key."
" One of %s or %s is required",
CONF_BELOW, CONF_ABOVE)
return None
2015-12-14 18:08:31 +00:00
if value_template is not None:
2015-12-15 15:57:30 +00:00
renderer = lambda value: template.render(hass,
value_template,
2015-12-15 17:12:43 +00:00
{'state': value})
2015-12-14 18:08:31 +00:00
else:
2015-12-15 15:57:30 +00:00
renderer = lambda value: value.state
2015-12-14 18:08:31 +00:00
def if_numeric_state():
""" Test numeric state condition. """
state = hass.states.get(entity_id)
2015-12-15 15:57:30 +00:00
return state is not None and _in_range(above, below, renderer(state))
return if_numeric_state
2015-12-15 15:57:30 +00:00
def _in_range(range_start, range_end, value):
""" Checks if value is inside the range """
try:
value = float(value)
except ValueError:
2015-11-29 21:49:05 +00:00
_LOGGER.warning("Missing value in numeric check")
return False
if range_start is not None and range_end is not None:
return float(range_start) <= value < float(range_end)
elif range_end is not None:
return value < float(range_end)
else:
return float(range_start) <= value