2016-01-09 00:58:44 +00:00
|
|
|
"""Service calling related helpers."""
|
2016-01-25 03:46:30 +00:00
|
|
|
import functools
|
2016-01-09 00:58:44 +00:00
|
|
|
import logging
|
|
|
|
|
|
|
|
from homeassistant.const import ATTR_ENTITY_ID
|
2016-01-24 06:49:49 +00:00
|
|
|
from homeassistant.helpers.entity import split_entity_id
|
2016-01-24 06:57:14 +00:00
|
|
|
from homeassistant.loader import get_component
|
2016-01-09 00:58:44 +00:00
|
|
|
|
2016-01-25 03:46:30 +00:00
|
|
|
HASS = None
|
|
|
|
|
2016-01-09 00:58:44 +00:00
|
|
|
CONF_SERVICE = 'service'
|
|
|
|
CONF_SERVICE_ENTITY_ID = 'entity_id'
|
|
|
|
CONF_SERVICE_DATA = 'data'
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2016-01-25 03:46:30 +00:00
|
|
|
def service(domain, service_name):
|
|
|
|
""" Decorator factory to register a service """
|
|
|
|
|
|
|
|
def register_service_decorator(action):
|
|
|
|
""" Decorator to register a service """
|
|
|
|
HASS.services.register(domain, service_name,
|
2016-01-25 05:14:16 +00:00
|
|
|
functools.partial(action, HASS))
|
2016-01-25 03:46:30 +00:00
|
|
|
return action
|
|
|
|
|
|
|
|
return register_service_decorator
|
|
|
|
|
|
|
|
|
2016-01-09 00:58:44 +00:00
|
|
|
def call_from_config(hass, config, blocking=False):
|
|
|
|
"""Call a service based on a config hash."""
|
2016-01-10 00:01:27 +00:00
|
|
|
if not isinstance(config, dict) or CONF_SERVICE not in config:
|
2016-01-09 00:58:44 +00:00
|
|
|
_LOGGER.error('Missing key %s: %s', CONF_SERVICE, config)
|
|
|
|
return
|
|
|
|
|
2016-01-10 00:01:27 +00:00
|
|
|
try:
|
2016-01-25 03:46:30 +00:00
|
|
|
domain, service_name = split_entity_id(config[CONF_SERVICE])
|
2016-01-10 00:01:27 +00:00
|
|
|
except ValueError:
|
|
|
|
_LOGGER.error('Invalid service specified: %s', config[CONF_SERVICE])
|
|
|
|
return
|
|
|
|
|
2016-01-09 00:58:44 +00:00
|
|
|
service_data = config.get(CONF_SERVICE_DATA)
|
|
|
|
|
|
|
|
if service_data is None:
|
|
|
|
service_data = {}
|
|
|
|
elif isinstance(service_data, dict):
|
|
|
|
service_data = dict(service_data)
|
|
|
|
else:
|
|
|
|
_LOGGER.error("%s should be a dictionary", CONF_SERVICE_DATA)
|
|
|
|
service_data = {}
|
|
|
|
|
|
|
|
entity_id = config.get(CONF_SERVICE_ENTITY_ID)
|
|
|
|
if isinstance(entity_id, str):
|
2016-01-09 23:51:51 +00:00
|
|
|
service_data[ATTR_ENTITY_ID] = [ent.strip() for ent in
|
|
|
|
entity_id.split(",")]
|
2016-01-09 00:58:44 +00:00
|
|
|
elif entity_id is not None:
|
|
|
|
service_data[ATTR_ENTITY_ID] = entity_id
|
|
|
|
|
2016-01-25 03:46:30 +00:00
|
|
|
hass.services.call(domain, service_name, service_data, blocking)
|
2016-01-24 06:57:14 +00:00
|
|
|
|
|
|
|
|
2016-01-25 04:00:43 +00:00
|
|
|
def extract_entity_ids(hass, service_call):
|
2016-01-24 06:57:14 +00:00
|
|
|
"""
|
|
|
|
Helper method to extract a list of entity ids from a service call.
|
|
|
|
Will convert group entity ids to the entity ids it represents.
|
|
|
|
"""
|
2016-01-25 04:00:43 +00:00
|
|
|
if not (service_call.data and ATTR_ENTITY_ID in service_call.data):
|
2016-01-24 06:57:14 +00:00
|
|
|
return []
|
|
|
|
|
|
|
|
group = get_component('group')
|
|
|
|
|
|
|
|
# Entity ID attr can be a list or a string
|
2016-01-25 04:00:43 +00:00
|
|
|
service_ent_id = service_call.data[ATTR_ENTITY_ID]
|
2016-01-24 06:57:14 +00:00
|
|
|
|
|
|
|
if isinstance(service_ent_id, str):
|
|
|
|
return group.expand_entity_ids(hass, [service_ent_id])
|
|
|
|
|
|
|
|
return [ent_id for ent_id in group.expand_entity_ids(hass, service_ent_id)]
|