mirror of
https://github.com/home-assistant/core.git
synced 2025-07-09 06:17:07 +00:00
Template sensors to not track all state changes (#17276)
* Disable template sensor match all * Only manual update template sensors that match all
This commit is contained in:
parent
e5c3a4be80
commit
a1dac28e4b
@ -57,22 +57,36 @@ async def async_setup_platform(hass, config, async_add_entities,
|
|||||||
|
|
||||||
entity_ids = set()
|
entity_ids = set()
|
||||||
manual_entity_ids = device_config.get(ATTR_ENTITY_ID)
|
manual_entity_ids = device_config.get(ATTR_ENTITY_ID)
|
||||||
|
invalid_templates = []
|
||||||
|
|
||||||
for template in (state_template, icon_template,
|
for tpl_name, template in (
|
||||||
entity_picture_template, friendly_name_template):
|
(CONF_VALUE_TEMPLATE, state_template),
|
||||||
|
(CONF_ICON_TEMPLATE, icon_template),
|
||||||
|
(CONF_ENTITY_PICTURE_TEMPLATE, entity_picture_template),
|
||||||
|
(CONF_FRIENDLY_NAME_TEMPLATE, friendly_name_template),
|
||||||
|
):
|
||||||
if template is None:
|
if template is None:
|
||||||
continue
|
continue
|
||||||
template.hass = hass
|
template.hass = hass
|
||||||
|
|
||||||
if entity_ids == MATCH_ALL or manual_entity_ids is not None:
|
if manual_entity_ids is not None:
|
||||||
continue
|
continue
|
||||||
|
|
||||||
template_entity_ids = template.extract_entities()
|
template_entity_ids = template.extract_entities()
|
||||||
if template_entity_ids == MATCH_ALL:
|
if template_entity_ids == MATCH_ALL:
|
||||||
entity_ids = MATCH_ALL
|
entity_ids = MATCH_ALL
|
||||||
else:
|
# Cut off _template from name
|
||||||
|
invalid_templates.append(tpl_name[:-9])
|
||||||
|
elif entity_ids != MATCH_ALL:
|
||||||
entity_ids |= set(template_entity_ids)
|
entity_ids |= set(template_entity_ids)
|
||||||
|
|
||||||
|
if invalid_templates:
|
||||||
|
_LOGGER.warning(
|
||||||
|
'Template sensor %s has no entity ids configured to track nor'
|
||||||
|
' were we able to extract the entities to track from the %s '
|
||||||
|
'template(s). This entity will only be able to be updated '
|
||||||
|
'manually.', device, ', '.join(invalid_templates))
|
||||||
|
|
||||||
if manual_entity_ids is not None:
|
if manual_entity_ids is not None:
|
||||||
entity_ids = manual_entity_ids
|
entity_ids = manual_entity_ids
|
||||||
elif entity_ids != MATCH_ALL:
|
elif entity_ids != MATCH_ALL:
|
||||||
@ -123,6 +137,10 @@ class SensorTemplate(Entity):
|
|||||||
|
|
||||||
async def async_added_to_hass(self):
|
async def async_added_to_hass(self):
|
||||||
"""Register callbacks."""
|
"""Register callbacks."""
|
||||||
|
# We don't render on every update
|
||||||
|
if self._entities == MATCH_ALL:
|
||||||
|
return
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def template_sensor_state_listener(entity, old_state, new_state):
|
def template_sensor_state_listener(entity, old_state, new_state):
|
||||||
"""Handle device state changes."""
|
"""Handle device state changes."""
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
"""The test for the Template sensor platform."""
|
"""The test for the Template sensor platform."""
|
||||||
from homeassistant.setup import setup_component
|
from homeassistant.setup import setup_component, async_setup_component
|
||||||
|
|
||||||
from tests.common import get_test_home_assistant, assert_setup_component
|
from tests.common import get_test_home_assistant, assert_setup_component
|
||||||
|
|
||||||
@ -52,7 +52,8 @@ class TestTemplateSensor:
|
|||||||
'platform': 'template',
|
'platform': 'template',
|
||||||
'sensors': {
|
'sensors': {
|
||||||
'test_template_sensor': {
|
'test_template_sensor': {
|
||||||
'value_template': "State",
|
'value_template':
|
||||||
|
"{{ states.sensor.test_state.state }}",
|
||||||
'icon_template':
|
'icon_template':
|
||||||
"{% if states.sensor.test_state.state == "
|
"{% if states.sensor.test_state.state == "
|
||||||
"'Works' %}"
|
"'Works' %}"
|
||||||
@ -82,7 +83,8 @@ class TestTemplateSensor:
|
|||||||
'platform': 'template',
|
'platform': 'template',
|
||||||
'sensors': {
|
'sensors': {
|
||||||
'test_template_sensor': {
|
'test_template_sensor': {
|
||||||
'value_template': "State",
|
'value_template':
|
||||||
|
"{{ states.sensor.test_state.state }}",
|
||||||
'entity_picture_template':
|
'entity_picture_template':
|
||||||
"{% if states.sensor.test_state.state == "
|
"{% if states.sensor.test_state.state == "
|
||||||
"'Works' %}"
|
"'Works' %}"
|
||||||
@ -112,7 +114,8 @@ class TestTemplateSensor:
|
|||||||
'platform': 'template',
|
'platform': 'template',
|
||||||
'sensors': {
|
'sensors': {
|
||||||
'test_template_sensor': {
|
'test_template_sensor': {
|
||||||
'value_template': "State",
|
'value_template':
|
||||||
|
"{{ states.sensor.test_state.state }}",
|
||||||
'friendly_name_template':
|
'friendly_name_template':
|
||||||
"It {{ states.sensor.test_state.state }}."
|
"It {{ states.sensor.test_state.state }}."
|
||||||
}
|
}
|
||||||
@ -276,7 +279,8 @@ class TestTemplateSensor:
|
|||||||
'platform': 'template',
|
'platform': 'template',
|
||||||
'sensors': {
|
'sensors': {
|
||||||
'test': {
|
'test': {
|
||||||
'value_template': '{{ foo }}',
|
'value_template':
|
||||||
|
'{{ states.sensor.test_sensor.state }}',
|
||||||
'device_class': 'foobarnotreal',
|
'device_class': 'foobarnotreal',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@ -291,10 +295,14 @@ class TestTemplateSensor:
|
|||||||
'platform': 'template',
|
'platform': 'template',
|
||||||
'sensors': {
|
'sensors': {
|
||||||
'test1': {
|
'test1': {
|
||||||
'value_template': '{{ foo }}',
|
'value_template':
|
||||||
|
'{{ states.sensor.test_sensor.state }}',
|
||||||
'device_class': 'temperature',
|
'device_class': 'temperature',
|
||||||
},
|
},
|
||||||
'test2': {'value_template': '{{ foo }}'},
|
'test2': {
|
||||||
|
'value_template':
|
||||||
|
'{{ states.sensor.test_sensor.state }}'
|
||||||
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
@ -304,3 +312,73 @@ class TestTemplateSensor:
|
|||||||
assert state.attributes['device_class'] == 'temperature'
|
assert state.attributes['device_class'] == 'temperature'
|
||||||
state = self.hass.states.get('sensor.test2')
|
state = self.hass.states.get('sensor.test2')
|
||||||
assert 'device_class' not in state.attributes
|
assert 'device_class' not in state.attributes
|
||||||
|
|
||||||
|
|
||||||
|
async def test_no_template_match_all(hass, caplog):
|
||||||
|
"""Test that we do not allow sensors that match on all."""
|
||||||
|
await async_setup_component(hass, 'sensor', {
|
||||||
|
'sensor': {
|
||||||
|
'platform': 'template',
|
||||||
|
'sensors': {
|
||||||
|
'invalid_state': {
|
||||||
|
'value_template': '{{ 1 + 1 }}',
|
||||||
|
},
|
||||||
|
'invalid_icon': {
|
||||||
|
'value_template':
|
||||||
|
'{{ states.sensor.test_sensor.state }}',
|
||||||
|
'icon_template': '{{ 1 + 1 }}',
|
||||||
|
},
|
||||||
|
'invalid_entity_picture': {
|
||||||
|
'value_template':
|
||||||
|
'{{ states.sensor.test_sensor.state }}',
|
||||||
|
'entity_picture_template': '{{ 1 + 1 }}',
|
||||||
|
},
|
||||||
|
'invalid_friendly_name': {
|
||||||
|
'value_template':
|
||||||
|
'{{ states.sensor.test_sensor.state }}',
|
||||||
|
'friendly_name_template': '{{ 1 + 1 }}',
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
assert len(hass.states.async_all()) == 4
|
||||||
|
assert ('Template sensor invalid_state has no entity ids '
|
||||||
|
'configured to track nor were we able to extract the entities to '
|
||||||
|
'track from the value template') in caplog.text
|
||||||
|
assert ('Template sensor invalid_icon has no entity ids '
|
||||||
|
'configured to track nor were we able to extract the entities to '
|
||||||
|
'track from the icon template') in caplog.text
|
||||||
|
assert ('Template sensor invalid_entity_picture has no entity ids '
|
||||||
|
'configured to track nor were we able to extract the entities to '
|
||||||
|
'track from the entity_picture template') in caplog.text
|
||||||
|
assert ('Template sensor invalid_friendly_name has no entity ids '
|
||||||
|
'configured to track nor were we able to extract the entities to '
|
||||||
|
'track from the friendly_name template') in caplog.text
|
||||||
|
|
||||||
|
assert hass.states.get('sensor.invalid_state').state == 'unknown'
|
||||||
|
assert hass.states.get('sensor.invalid_icon').state == 'unknown'
|
||||||
|
assert hass.states.get('sensor.invalid_entity_picture').state == 'unknown'
|
||||||
|
assert hass.states.get('sensor.invalid_friendly_name').state == 'unknown'
|
||||||
|
|
||||||
|
hass.states.async_set('sensor.test_sensor', 'hello')
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
assert hass.states.get('sensor.invalid_state').state == 'unknown'
|
||||||
|
assert hass.states.get('sensor.invalid_icon').state == 'unknown'
|
||||||
|
assert hass.states.get('sensor.invalid_entity_picture').state == 'unknown'
|
||||||
|
assert hass.states.get('sensor.invalid_friendly_name').state == 'unknown'
|
||||||
|
|
||||||
|
await hass.helpers.entity_component.async_update_entity(
|
||||||
|
'sensor.invalid_state')
|
||||||
|
await hass.helpers.entity_component.async_update_entity(
|
||||||
|
'sensor.invalid_icon')
|
||||||
|
await hass.helpers.entity_component.async_update_entity(
|
||||||
|
'sensor.invalid_entity_picture')
|
||||||
|
await hass.helpers.entity_component.async_update_entity(
|
||||||
|
'sensor.invalid_friendly_name')
|
||||||
|
|
||||||
|
assert hass.states.get('sensor.invalid_state').state == '2'
|
||||||
|
assert hass.states.get('sensor.invalid_icon').state == 'hello'
|
||||||
|
assert hass.states.get('sensor.invalid_entity_picture').state == 'hello'
|
||||||
|
assert hass.states.get('sensor.invalid_friendly_name').state == 'hello'
|
||||||
|
Loading…
x
Reference in New Issue
Block a user