core/tests/helpers/test_template.py

691 lines
25 KiB
Python
Raw Normal View History

2016-03-09 09:25:50 +00:00
"""Test Home Assistant template helper methods."""
2016-11-11 06:57:44 +00:00
from datetime import datetime
2015-12-10 00:20:09 +00:00
import unittest
from unittest.mock import patch
2016-02-21 19:13:40 +00:00
from homeassistant.components import group
from homeassistant.exceptions import TemplateError
2016-02-23 20:06:50 +00:00
from homeassistant.helpers import template
2016-08-09 03:42:25 +00:00
from homeassistant.util.unit_system import UnitSystem
Add unit system support Add unit symbol constants Initial unit system object Import more constants Pydoc for unit system file Import constants for configuration validation Unit system validation method Typing for constants Inches are valid lengths too Typings Change base class to dict - needed for remote api call serialization Validation Use dictionary keys Defined unit systems Update location util to use metric instead of us fahrenheit Update constant imports Import defined unit systems Update configuration to use unit system Update schema to use unit system Update constants Add imports to core for unit system and distance Type for config Default unit system Convert distance from HASS instance Update temperature conversion to use unit system Update temperature conversion Set unit system based on configuration Set info unit system Return unit system dictionary with config dictionary Auto discover unit system Update location test for use metric Update forecast unit system Update mold indicator unit system Update thermostat unit system Update thermostat demo test Unit tests around unit system Update test common hass configuration Update configuration unit tests There should always be a unit system! Update core unit tests Constants typing Linting issues Remove unused import Update fitbit sensor to use application unit system Update google travel time to use application unit system Update configuration example Update dht sensor Update DHT temperature conversion to use the utility function Update swagger config Update my sensors metric flag Update hvac component temperature conversion HVAC conversion for temperature Pull unit from sensor type map Pull unit from sensor type map Update the temper sensor unit Update yWeather sensor unit Update hvac demo unit test Set unit test config unit system to metric Use hass unit system length for default in proximity Use the name of the system instead of temperature Use constants from const Unused import Forecasted temperature Fix calculation in case furthest distance is greater than 1000000 units Remove unneeded constants Set default length to km or miles Use constants Linting doesn't like importing just for typing Fix reference Test is expecting meters - set config to meters Use constant Use constant PyDoc for unit test Should be not in Rename to units Change unit system to be an object - not a dictionary Return tuple in conversion Move convert to temperature util Temperature conversion is now in unit system Update imports Rename to units Units is now an object Use temperature util conversion Unit system is now an object Validate and convert unit system config Return the scalar value in template distance Test is expecting meters Update unit tests around unit system Distance util returns tuple Fix location info test Set units Update unit tests Convert distance DOH Pull out the scalar from the vector Linting I really hate python linting Linting again BLARG Unit test documentation Unit test around is metric flag Break ternary statement into if/else blocks Don't use dictionary - use members is metric flag Rename constants Use is metric flag Move constants to CONST file Move to const file Raise error if unit is not expected Typing No need to return unit since only performing conversion if it can work Use constants Line wrapping Raise error if invalid value Remove subscripts from conversion as they are no longer returned as tuples No longer tuples No longer tuples Check for numeric type Fix string format to use correct variable Typing Assert errors raised Remove subscript Only convert temperature if we know the unit If no unit of measurement set - default to HASS config Convert only if we know the unit Remove subscription Fix not in clause Linting fixes Wants a boolean Clearer if-block Check if the key is in the config first Missed a couple expecting tuples Backwards compatibility No like-y ternary! Error handling around state setting Pretty unit system configuration validation More tuple crap Use is metric flag Error handling around min/max temp Explode if no unit Pull unit from config Celsius has a decimal Unused import Check if it's a temperature before we try to convert it to a temperature Linting says too many statements - combine lat/long in a fairly reasonable manner Backwards compatibility unit test Better doc
2016-07-31 20:24:49 +00:00
from homeassistant.const import (
LENGTH_METERS,
TEMP_CELSIUS,
MASS_GRAMS,
VOLUME_LITERS,
MATCH_ALL,
Add unit system support Add unit symbol constants Initial unit system object Import more constants Pydoc for unit system file Import constants for configuration validation Unit system validation method Typing for constants Inches are valid lengths too Typings Change base class to dict - needed for remote api call serialization Validation Use dictionary keys Defined unit systems Update location util to use metric instead of us fahrenheit Update constant imports Import defined unit systems Update configuration to use unit system Update schema to use unit system Update constants Add imports to core for unit system and distance Type for config Default unit system Convert distance from HASS instance Update temperature conversion to use unit system Update temperature conversion Set unit system based on configuration Set info unit system Return unit system dictionary with config dictionary Auto discover unit system Update location test for use metric Update forecast unit system Update mold indicator unit system Update thermostat unit system Update thermostat demo test Unit tests around unit system Update test common hass configuration Update configuration unit tests There should always be a unit system! Update core unit tests Constants typing Linting issues Remove unused import Update fitbit sensor to use application unit system Update google travel time to use application unit system Update configuration example Update dht sensor Update DHT temperature conversion to use the utility function Update swagger config Update my sensors metric flag Update hvac component temperature conversion HVAC conversion for temperature Pull unit from sensor type map Pull unit from sensor type map Update the temper sensor unit Update yWeather sensor unit Update hvac demo unit test Set unit test config unit system to metric Use hass unit system length for default in proximity Use the name of the system instead of temperature Use constants from const Unused import Forecasted temperature Fix calculation in case furthest distance is greater than 1000000 units Remove unneeded constants Set default length to km or miles Use constants Linting doesn't like importing just for typing Fix reference Test is expecting meters - set config to meters Use constant Use constant PyDoc for unit test Should be not in Rename to units Change unit system to be an object - not a dictionary Return tuple in conversion Move convert to temperature util Temperature conversion is now in unit system Update imports Rename to units Units is now an object Use temperature util conversion Unit system is now an object Validate and convert unit system config Return the scalar value in template distance Test is expecting meters Update unit tests around unit system Distance util returns tuple Fix location info test Set units Update unit tests Convert distance DOH Pull out the scalar from the vector Linting I really hate python linting Linting again BLARG Unit test documentation Unit test around is metric flag Break ternary statement into if/else blocks Don't use dictionary - use members is metric flag Rename constants Use is metric flag Move constants to CONST file Move to const file Raise error if unit is not expected Typing No need to return unit since only performing conversion if it can work Use constants Line wrapping Raise error if invalid value Remove subscripts from conversion as they are no longer returned as tuples No longer tuples No longer tuples Check for numeric type Fix string format to use correct variable Typing Assert errors raised Remove subscript Only convert temperature if we know the unit If no unit of measurement set - default to HASS config Convert only if we know the unit Remove subscription Fix not in clause Linting fixes Wants a boolean Clearer if-block Check if the key is in the config first Missed a couple expecting tuples Backwards compatibility No like-y ternary! Error handling around state setting Pretty unit system configuration validation More tuple crap Use is metric flag Error handling around min/max temp Explode if no unit Pull unit from config Celsius has a decimal Unused import Check if it's a temperature before we try to convert it to a temperature Linting says too many statements - combine lat/long in a fairly reasonable manner Backwards compatibility unit test Better doc
2016-07-31 20:24:49 +00:00
)
import homeassistant.util.dt as dt_util
2015-12-10 00:20:09 +00:00
2016-02-14 23:08:23 +00:00
from tests.common import get_test_home_assistant
2015-12-10 00:20:09 +00:00
class TestHelpersTemplate(unittest.TestCase):
2016-03-09 09:25:50 +00:00
"""Test the Template."""
2015-12-10 00:20:09 +00:00
# pylint: disable=invalid-name
def setUp(self):
2016-03-09 09:25:50 +00:00
"""Setup the tests."""
2016-02-14 23:08:23 +00:00
self.hass = get_test_home_assistant()
Add unit system support Add unit symbol constants Initial unit system object Import more constants Pydoc for unit system file Import constants for configuration validation Unit system validation method Typing for constants Inches are valid lengths too Typings Change base class to dict - needed for remote api call serialization Validation Use dictionary keys Defined unit systems Update location util to use metric instead of us fahrenheit Update constant imports Import defined unit systems Update configuration to use unit system Update schema to use unit system Update constants Add imports to core for unit system and distance Type for config Default unit system Convert distance from HASS instance Update temperature conversion to use unit system Update temperature conversion Set unit system based on configuration Set info unit system Return unit system dictionary with config dictionary Auto discover unit system Update location test for use metric Update forecast unit system Update mold indicator unit system Update thermostat unit system Update thermostat demo test Unit tests around unit system Update test common hass configuration Update configuration unit tests There should always be a unit system! Update core unit tests Constants typing Linting issues Remove unused import Update fitbit sensor to use application unit system Update google travel time to use application unit system Update configuration example Update dht sensor Update DHT temperature conversion to use the utility function Update swagger config Update my sensors metric flag Update hvac component temperature conversion HVAC conversion for temperature Pull unit from sensor type map Pull unit from sensor type map Update the temper sensor unit Update yWeather sensor unit Update hvac demo unit test Set unit test config unit system to metric Use hass unit system length for default in proximity Use the name of the system instead of temperature Use constants from const Unused import Forecasted temperature Fix calculation in case furthest distance is greater than 1000000 units Remove unneeded constants Set default length to km or miles Use constants Linting doesn't like importing just for typing Fix reference Test is expecting meters - set config to meters Use constant Use constant PyDoc for unit test Should be not in Rename to units Change unit system to be an object - not a dictionary Return tuple in conversion Move convert to temperature util Temperature conversion is now in unit system Update imports Rename to units Units is now an object Use temperature util conversion Unit system is now an object Validate and convert unit system config Return the scalar value in template distance Test is expecting meters Update unit tests around unit system Distance util returns tuple Fix location info test Set units Update unit tests Convert distance DOH Pull out the scalar from the vector Linting I really hate python linting Linting again BLARG Unit test documentation Unit test around is metric flag Break ternary statement into if/else blocks Don't use dictionary - use members is metric flag Rename constants Use is metric flag Move constants to CONST file Move to const file Raise error if unit is not expected Typing No need to return unit since only performing conversion if it can work Use constants Line wrapping Raise error if invalid value Remove subscripts from conversion as they are no longer returned as tuples No longer tuples No longer tuples Check for numeric type Fix string format to use correct variable Typing Assert errors raised Remove subscript Only convert temperature if we know the unit If no unit of measurement set - default to HASS config Convert only if we know the unit Remove subscription Fix not in clause Linting fixes Wants a boolean Clearer if-block Check if the key is in the config first Missed a couple expecting tuples Backwards compatibility No like-y ternary! Error handling around state setting Pretty unit system configuration validation More tuple crap Use is metric flag Error handling around min/max temp Explode if no unit Pull unit from config Celsius has a decimal Unused import Check if it's a temperature before we try to convert it to a temperature Linting says too many statements - combine lat/long in a fairly reasonable manner Backwards compatibility unit test Better doc
2016-07-31 20:24:49 +00:00
self.hass.config.units = UnitSystem('custom', TEMP_CELSIUS,
LENGTH_METERS, VOLUME_LITERS,
MASS_GRAMS)
2015-12-10 00:20:09 +00:00
# pylint: disable=invalid-name
def tearDown(self):
2016-02-23 20:06:50 +00:00
"""Stop down stuff we started."""
2015-12-10 00:20:09 +00:00
self.hass.stop()
def test_referring_states_by_entity_id(self):
"""Test referring states by entity id."""
2015-12-10 00:20:09 +00:00
self.hass.states.set('test.object', 'happy')
self.assertEqual(
'happy',
template.Template(
'{{ states.test.object.state }}', self.hass).render())
2015-12-10 00:20:09 +00:00
def test_iterating_all_states(self):
"""Test iterating all states."""
2015-12-10 00:20:09 +00:00
self.hass.states.set('test.object', 'happy')
self.hass.states.set('sensor.temperature', 10)
self.assertEqual(
'10happy',
template.Template(
'{% for state in states %}{{ state.state }}{% endfor %}',
self.hass).render())
2015-12-10 00:20:09 +00:00
def test_iterating_domain_states(self):
"""Test iterating domain states."""
2015-12-10 00:20:09 +00:00
self.hass.states.set('test.object', 'happy')
self.hass.states.set('sensor.back_door', 'open')
self.hass.states.set('sensor.temperature', 10)
self.assertEqual(
'open10',
template.Template("""
2016-02-14 21:07:21 +00:00
{% for state in states.sensor %}{{ state.state }}{% endfor %}
""", self.hass).render())
2015-12-10 00:20:09 +00:00
2016-02-24 18:41:49 +00:00
def test_float(self):
"""Test float."""
2016-02-24 18:41:49 +00:00
self.hass.states.set('sensor.temperature', '12')
self.assertEqual(
'12.0',
template.Template(
'{{ float(states.sensor.temperature.state) }}',
self.hass).render())
2016-02-24 18:41:49 +00:00
self.assertEqual(
'True',
template.Template(
'{{ float(states.sensor.temperature.state) > 11 }}',
self.hass).render())
2016-02-24 18:41:49 +00:00
2015-12-10 00:20:09 +00:00
def test_rounding_value(self):
"""Test rounding value."""
2015-12-11 04:46:15 +00:00
self.hass.states.set('sensor.temperature', 12.78)
2015-12-10 00:20:09 +00:00
self.assertEqual(
2015-12-11 04:46:15 +00:00
'12.8',
template.Template(
'{{ states.sensor.temperature.state | round(1) }}',
self.hass).render())
2015-12-10 00:36:47 +00:00
self.assertEqual(
2015-12-12 02:45:53 +00:00
'128',
template.Template(
'{{ states.sensor.temperature.state | multiply(10) | round }}',
self.hass).render())
2016-02-21 05:59:16 +00:00
def test_rounding_value_get_original_value_on_error(self):
"""Test rounding value get original value on error."""
2016-02-21 05:59:16 +00:00
self.assertEqual(
'None',
template.Template('{{ None | round }}', self.hass).render())
2016-02-21 05:59:16 +00:00
self.assertEqual(
'no_number',
template.Template(
'{{ "no_number" | round }}', self.hass).render())
2016-02-21 05:59:16 +00:00
2016-02-21 19:12:37 +00:00
def test_multiply(self):
"""Test multiply."""
2016-02-21 19:12:37 +00:00
tests = {
None: 'None',
10: '100',
'"abcd"': 'abcd'
}
for inp, out in tests.items():
self.assertEqual(
out,
template.Template('{{ %s | multiply(10) | round }}' % inp,
self.hass).render())
2016-02-21 19:12:37 +00:00
2016-11-11 06:57:44 +00:00
def test_strptime(self):
"""Test the parse timestamp method."""
tests = [
('2016-10-19 15:22:05.588122 UTC',
'%Y-%m-%d %H:%M:%S.%f %Z', None),
('2016-10-19 15:22:05.588122+0100',
'%Y-%m-%d %H:%M:%S.%f%z', None),
('2016-10-19 15:22:05.588122',
'%Y-%m-%d %H:%M:%S.%f', None),
('2016-10-19', '%Y-%m-%d', None),
('2016', '%Y', None),
('15:22:05', '%H:%M:%S', None),
('1469119144', '%Y', '1469119144'),
('invalid', '%Y', 'invalid')
]
for inp, fmt, expected in tests:
if expected is None:
expected = datetime.strptime(inp, fmt)
temp = '{{ strptime(\'%s\', \'%s\') }}' % (inp, fmt)
self.assertEqual(
str(expected),
template.Template(temp, self.hass).render())
def test_timestamp_custom(self):
"""Test the timestamps to custom filter."""
tests = [
(None, None, None, 'None'),
(1469119144, None, True, '2016-07-21 16:39:04'),
(1469119144, '%Y', True, '2016'),
(1469119144, 'invalid', True, 'invalid'),
(dt_util.as_timestamp(dt_util.utcnow()), None, False,
dt_util.now().strftime('%Y-%m-%d %H:%M:%S'))
]
for inp, fmt, local, out in tests:
if fmt:
fil = 'timestamp_custom(\'{}\')'.format(fmt)
elif fmt and local:
fil = 'timestamp_custom(\'{0}\', {1})'.format(fmt, local)
else:
fil = 'timestamp_custom'
self.assertEqual(
out,
template.Template('{{ %s | %s }}' % (inp, fil),
self.hass).render())
2016-07-23 02:47:43 +00:00
def test_timestamp_local(self):
"""Test the timestamps to local filter."""
tests = {
None: 'None',
1469119144: '2016-07-21 16:39:04',
}
for inp, out in tests.items():
self.assertEqual(
out,
template.Template('{{ %s | timestamp_local }}' % inp,
self.hass).render())
2016-07-23 02:47:43 +00:00
def test_timestamp_utc(self):
"""Test the timestamps to local filter."""
tests = {
None: 'None',
1469119144: '2016-07-21 16:39:04',
dt_util.as_timestamp(dt_util.utcnow()):
dt_util.now().strftime('%Y-%m-%d %H:%M:%S')
}
for inp, out in tests.items():
self.assertEqual(
out,
template.Template('{{ %s | timestamp_utc }}' % inp,
self.hass).render())
2016-07-23 02:47:43 +00:00
def test_passing_vars_as_keywords(self):
"""Test passing variables as keywords."""
self.assertEqual(
'127',
template.Template('{{ hello }}', self.hass).render(hello=127))
def test_passing_vars_as_vars(self):
"""Test passing variables as variables."""
self.assertEqual(
'127',
template.Template('{{ hello }}', self.hass).render({'hello': 127}))
2015-12-11 05:38:35 +00:00
def test_render_with_possible_json_value_with_valid_json(self):
"""Render with possible JSON value with valid JSON."""
tpl = template.Template('{{ value_json.hello }}', self.hass)
2015-12-11 05:38:35 +00:00
self.assertEqual(
'world',
tpl.render_with_possible_json_value('{"hello": "world"}'))
2015-12-11 05:38:35 +00:00
def test_render_with_possible_json_value_with_invalid_json(self):
"""Render with possible JSON value with invalid JSON."""
tpl = template.Template('{{ value_json }}', self.hass)
2015-12-11 05:38:35 +00:00
self.assertEqual(
'',
tpl.render_with_possible_json_value('{ I AM NOT JSON }'))
2016-02-14 21:07:21 +00:00
def test_render_with_possible_json_value_with_template_error_value(self):
"""Render with possible JSON value with template error value."""
tpl = template.Template('{{ non_existing.variable }}', self.hass)
self.assertEqual(
'-',
tpl.render_with_possible_json_value('hello', '-'))
def test_render_with_possible_json_value_with_missing_json_value(self):
"""Render with possible JSON value with unknown JSON object."""
tpl = template.Template('{{ value_json.goodbye }}', self.hass)
self.assertEqual(
'',
tpl.render_with_possible_json_value('{"hello": "world"}'))
def test_render_with_possible_json_value_valid_with_is_defined(self):
"""Render with possible JSON value with known JSON object."""
tpl = template.Template('{{ value_json.hello|is_defined }}', self.hass)
self.assertEqual(
'world',
tpl.render_with_possible_json_value('{"hello": "world"}'))
def test_render_with_possible_json_value_undefined_json(self):
"""Render with possible JSON value with unknown JSON object."""
tpl = template.Template('{{ value_json.bye|is_defined }}', self.hass)
self.assertEqual(
'{"hello": "world"}',
tpl.render_with_possible_json_value('{"hello": "world"}'))
def test_render_with_possible_json_value_undefined_json_error_value(self):
"""Render with possible JSON value with unknown JSON object."""
tpl = template.Template('{{ value_json.bye|is_defined }}', self.hass)
self.assertEqual(
'',
tpl.render_with_possible_json_value('{"hello": "world"}', ''))
def test_raise_exception_on_error(self):
"""Test raising an exception on error."""
with self.assertRaises(TemplateError):
template.Template('{{ invalid_syntax').ensure_valid()
2015-12-13 06:19:12 +00:00
2015-12-13 06:19:37 +00:00
def test_if_state_exists(self):
"""Test if state exists works."""
2015-12-13 06:19:37 +00:00
self.hass.states.set('test.object', 'available')
tpl = template.Template(
'{% if states.test.object %}exists{% else %}not exists{% endif %}',
self.hass)
self.assertEqual('exists', tpl.render())
2015-12-13 06:19:37 +00:00
2015-12-13 06:19:12 +00:00
def test_is_state(self):
"""Test is_state method."""
2015-12-13 06:19:12 +00:00
self.hass.states.set('test.object', 'available')
tpl = template.Template("""
2016-02-14 21:07:21 +00:00
{% if is_state("test.object", "available") %}yes{% else %}no{% endif %}
""", self.hass)
self.assertEqual('yes', tpl.render())
2015-12-13 06:19:37 +00:00
def test_is_state_attr(self):
"""Test is_state_attr method."""
self.hass.states.set('test.object', 'available', {'mode': 'on'})
tpl = template.Template("""
2016-02-14 21:07:21 +00:00
{% if is_state_attr("test.object", "mode", "on") %}yes{% else %}no{% endif %}
""", self.hass)
self.assertEqual('yes', tpl.render())
2015-12-13 06:19:37 +00:00
def test_states_function(self):
"""Test using states as a function."""
2015-12-13 06:19:37 +00:00
self.hass.states.set('test.object', 'available')
tpl = template.Template('{{ states("test.object") }}', self.hass)
self.assertEqual('available', tpl.render())
tpl2 = template.Template('{{ states("test.object2") }}', self.hass)
self.assertEqual('unknown', tpl2.render())
2016-02-23 21:06:45 +00:00
@patch('homeassistant.helpers.template.TemplateEnvironment.'
'is_safe_callable', return_value=True)
def test_now(self, mock_is_safe):
"""Test now method."""
now = dt_util.now()
with patch.dict(template.ENV.globals, {'now': lambda: now}):
self.assertEqual(
now.isoformat(),
template.Template('{{ now().isoformat() }}',
self.hass).render())
2016-02-23 21:06:45 +00:00
@patch('homeassistant.helpers.template.TemplateEnvironment.'
'is_safe_callable', return_value=True)
def test_utcnow(self, mock_is_safe):
"""Test utcnow method."""
now = dt_util.utcnow()
with patch.dict(template.ENV.globals, {'utcnow': lambda: now}):
self.assertEqual(
now.isoformat(),
template.Template('{{ utcnow().isoformat() }}',
self.hass).render())
2016-02-21 05:58:53 +00:00
def test_distance_function_with_1_state(self):
"""Test distance function with 1 state."""
2016-02-21 05:58:53 +00:00
self.hass.states.set('test.object', 'happy', {
'latitude': 32.87336,
'longitude': -117.22943,
})
tpl = template.Template('{{ distance(states.test.object) | round }}',
self.hass)
self.assertEqual('187', tpl.render())
2016-02-21 05:58:53 +00:00
def test_distance_function_with_2_states(self):
"""Test distance function with 2 states."""
2016-02-21 05:58:53 +00:00
self.hass.states.set('test.object', 'happy', {
'latitude': 32.87336,
'longitude': -117.22943,
})
self.hass.states.set('test.object_2', 'happy', {
'latitude': self.hass.config.latitude,
'longitude': self.hass.config.longitude,
})
tpl = template.Template(
'{{ distance(states.test.object, states.test.object_2) | round }}',
self.hass)
self.assertEqual('187', tpl.render())
2016-02-21 05:58:53 +00:00
def test_distance_function_with_1_coord(self):
"""Test distance function with 1 coord."""
tpl = template.Template(
'{{ distance("32.87336", "-117.22943") | round }}', self.hass)
2016-02-21 05:58:53 +00:00
self.assertEqual(
'187',
tpl.render())
2016-02-21 05:58:53 +00:00
def test_distance_function_with_2_coords(self):
"""Test distance function with 2 coords."""
2016-02-21 05:58:53 +00:00
self.assertEqual(
'187',
template.Template(
2016-02-21 05:58:53 +00:00
'{{ distance("32.87336", "-117.22943", %s, %s) | round }}'
% (self.hass.config.latitude, self.hass.config.longitude),
self.hass).render())
2016-02-21 05:58:53 +00:00
def test_distance_function_with_1_state_1_coord(self):
"""Test distance function with 1 state 1 coord."""
2016-02-21 05:58:53 +00:00
self.hass.states.set('test.object_2', 'happy', {
'latitude': self.hass.config.latitude,
'longitude': self.hass.config.longitude,
})
tpl = template.Template(
'{{ distance("32.87336", "-117.22943", states.test.object_2) '
'| round }}', self.hass)
self.assertEqual('187', tpl.render())
2016-02-21 05:58:53 +00:00
tpl2 = template.Template(
'{{ distance(states.test.object_2, "32.87336", "-117.22943") '
'| round }}', self.hass)
self.assertEqual('187', tpl2.render())
2016-02-21 05:58:53 +00:00
def test_distance_function_return_None_if_invalid_state(self):
"""Test distance function return None if invalid state."""
2016-02-21 05:58:53 +00:00
self.hass.states.set('test.object_2', 'happy', {
'latitude': 10,
})
tpl = template.Template('{{ distance(states.test.object_2) | round }}',
self.hass)
2016-02-21 05:58:53 +00:00
self.assertEqual(
'None',
tpl.render())
2016-02-21 05:58:53 +00:00
def test_distance_function_return_None_if_invalid_coord(self):
"""Test distance function return None if invalid coord."""
2016-02-21 05:58:53 +00:00
self.assertEqual(
'None',
template.Template(
'{{ distance("123", "abc") }}', self.hass).render())
2016-02-21 19:13:40 +00:00
self.assertEqual(
'None',
template.Template('{{ distance("123") }}', self.hass).render())
2016-02-21 19:13:40 +00:00
self.hass.states.set('test.object_2', 'happy', {
'latitude': self.hass.config.latitude,
'longitude': self.hass.config.longitude,
})
tpl = template.Template('{{ distance("123", states.test_object_2) }}',
self.hass)
2016-02-21 19:13:40 +00:00
self.assertEqual(
'None',
tpl.render())
2016-02-21 19:13:40 +00:00
def test_closest_function_home_vs_domain(self):
"""Test closest function home vs domain."""
2016-02-21 19:13:40 +00:00
self.hass.states.set('test_domain.object', 'happy', {
'latitude': self.hass.config.latitude + 0.1,
'longitude': self.hass.config.longitude + 0.1,
})
self.hass.states.set('not_test_domain.but_closer', 'happy', {
'latitude': self.hass.config.latitude,
'longitude': self.hass.config.longitude,
})
self.assertEqual(
'test_domain.object',
template.Template('{{ closest(states.test_domain).entity_id }}',
self.hass).render())
2016-02-21 19:13:40 +00:00
def test_closest_function_home_vs_all_states(self):
"""Test closest function home vs all states."""
2016-02-21 19:13:40 +00:00
self.hass.states.set('test_domain.object', 'happy', {
'latitude': self.hass.config.latitude + 0.1,
'longitude': self.hass.config.longitude + 0.1,
})
self.hass.states.set('test_domain_2.and_closer', 'happy', {
'latitude': self.hass.config.latitude,
'longitude': self.hass.config.longitude,
})
self.assertEqual(
'test_domain_2.and_closer',
template.Template('{{ closest(states).entity_id }}',
self.hass).render())
2016-02-21 19:13:40 +00:00
def test_closest_function_home_vs_group_entity_id(self):
"""Test closest function home vs group entity id."""
2016-02-21 19:13:40 +00:00
self.hass.states.set('test_domain.object', 'happy', {
'latitude': self.hass.config.latitude + 0.1,
'longitude': self.hass.config.longitude + 0.1,
})
self.hass.states.set('not_in_group.but_closer', 'happy', {
'latitude': self.hass.config.latitude,
'longitude': self.hass.config.longitude,
})
group.Group.create_group(
self.hass, 'location group', ['test_domain.object'])
2016-02-21 19:13:40 +00:00
self.assertEqual(
'test_domain.object',
template.Template(
'{{ closest("group.location_group").entity_id }}',
self.hass).render())
2016-02-21 19:13:40 +00:00
def test_closest_function_home_vs_group_state(self):
"""Test closest function home vs group state."""
2016-02-21 19:13:40 +00:00
self.hass.states.set('test_domain.object', 'happy', {
'latitude': self.hass.config.latitude + 0.1,
'longitude': self.hass.config.longitude + 0.1,
})
self.hass.states.set('not_in_group.but_closer', 'happy', {
'latitude': self.hass.config.latitude,
'longitude': self.hass.config.longitude,
})
group.Group.create_group(
self.hass, 'location group', ['test_domain.object'])
2016-02-21 19:13:40 +00:00
self.assertEqual(
'test_domain.object',
template.Template(
'{{ closest(states.group.location_group).entity_id }}',
self.hass).render())
2016-02-21 19:13:40 +00:00
def test_closest_function_to_coord(self):
"""Test closest function to coord."""
2016-02-21 19:13:40 +00:00
self.hass.states.set('test_domain.closest_home', 'happy', {
'latitude': self.hass.config.latitude + 0.1,
'longitude': self.hass.config.longitude + 0.1,
})
self.hass.states.set('test_domain.closest_zone', 'happy', {
'latitude': self.hass.config.latitude + 0.2,
'longitude': self.hass.config.longitude + 0.2,
})
self.hass.states.set('zone.far_away', 'zoning', {
'latitude': self.hass.config.latitude + 0.3,
'longitude': self.hass.config.longitude + 0.3,
})
tpl = template.Template(
'{{ closest("%s", %s, states.test_domain).entity_id }}'
% (self.hass.config.latitude + 0.3,
self.hass.config.longitude + 0.3), self.hass)
2016-02-21 19:13:40 +00:00
self.assertEqual(
'test_domain.closest_zone',
tpl.render())
2016-02-21 19:13:40 +00:00
def test_closest_function_to_entity_id(self):
"""Test closest function to entity id."""
2016-02-21 19:13:40 +00:00
self.hass.states.set('test_domain.closest_home', 'happy', {
'latitude': self.hass.config.latitude + 0.1,
'longitude': self.hass.config.longitude + 0.1,
})
self.hass.states.set('test_domain.closest_zone', 'happy', {
'latitude': self.hass.config.latitude + 0.2,
'longitude': self.hass.config.longitude + 0.2,
})
self.hass.states.set('zone.far_away', 'zoning', {
'latitude': self.hass.config.latitude + 0.3,
'longitude': self.hass.config.longitude + 0.3,
})
self.assertEqual(
'test_domain.closest_zone',
template.Template(
'{{ closest("zone.far_away", '
'states.test_domain).entity_id }}', self.hass).render())
2016-02-21 19:13:40 +00:00
def test_closest_function_to_state(self):
"""Test closest function to state."""
2016-02-21 19:13:40 +00:00
self.hass.states.set('test_domain.closest_home', 'happy', {
'latitude': self.hass.config.latitude + 0.1,
'longitude': self.hass.config.longitude + 0.1,
})
self.hass.states.set('test_domain.closest_zone', 'happy', {
'latitude': self.hass.config.latitude + 0.2,
'longitude': self.hass.config.longitude + 0.2,
})
self.hass.states.set('zone.far_away', 'zoning', {
'latitude': self.hass.config.latitude + 0.3,
'longitude': self.hass.config.longitude + 0.3,
})
self.assertEqual(
'test_domain.closest_zone',
template.Template(
2016-02-21 19:13:40 +00:00
'{{ closest(states.zone.far_away, '
'states.test_domain).entity_id }}', self.hass).render())
2016-02-21 19:13:40 +00:00
def test_closest_function_invalid_state(self):
"""Test closest function invalid state."""
2016-02-21 19:13:40 +00:00
self.hass.states.set('test_domain.closest_home', 'happy', {
'latitude': self.hass.config.latitude + 0.1,
'longitude': self.hass.config.longitude + 0.1,
})
for state in ('states.zone.non_existing', '"zone.non_existing"'):
self.assertEqual(
'None',
template.Template('{{ closest(%s, states) }}' % state,
self.hass).render())
2016-02-21 19:13:40 +00:00
def test_closest_function_state_with_invalid_location(self):
"""Test closest function state with invalid location."""
2016-02-21 19:13:40 +00:00
self.hass.states.set('test_domain.closest_home', 'happy', {
'latitude': 'invalid latitude',
'longitude': self.hass.config.longitude + 0.1,
})
self.assertEqual(
'None',
template.Template(
2016-02-21 19:13:40 +00:00
'{{ closest(states.test_domain.closest_home, '
'states) }}', self.hass).render())
2016-02-21 19:13:40 +00:00
def test_closest_function_invalid_coordinates(self):
"""Test closest function invalid coordinates."""
2016-02-21 19:13:40 +00:00
self.hass.states.set('test_domain.closest_home', 'happy', {
'latitude': self.hass.config.latitude + 0.1,
'longitude': self.hass.config.longitude + 0.1,
})
self.assertEqual(
'None',
template.Template('{{ closest("invalid", "coord", states) }}',
self.hass).render())
2016-02-21 19:13:40 +00:00
def test_closest_function_no_location_states(self):
"""Test closest function without location states."""
self.assertEqual(
'None',
template.Template('{{ closest(states) }}', self.hass).render())
def test_extract_entities_none_exclude_stuff(self):
"""Test extract entities function with none or exclude stuff."""
self.assertEqual(MATCH_ALL, template.extract_entities(None))
self.assertEqual(
MATCH_ALL,
template.extract_entities(
'{{ closest(states.zone.far_away, '
'states.test_domain).entity_id }}'))
self.assertEqual(
MATCH_ALL,
template.extract_entities(
'{{ distance("123", states.test_object_2) }}'))
def test_extract_entities_no_match_entities(self):
"""Test extract entities function with none entities stuff."""
self.assertEqual(
MATCH_ALL,
template.extract_entities(
"{{ value_json.tst | timestamp_custom('%Y' True) }}"))
self.assertEqual(
MATCH_ALL,
template.extract_entities("""
{% for state in states.sensor %}
{{ state.entity_id }}={{ state.state }},
{% endfor %}
"""))
def test_extract_entities_match_entities(self):
"""Test extract entities function with entities stuff."""
self.assertListEqual(
['device_tracker.phone_1'],
template.extract_entities("""
{% if is_state('device_tracker.phone_1', 'home') %}
Ha, Hercules is home!
{% else %}
Hercules is at {{ states('device_tracker.phone_1') }}.
{% endif %}
"""))
self.assertListEqual(
['binary_sensor.garage_door'],
template.extract_entities("""
{{ as_timestamp(states.binary_sensor.garage_door.last_changed) }}
"""))
self.assertListEqual(
['binary_sensor.garage_door'],
template.extract_entities("""
{{ states("binary_sensor.garage_door") }}
"""))
self.assertListEqual(
['device_tracker.phone_2'],
template.extract_entities("""
is_state_attr('device_tracker.phone_2', 'battery', 40)
"""))
self.assertListEqual(
sorted([
'device_tracker.phone_1',
'device_tracker.phone_2',
]),
sorted(template.extract_entities("""
{% if is_state('device_tracker.phone_1', 'home') %}
Ha, Hercules is home!
{% elif states.device_tracker.phone_2.attributes.battery < 40 %}
Hercules you power goes done!.
{% endif %}
""")))
self.assertListEqual(
sorted([
'sensor.pick_humidity',
'sensor.pick_temperature',
]),
sorted(template.extract_entities("""
{{
states.sensor.pick_temperature.state ~ °C ( ~
states.sensor.pick_humidity.state ~ %
}}
""")))