2016-03-28 01:48:51 +00:00
|
|
|
"""Helpers for config validation using voluptuous."""
|
2016-04-04 19:18:58 +00:00
|
|
|
from datetime import timedelta
|
|
|
|
|
2016-04-03 17:19:09 +00:00
|
|
|
import jinja2
|
2016-03-28 01:48:51 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
2016-04-04 19:18:58 +00:00
|
|
|
from homeassistant.loader import get_platform
|
2016-03-28 01:48:51 +00:00
|
|
|
from homeassistant.const import (
|
2016-04-21 22:52:20 +00:00
|
|
|
CONF_PLATFORM, CONF_SCAN_INTERVAL, TEMP_CELSIUS, TEMP_FAHRENHEIT,
|
2016-04-28 10:03:57 +00:00
|
|
|
CONF_ALIAS, CONF_ENTITY_ID, CONF_VALUE_TEMPLATE, WEEKDAYS,
|
2016-05-03 05:05:09 +00:00
|
|
|
CONF_CONDITION, CONF_BELOW, CONF_ABOVE, SUN_EVENT_SUNSET,
|
|
|
|
SUN_EVENT_SUNRISE)
|
2016-03-28 01:48:51 +00:00
|
|
|
from homeassistant.helpers.entity import valid_entity_id
|
|
|
|
import homeassistant.util.dt as dt_util
|
2016-04-03 17:19:09 +00:00
|
|
|
from homeassistant.util import slugify
|
2016-03-28 01:48:51 +00:00
|
|
|
|
|
|
|
# pylint: disable=invalid-name
|
|
|
|
|
2016-04-28 10:03:57 +00:00
|
|
|
TIME_PERIOD_ERROR = "offset {} should be format 'HH:MM' or 'HH:MM:SS'"
|
|
|
|
|
2016-04-03 17:19:09 +00:00
|
|
|
# Home Assistant types
|
2016-04-01 03:19:59 +00:00
|
|
|
byte = vol.All(vol.Coerce(int), vol.Range(min=0, max=255))
|
|
|
|
small_float = vol.All(vol.Coerce(float), vol.Range(min=0, max=1))
|
2016-04-07 19:19:28 +00:00
|
|
|
positive_int = vol.All(vol.Coerce(int), vol.Range(min=0))
|
2016-04-03 17:19:09 +00:00
|
|
|
latitude = vol.All(vol.Coerce(float), vol.Range(min=-90, max=90),
|
|
|
|
msg='invalid latitude')
|
|
|
|
longitude = vol.All(vol.Coerce(float), vol.Range(min=-180, max=180),
|
|
|
|
msg='invalid longitude')
|
2016-05-03 05:05:09 +00:00
|
|
|
sun_event = vol.All(vol.Lower, vol.Any(SUN_EVENT_SUNSET, SUN_EVENT_SUNRISE))
|
2016-04-03 17:19:09 +00:00
|
|
|
|
|
|
|
|
2016-04-21 22:52:20 +00:00
|
|
|
# Adapted from:
|
|
|
|
# https://github.com/alecthomas/voluptuous/issues/115#issuecomment-144464666
|
|
|
|
def has_at_least_one_key(*keys):
|
|
|
|
"""Validator that at least one key exists."""
|
|
|
|
def validate(obj):
|
|
|
|
"""Test keys exist in dict."""
|
|
|
|
if not isinstance(obj, dict):
|
|
|
|
raise vol.Invalid('expected dictionary')
|
|
|
|
|
|
|
|
for k in obj.keys():
|
|
|
|
if k in keys:
|
|
|
|
return obj
|
|
|
|
raise vol.Invalid('must contain one of {}.'.format(', '.join(keys)))
|
|
|
|
|
|
|
|
return validate
|
|
|
|
|
|
|
|
|
2016-04-03 17:19:09 +00:00
|
|
|
def boolean(value):
|
|
|
|
"""Validate and coerce a boolean value."""
|
|
|
|
if isinstance(value, str):
|
2016-04-04 04:38:58 +00:00
|
|
|
value = value.lower()
|
2016-04-03 17:19:09 +00:00
|
|
|
if value in ('1', 'true', 'yes', 'on', 'enable'):
|
|
|
|
return True
|
|
|
|
if value in ('0', 'false', 'no', 'off', 'disable'):
|
|
|
|
return False
|
|
|
|
raise vol.Invalid('invalid boolean value {}'.format(value))
|
|
|
|
return bool(value)
|
2016-03-28 01:48:51 +00:00
|
|
|
|
|
|
|
|
2016-04-07 17:52:25 +00:00
|
|
|
def isfile(value):
|
|
|
|
"""Validate that the value is an existing file."""
|
|
|
|
return vol.IsFile('not a file')(value)
|
|
|
|
|
|
|
|
|
2016-04-04 19:18:58 +00:00
|
|
|
def ensure_list(value):
|
|
|
|
"""Wrap value in list if it is not one."""
|
|
|
|
return value if isinstance(value, list) else [value]
|
|
|
|
|
|
|
|
|
2016-03-28 01:48:51 +00:00
|
|
|
def entity_id(value):
|
|
|
|
"""Validate Entity ID."""
|
2016-04-10 22:20:20 +00:00
|
|
|
value = string(value).lower()
|
2016-03-28 01:48:51 +00:00
|
|
|
if valid_entity_id(value):
|
|
|
|
return value
|
2016-06-09 03:55:08 +00:00
|
|
|
raise vol.Invalid('Entity ID {} is an invalid entity id'.format(value))
|
2016-03-28 01:48:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
def entity_ids(value):
|
|
|
|
"""Validate Entity IDs."""
|
2016-06-09 03:55:08 +00:00
|
|
|
if value is None:
|
|
|
|
raise vol.Invalid('Entity IDs can not be None')
|
2016-03-28 01:48:51 +00:00
|
|
|
if isinstance(value, str):
|
|
|
|
value = [ent_id.strip() for ent_id in value.split(',')]
|
|
|
|
|
2016-04-10 22:20:20 +00:00
|
|
|
return [entity_id(ent_id) for ent_id in value]
|
2016-03-28 01:48:51 +00:00
|
|
|
|
|
|
|
|
|
|
|
def icon(value):
|
|
|
|
"""Validate icon."""
|
|
|
|
value = str(value)
|
|
|
|
|
|
|
|
if value.startswith('mdi:'):
|
|
|
|
return value
|
|
|
|
|
|
|
|
raise vol.Invalid('Icons should start with prefix "mdi:"')
|
|
|
|
|
|
|
|
|
2016-04-21 22:52:20 +00:00
|
|
|
time_period_dict = vol.All(
|
|
|
|
dict, vol.Schema({
|
|
|
|
'days': vol.Coerce(int),
|
|
|
|
'hours': vol.Coerce(int),
|
|
|
|
'minutes': vol.Coerce(int),
|
|
|
|
'seconds': vol.Coerce(int),
|
|
|
|
'milliseconds': vol.Coerce(int),
|
|
|
|
}),
|
|
|
|
has_at_least_one_key('days', 'hours', 'minutes',
|
|
|
|
'seconds', 'milliseconds'),
|
|
|
|
lambda value: timedelta(**value))
|
|
|
|
|
|
|
|
|
|
|
|
def time_period_str(value):
|
2016-04-04 19:18:58 +00:00
|
|
|
"""Validate and transform time offset."""
|
2016-04-28 10:03:57 +00:00
|
|
|
if isinstance(value, int):
|
|
|
|
raise vol.Invalid('Make sure you wrap time values in quotes')
|
|
|
|
elif not isinstance(value, str):
|
|
|
|
raise vol.Invalid(TIME_PERIOD_ERROR.format(value))
|
2016-04-04 19:18:58 +00:00
|
|
|
|
|
|
|
negative_offset = False
|
|
|
|
if value.startswith('-'):
|
|
|
|
negative_offset = True
|
|
|
|
value = value[1:]
|
|
|
|
elif value.startswith('+'):
|
|
|
|
value = value[1:]
|
|
|
|
|
|
|
|
try:
|
|
|
|
parsed = [int(x) for x in value.split(':')]
|
|
|
|
except ValueError:
|
2016-04-28 10:03:57 +00:00
|
|
|
raise vol.Invalid(TIME_PERIOD_ERROR.format(value))
|
2016-04-04 19:18:58 +00:00
|
|
|
|
|
|
|
if len(parsed) == 2:
|
|
|
|
hour, minute = parsed
|
|
|
|
second = 0
|
|
|
|
elif len(parsed) == 3:
|
|
|
|
hour, minute, second = parsed
|
|
|
|
else:
|
2016-04-28 10:03:57 +00:00
|
|
|
raise vol.Invalid(TIME_PERIOD_ERROR.format(value))
|
2016-04-04 19:18:58 +00:00
|
|
|
|
|
|
|
offset = timedelta(hours=hour, minutes=minute, seconds=second)
|
|
|
|
|
|
|
|
if negative_offset:
|
|
|
|
offset *= -1
|
|
|
|
|
|
|
|
return offset
|
|
|
|
|
|
|
|
|
2016-04-21 22:52:20 +00:00
|
|
|
time_period = vol.Any(time_period_str, timedelta, time_period_dict)
|
|
|
|
|
|
|
|
|
2016-05-08 05:24:04 +00:00
|
|
|
def log_exception(logger, ex, domain, config):
|
2016-05-07 14:35:42 +00:00
|
|
|
"""Generate log exception for config validation."""
|
|
|
|
message = 'Invalid config for [{}]: '.format(domain)
|
|
|
|
if 'extra keys not allowed' in ex.error_message:
|
|
|
|
message += '[{}] is an invalid option for [{}]. Check: {}->{}.'\
|
|
|
|
.format(ex.path[-1], domain, domain,
|
|
|
|
'->'.join('%s' % m for m in ex.path))
|
|
|
|
else:
|
2016-05-08 05:24:04 +00:00
|
|
|
message += str(ex)
|
|
|
|
|
|
|
|
if hasattr(config, '__line__'):
|
|
|
|
message += " (See {}:{})".format(config.__config_file__,
|
|
|
|
config.__line__ or '?')
|
|
|
|
|
2016-05-07 14:35:42 +00:00
|
|
|
logger.error(message)
|
|
|
|
|
|
|
|
|
2016-04-04 19:18:58 +00:00
|
|
|
def match_all(value):
|
|
|
|
"""Validator that matches all values."""
|
|
|
|
return value
|
|
|
|
|
|
|
|
|
|
|
|
def platform_validator(domain):
|
|
|
|
"""Validate if platform exists for given domain."""
|
|
|
|
def validator(value):
|
|
|
|
"""Test if platform exists."""
|
|
|
|
if value is None:
|
|
|
|
raise vol.Invalid('platform cannot be None')
|
|
|
|
if get_platform(domain, str(value)):
|
|
|
|
return value
|
|
|
|
raise vol.Invalid(
|
|
|
|
'platform {} does not exist for {}'.format(value, domain))
|
|
|
|
return validator
|
|
|
|
|
|
|
|
|
2016-04-21 22:52:20 +00:00
|
|
|
def positive_timedelta(value):
|
|
|
|
"""Validate timedelta is positive."""
|
|
|
|
if value < timedelta(0):
|
|
|
|
raise vol.Invalid('Time period should be positive')
|
|
|
|
return value
|
|
|
|
|
|
|
|
|
2016-04-03 17:19:09 +00:00
|
|
|
def service(value):
|
|
|
|
"""Validate service."""
|
|
|
|
# Services use same format as entities so we can use same helper.
|
|
|
|
if valid_entity_id(value):
|
|
|
|
return value
|
|
|
|
raise vol.Invalid('Service {} does not match format <domain>.<name>'
|
|
|
|
.format(value))
|
|
|
|
|
|
|
|
|
|
|
|
def slug(value):
|
|
|
|
"""Validate value is a valid slug."""
|
|
|
|
if value is None:
|
|
|
|
raise vol.Invalid('Slug should not be None')
|
|
|
|
value = str(value)
|
|
|
|
slg = slugify(value)
|
|
|
|
if value == slg:
|
|
|
|
return value
|
|
|
|
raise vol.Invalid('invalid slug {} (try {})'.format(value, slg))
|
|
|
|
|
|
|
|
|
2016-04-02 07:51:03 +00:00
|
|
|
def string(value):
|
|
|
|
"""Coerce value to string, except for None."""
|
|
|
|
if value is not None:
|
|
|
|
return str(value)
|
2016-04-03 17:19:09 +00:00
|
|
|
raise vol.Invalid('string value is None')
|
2016-04-02 07:51:03 +00:00
|
|
|
|
|
|
|
|
2016-03-28 01:48:51 +00:00
|
|
|
def temperature_unit(value):
|
|
|
|
"""Validate and transform temperature unit."""
|
2016-04-03 17:19:09 +00:00
|
|
|
value = str(value).upper()
|
|
|
|
if value == 'C':
|
2016-04-20 03:30:44 +00:00
|
|
|
return TEMP_CELSIUS
|
2016-04-03 17:19:09 +00:00
|
|
|
elif value == 'F':
|
|
|
|
return TEMP_FAHRENHEIT
|
|
|
|
raise vol.Invalid('invalid temperature unit (expected C or F)')
|
|
|
|
|
|
|
|
|
|
|
|
def template(value):
|
|
|
|
"""Validate a jinja2 template."""
|
|
|
|
if value is None:
|
|
|
|
raise vol.Invalid('template value is None')
|
|
|
|
|
|
|
|
value = str(value)
|
|
|
|
try:
|
|
|
|
jinja2.Environment().parse(value)
|
|
|
|
return value
|
|
|
|
except jinja2.exceptions.TemplateSyntaxError as ex:
|
|
|
|
raise vol.Invalid('invalid template ({})'.format(ex))
|
2016-03-28 01:48:51 +00:00
|
|
|
|
|
|
|
|
2016-04-28 10:03:57 +00:00
|
|
|
def time(value):
|
|
|
|
"""Validate time."""
|
|
|
|
time_val = dt_util.parse_time(value)
|
|
|
|
|
|
|
|
if time_val is None:
|
|
|
|
raise vol.Invalid('Invalid time specified: {}'.format(value))
|
|
|
|
|
|
|
|
return time_val
|
|
|
|
|
|
|
|
|
2016-03-28 01:48:51 +00:00
|
|
|
def time_zone(value):
|
|
|
|
"""Validate timezone."""
|
|
|
|
if dt_util.get_time_zone(value) is not None:
|
|
|
|
return value
|
|
|
|
raise vol.Invalid(
|
|
|
|
'Invalid time zone passed in. Valid options can be found here: '
|
|
|
|
'http://en.wikipedia.org/wiki/List_of_tz_database_time_zones')
|
2016-04-03 17:19:09 +00:00
|
|
|
|
2016-04-28 10:03:57 +00:00
|
|
|
weekdays = vol.All(ensure_list, [vol.In(WEEKDAYS)])
|
|
|
|
|
2016-04-03 17:19:09 +00:00
|
|
|
|
|
|
|
# Validator helpers
|
|
|
|
|
2016-04-04 19:18:58 +00:00
|
|
|
def key_dependency(key, dependency):
|
|
|
|
"""Validate that all dependencies exist for key."""
|
|
|
|
def validator(value):
|
|
|
|
"""Test dependencies."""
|
|
|
|
if not isinstance(value, dict):
|
|
|
|
raise vol.Invalid('key dependencies require a dict')
|
|
|
|
if key in value and dependency not in value:
|
|
|
|
raise vol.Invalid('dependency violation - key "{}" requires '
|
|
|
|
'key "{}" to exist'.format(key, dependency))
|
2016-04-03 17:19:09 +00:00
|
|
|
|
2016-04-04 19:18:58 +00:00
|
|
|
return value
|
|
|
|
return validator
|
2016-04-03 17:19:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
# Schemas
|
|
|
|
|
|
|
|
PLATFORM_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(CONF_PLATFORM): string,
|
|
|
|
CONF_SCAN_INTERVAL: vol.All(vol.Coerce(int), vol.Range(min=1)),
|
|
|
|
}, extra=vol.ALLOW_EXTRA)
|
|
|
|
|
|
|
|
EVENT_SCHEMA = vol.Schema({
|
2016-04-21 22:52:20 +00:00
|
|
|
vol.Optional(CONF_ALIAS): string,
|
2016-04-03 17:19:09 +00:00
|
|
|
vol.Required('event'): string,
|
2016-04-21 22:52:20 +00:00
|
|
|
vol.Optional('event_data'): dict,
|
2016-04-03 17:19:09 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
SERVICE_SCHEMA = vol.All(vol.Schema({
|
2016-04-21 22:52:20 +00:00
|
|
|
vol.Optional(CONF_ALIAS): string,
|
2016-04-03 17:19:09 +00:00
|
|
|
vol.Exclusive('service', 'service name'): service,
|
2016-04-21 19:22:19 +00:00
|
|
|
vol.Exclusive('service_template', 'service name'): template,
|
|
|
|
vol.Optional('data'): dict,
|
|
|
|
vol.Optional('data_template'): {match_all: template},
|
2016-04-28 10:03:57 +00:00
|
|
|
vol.Optional(CONF_ENTITY_ID): entity_ids,
|
2016-04-04 19:18:58 +00:00
|
|
|
}), has_at_least_one_key('service', 'service_template'))
|
2016-04-21 22:52:20 +00:00
|
|
|
|
2016-04-28 10:03:57 +00:00
|
|
|
NUMERIC_STATE_CONDITION_SCHEMA = vol.All(vol.Schema({
|
|
|
|
vol.Required(CONF_CONDITION): 'numeric_state',
|
|
|
|
vol.Required(CONF_ENTITY_ID): entity_id,
|
|
|
|
CONF_BELOW: vol.Coerce(float),
|
|
|
|
CONF_ABOVE: vol.Coerce(float),
|
|
|
|
vol.Optional(CONF_VALUE_TEMPLATE): template,
|
|
|
|
}), has_at_least_one_key(CONF_BELOW, CONF_ABOVE))
|
|
|
|
|
|
|
|
STATE_CONDITION_SCHEMA = vol.All(vol.Schema({
|
|
|
|
vol.Required(CONF_CONDITION): 'state',
|
|
|
|
vol.Required(CONF_ENTITY_ID): entity_id,
|
|
|
|
vol.Required('state'): str,
|
|
|
|
vol.Optional('for'): vol.All(time_period, positive_timedelta),
|
|
|
|
# To support use_trigger_value in automation
|
|
|
|
# Deprecated 2016/04/25
|
|
|
|
vol.Optional('from'): str,
|
|
|
|
}), key_dependency('for', 'state'))
|
|
|
|
|
|
|
|
SUN_CONDITION_SCHEMA = vol.All(vol.Schema({
|
|
|
|
vol.Required(CONF_CONDITION): 'sun',
|
2016-05-03 05:05:09 +00:00
|
|
|
vol.Optional('before'): sun_event,
|
2016-04-28 10:03:57 +00:00
|
|
|
vol.Optional('before_offset'): time_period,
|
|
|
|
vol.Optional('after'): vol.All(vol.Lower, vol.Any('sunset', 'sunrise')),
|
|
|
|
vol.Optional('after_offset'): time_period,
|
|
|
|
}), has_at_least_one_key('before', 'after'))
|
|
|
|
|
|
|
|
TEMPLATE_CONDITION_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(CONF_CONDITION): 'template',
|
|
|
|
vol.Required(CONF_VALUE_TEMPLATE): template,
|
|
|
|
})
|
|
|
|
|
|
|
|
TIME_CONDITION_SCHEMA = vol.All(vol.Schema({
|
|
|
|
vol.Required(CONF_CONDITION): 'time',
|
|
|
|
'before': time,
|
|
|
|
'after': time,
|
|
|
|
'weekday': weekdays,
|
|
|
|
}), has_at_least_one_key('before', 'after', 'weekday'))
|
|
|
|
|
|
|
|
ZONE_CONDITION_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(CONF_CONDITION): 'zone',
|
|
|
|
vol.Required(CONF_ENTITY_ID): entity_id,
|
|
|
|
'zone': entity_id,
|
|
|
|
# To support use_trigger_value in automation
|
|
|
|
# Deprecated 2016/04/25
|
|
|
|
vol.Optional('event'): vol.Any('enter', 'leave'),
|
|
|
|
})
|
|
|
|
|
|
|
|
AND_CONDITION_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(CONF_CONDITION): 'and',
|
|
|
|
vol.Required('conditions'): vol.All(
|
|
|
|
ensure_list,
|
|
|
|
# pylint: disable=unnecessary-lambda
|
|
|
|
[lambda value: CONDITION_SCHEMA(value)],
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
OR_CONDITION_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(CONF_CONDITION): 'or',
|
|
|
|
vol.Required('conditions'): vol.All(
|
|
|
|
ensure_list,
|
|
|
|
# pylint: disable=unnecessary-lambda
|
|
|
|
[lambda value: CONDITION_SCHEMA(value)],
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
CONDITION_SCHEMA = vol.Any(
|
|
|
|
NUMERIC_STATE_CONDITION_SCHEMA,
|
|
|
|
STATE_CONDITION_SCHEMA,
|
|
|
|
SUN_CONDITION_SCHEMA,
|
|
|
|
TEMPLATE_CONDITION_SCHEMA,
|
|
|
|
TIME_CONDITION_SCHEMA,
|
|
|
|
ZONE_CONDITION_SCHEMA,
|
2016-05-11 04:49:58 +00:00
|
|
|
AND_CONDITION_SCHEMA,
|
|
|
|
OR_CONDITION_SCHEMA,
|
2016-04-28 10:03:57 +00:00
|
|
|
)
|
|
|
|
|
2016-04-23 05:11:21 +00:00
|
|
|
_SCRIPT_DELAY_SCHEMA = vol.Schema({
|
2016-04-21 22:52:20 +00:00
|
|
|
vol.Optional(CONF_ALIAS): string,
|
2016-07-20 18:26:17 +00:00
|
|
|
vol.Required("delay"): vol.Any(
|
|
|
|
vol.All(time_period, positive_timedelta),
|
|
|
|
template)
|
2016-04-21 22:52:20 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
SCRIPT_SCHEMA = vol.All(
|
|
|
|
ensure_list,
|
2016-04-28 10:03:57 +00:00
|
|
|
[vol.Any(SERVICE_SCHEMA, _SCRIPT_DELAY_SCHEMA, EVENT_SCHEMA,
|
|
|
|
CONDITION_SCHEMA)],
|
2016-04-21 22:52:20 +00:00
|
|
|
)
|