2020-08-20 13:06:41 +00:00
|
|
|
"""TemplateEntity utility class."""
|
2021-03-18 13:43:52 +00:00
|
|
|
from __future__ import annotations
|
2020-08-20 13:06:41 +00:00
|
|
|
|
|
|
|
import logging
|
2021-03-18 13:43:52 +00:00
|
|
|
from typing import Any, Callable
|
2020-08-20 13:06:41 +00:00
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
2020-09-12 12:20:21 +00:00
|
|
|
from homeassistant.const import ATTR_ENTITY_ID
|
2020-08-21 23:31:48 +00:00
|
|
|
from homeassistant.core import EVENT_HOMEASSISTANT_START, CoreState, callback
|
2020-08-20 13:06:41 +00:00
|
|
|
from homeassistant.exceptions import TemplateError
|
2020-08-20 13:51:27 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2020-08-20 13:06:41 +00:00
|
|
|
from homeassistant.helpers.entity import Entity
|
2020-09-01 00:07:40 +00:00
|
|
|
from homeassistant.helpers.event import (
|
|
|
|
Event,
|
|
|
|
TrackTemplate,
|
|
|
|
TrackTemplateResult,
|
|
|
|
async_track_template_result,
|
|
|
|
)
|
2020-08-20 13:32:52 +00:00
|
|
|
from homeassistant.helpers.template import Template, result_as_boolean
|
2020-08-20 13:06:41 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
class _TemplateAttribute:
|
|
|
|
"""Attribute value linked to template result."""
|
|
|
|
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
entity: Entity,
|
|
|
|
attribute: str,
|
|
|
|
template: Template,
|
2020-08-21 12:33:53 +00:00
|
|
|
validator: Callable[[Any], Any] = None,
|
2021-03-18 13:43:52 +00:00
|
|
|
on_update: Callable[[Any], None] | None = None,
|
|
|
|
none_on_template_error: bool | None = False,
|
2020-08-20 13:06:41 +00:00
|
|
|
):
|
|
|
|
"""Template attribute."""
|
|
|
|
self._entity = entity
|
|
|
|
self._attribute = attribute
|
|
|
|
self.template = template
|
|
|
|
self.validator = validator
|
|
|
|
self.on_update = on_update
|
|
|
|
self.async_update = None
|
2020-08-20 13:53:45 +00:00
|
|
|
self.none_on_template_error = none_on_template_error
|
2020-08-20 13:06:41 +00:00
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_setup(self):
|
|
|
|
"""Config update path for the attribute."""
|
|
|
|
if self.on_update:
|
|
|
|
return
|
|
|
|
|
|
|
|
if not hasattr(self._entity, self._attribute):
|
|
|
|
raise AttributeError(f"Attribute '{self._attribute}' does not exist.")
|
|
|
|
|
|
|
|
self.on_update = self._default_update
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _default_update(self, result):
|
|
|
|
attr_result = None if isinstance(result, TemplateError) else result
|
|
|
|
setattr(self._entity, self._attribute, attr_result)
|
|
|
|
|
|
|
|
@callback
|
2020-09-01 00:07:40 +00:00
|
|
|
def handle_result(
|
2020-08-20 13:06:41 +00:00
|
|
|
self,
|
2021-03-18 13:43:52 +00:00
|
|
|
event: Event | None,
|
2020-08-20 13:06:41 +00:00
|
|
|
template: Template,
|
2021-03-18 13:43:52 +00:00
|
|
|
last_result: str | None | TemplateError,
|
|
|
|
result: str | TemplateError,
|
2020-08-20 13:06:41 +00:00
|
|
|
) -> None:
|
2020-09-01 00:07:40 +00:00
|
|
|
"""Handle a template result event callback."""
|
2020-08-20 13:06:41 +00:00
|
|
|
if isinstance(result, TemplateError):
|
|
|
|
_LOGGER.error(
|
|
|
|
"TemplateError('%s') "
|
|
|
|
"while processing template '%s' "
|
|
|
|
"for attribute '%s' in entity '%s'",
|
|
|
|
result,
|
|
|
|
self.template,
|
|
|
|
self._attribute,
|
|
|
|
self._entity.entity_id,
|
|
|
|
)
|
2020-08-20 13:53:45 +00:00
|
|
|
if self.none_on_template_error:
|
|
|
|
self._default_update(result)
|
|
|
|
else:
|
|
|
|
self.on_update(result)
|
2020-08-20 13:06:41 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
if not self.validator:
|
|
|
|
self.on_update(result)
|
|
|
|
return
|
|
|
|
|
|
|
|
try:
|
|
|
|
validated = self.validator(result)
|
|
|
|
except vol.Invalid as ex:
|
|
|
|
_LOGGER.error(
|
|
|
|
"Error validating template result '%s' "
|
|
|
|
"from template '%s' "
|
|
|
|
"for attribute '%s' in entity %s "
|
|
|
|
"validation message '%s'",
|
|
|
|
result,
|
|
|
|
self.template,
|
|
|
|
self._attribute,
|
|
|
|
self._entity.entity_id,
|
|
|
|
ex.msg,
|
|
|
|
)
|
|
|
|
self.on_update(None)
|
|
|
|
return
|
|
|
|
|
|
|
|
self.on_update(validated)
|
2020-09-01 00:07:40 +00:00
|
|
|
return
|
2020-08-20 13:06:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
class TemplateEntity(Entity):
|
|
|
|
"""Entity that uses templates to calculate attributes."""
|
|
|
|
|
2020-08-21 12:33:53 +00:00
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
*,
|
|
|
|
availability_template=None,
|
|
|
|
icon_template=None,
|
|
|
|
entity_picture_template=None,
|
|
|
|
attribute_templates=None,
|
|
|
|
):
|
2020-08-20 13:06:41 +00:00
|
|
|
"""Template Entity."""
|
2020-09-01 00:07:40 +00:00
|
|
|
self._template_attrs = {}
|
|
|
|
self._async_update = None
|
2020-08-21 12:33:53 +00:00
|
|
|
self._attribute_templates = attribute_templates
|
|
|
|
self._attributes = {}
|
|
|
|
self._availability_template = availability_template
|
|
|
|
self._available = True
|
|
|
|
self._icon_template = icon_template
|
|
|
|
self._entity_picture_template = entity_picture_template
|
|
|
|
self._icon = None
|
|
|
|
self._entity_picture = None
|
2020-09-12 12:20:21 +00:00
|
|
|
self._self_ref_update_count = 0
|
2020-08-20 13:06:41 +00:00
|
|
|
|
2020-08-20 13:32:52 +00:00
|
|
|
@property
|
|
|
|
def should_poll(self):
|
|
|
|
"""No polling needed."""
|
|
|
|
return False
|
|
|
|
|
2020-08-21 12:33:53 +00:00
|
|
|
@callback
|
|
|
|
def _update_available(self, result):
|
|
|
|
if isinstance(result, TemplateError):
|
|
|
|
self._available = True
|
|
|
|
return
|
|
|
|
|
|
|
|
self._available = result_as_boolean(result)
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _update_state(self, result):
|
|
|
|
if self._availability_template:
|
|
|
|
return
|
|
|
|
|
|
|
|
self._available = not isinstance(result, TemplateError)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def available(self) -> bool:
|
|
|
|
"""Return if the device is available."""
|
|
|
|
return self._available
|
|
|
|
|
|
|
|
@property
|
|
|
|
def icon(self):
|
|
|
|
"""Return the icon to use in the frontend, if any."""
|
|
|
|
return self._icon
|
|
|
|
|
|
|
|
@property
|
|
|
|
def entity_picture(self):
|
|
|
|
"""Return the entity_picture to use in the frontend, if any."""
|
|
|
|
return self._entity_picture
|
|
|
|
|
|
|
|
@property
|
2021-03-11 19:16:26 +00:00
|
|
|
def extra_state_attributes(self):
|
2020-08-21 12:33:53 +00:00
|
|
|
"""Return the state attributes."""
|
|
|
|
return self._attributes
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _add_attribute_template(self, attribute_key, attribute_template):
|
|
|
|
"""Create a template tracker for the attribute."""
|
|
|
|
|
|
|
|
def _update_attribute(result):
|
|
|
|
attr_result = None if isinstance(result, TemplateError) else result
|
|
|
|
self._attributes[attribute_key] = attr_result
|
|
|
|
|
|
|
|
self.add_template_attribute(
|
|
|
|
attribute_key, attribute_template, None, _update_attribute
|
|
|
|
)
|
|
|
|
|
2020-08-20 13:06:41 +00:00
|
|
|
def add_template_attribute(
|
|
|
|
self,
|
|
|
|
attribute: str,
|
|
|
|
template: Template,
|
2020-08-21 12:33:53 +00:00
|
|
|
validator: Callable[[Any], Any] = None,
|
2021-03-18 13:43:52 +00:00
|
|
|
on_update: Callable[[Any], None] | None = None,
|
2020-08-20 13:53:45 +00:00
|
|
|
none_on_template_error: bool = False,
|
2020-08-20 13:06:41 +00:00
|
|
|
) -> None:
|
|
|
|
"""
|
|
|
|
Call in the constructor to add a template linked to a attribute.
|
|
|
|
|
|
|
|
Parameters
|
|
|
|
----------
|
|
|
|
attribute
|
|
|
|
The name of the attribute to link to. This attribute must exist
|
|
|
|
unless a custom on_update method is supplied.
|
|
|
|
template
|
|
|
|
The template to calculate.
|
|
|
|
validator
|
|
|
|
Validator function to parse the result and ensure it's valid.
|
|
|
|
on_update
|
|
|
|
Called to store the template result rather than storing it
|
|
|
|
the supplied attribute. Passed the result of the validator, or None
|
|
|
|
if the template or validator resulted in an error.
|
|
|
|
|
|
|
|
"""
|
2021-04-12 05:49:09 +00:00
|
|
|
assert self.hass is not None, "hass cannot be None"
|
|
|
|
template.hass = self.hass
|
2020-08-20 13:53:45 +00:00
|
|
|
attribute = _TemplateAttribute(
|
|
|
|
self, attribute, template, validator, on_update, none_on_template_error
|
|
|
|
)
|
2020-09-01 00:07:40 +00:00
|
|
|
self._template_attrs.setdefault(template, [])
|
|
|
|
self._template_attrs[template].append(attribute)
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _handle_results(
|
|
|
|
self,
|
2021-03-18 13:43:52 +00:00
|
|
|
event: Event | None,
|
|
|
|
updates: list[TrackTemplateResult],
|
2020-09-01 00:07:40 +00:00
|
|
|
) -> None:
|
|
|
|
"""Call back the results to the attributes."""
|
|
|
|
if event:
|
|
|
|
self.async_set_context(event.context)
|
|
|
|
|
2020-09-12 12:20:21 +00:00
|
|
|
entity_id = event and event.data.get(ATTR_ENTITY_ID)
|
|
|
|
|
|
|
|
if entity_id and entity_id == self.entity_id:
|
|
|
|
self._self_ref_update_count += 1
|
|
|
|
else:
|
|
|
|
self._self_ref_update_count = 0
|
|
|
|
|
2020-10-01 20:11:11 +00:00
|
|
|
if self._self_ref_update_count > len(self._template_attrs):
|
2020-09-12 12:20:21 +00:00
|
|
|
for update in updates:
|
|
|
|
_LOGGER.warning(
|
|
|
|
"Template loop detected while processing event: %s, skipping template render for Template[%s]",
|
|
|
|
event,
|
|
|
|
update.template.template,
|
|
|
|
)
|
|
|
|
return
|
|
|
|
|
2020-09-01 00:07:40 +00:00
|
|
|
for update in updates:
|
|
|
|
for attr in self._template_attrs[update.template]:
|
|
|
|
attr.handle_result(
|
|
|
|
event, update.template, update.last_result, update.result
|
|
|
|
)
|
|
|
|
|
2020-09-22 14:28:02 +00:00
|
|
|
self.async_write_ha_state()
|
2020-08-20 13:06:41 +00:00
|
|
|
|
2020-08-21 23:31:48 +00:00
|
|
|
async def _async_template_startup(self, *_) -> None:
|
2020-09-22 14:28:02 +00:00
|
|
|
template_var_tups = []
|
|
|
|
for template, attributes in self._template_attrs.items():
|
|
|
|
template_var_tups.append(TrackTemplate(template, None))
|
|
|
|
for attribute in attributes:
|
|
|
|
attribute.async_setup()
|
2020-09-01 00:07:40 +00:00
|
|
|
|
|
|
|
result_info = async_track_template_result(
|
|
|
|
self.hass, template_var_tups, self._handle_results
|
|
|
|
)
|
|
|
|
self.async_on_remove(result_info.async_remove)
|
|
|
|
self._async_update = result_info.async_refresh
|
2020-09-22 14:28:02 +00:00
|
|
|
result_info.async_refresh()
|
2020-08-20 13:06:41 +00:00
|
|
|
|
|
|
|
async def async_added_to_hass(self) -> None:
|
|
|
|
"""Run when entity about to be added to hass."""
|
2020-08-20 13:32:52 +00:00
|
|
|
if self._availability_template is not None:
|
|
|
|
self.add_template_attribute(
|
|
|
|
"_available", self._availability_template, None, self._update_available
|
|
|
|
)
|
2020-08-21 12:33:53 +00:00
|
|
|
if self._attribute_templates is not None:
|
|
|
|
for key, value in self._attribute_templates.items():
|
|
|
|
self._add_attribute_template(key, value)
|
2020-08-20 13:51:27 +00:00
|
|
|
if self._icon_template is not None:
|
|
|
|
self.add_template_attribute(
|
|
|
|
"_icon", self._icon_template, vol.Or(cv.whitespace, cv.icon)
|
|
|
|
)
|
|
|
|
if self._entity_picture_template is not None:
|
|
|
|
self.add_template_attribute(
|
|
|
|
"_entity_picture", self._entity_picture_template
|
|
|
|
)
|
2020-08-21 23:31:48 +00:00
|
|
|
if self.hass.state == CoreState.running:
|
|
|
|
await self._async_template_startup()
|
|
|
|
return
|
2020-08-20 13:51:27 +00:00
|
|
|
|
2020-08-21 12:33:53 +00:00
|
|
|
self.hass.bus.async_listen_once(
|
|
|
|
EVENT_HOMEASSISTANT_START, self._async_template_startup
|
2020-08-20 14:07:58 +00:00
|
|
|
)
|
|
|
|
|
2020-08-21 12:33:53 +00:00
|
|
|
async def async_update(self) -> None:
|
|
|
|
"""Call for forced update."""
|
2020-09-01 00:07:40 +00:00
|
|
|
self._async_update()
|