core/homeassistant/components/__init__.py

138 lines
4.6 KiB
Python
Raw Normal View History

"""
This package contains components that can be plugged into Home Assistant.
2014-01-05 02:24:30 +00:00
Component design guidelines:
2016-03-08 16:55:57 +00:00
- Each component defines a constant DOMAIN that is equal to its filename.
- Each component that tracks states should create state entity names in the
format "<DOMAIN>.<OBJECT_ID>".
- Each component should publish services only under its own domain.
"""
2014-04-13 19:59:45 +00:00
import itertools as it
import logging
2015-08-17 03:44:46 +00:00
import homeassistant.core as ha
from homeassistant.helpers.service import extract_entity_ids
from homeassistant.loader import get_component
from homeassistant.const import (
ATTR_ENTITY_ID, SERVICE_TURN_ON, SERVICE_TURN_OFF, SERVICE_TOGGLE)
2014-11-08 21:57:08 +00:00
_LOGGER = logging.getLogger(__name__)
SERVICE_RELOAD_CORE_CONFIG = 'reload_core_config'
def is_on(hass, entity_id=None):
2016-03-08 16:55:57 +00:00
"""Load up the module to call the is_on method.
If there is no entity id given we will check all.
"""
2014-04-13 19:59:45 +00:00
if entity_id:
group = get_component('group')
2014-04-13 19:59:45 +00:00
entity_ids = group.expand_entity_ids(hass, [entity_id])
2014-04-13 19:59:45 +00:00
else:
2014-11-29 07:19:59 +00:00
entity_ids = hass.states.entity_ids()
for entity_id in entity_ids:
2016-08-09 03:21:40 +00:00
domain = ha.split_entity_id(entity_id)[0]
module = get_component(domain)
try:
if module.is_on(hass, entity_id):
return True
except AttributeError:
# module is None or method is_on does not exist
2014-11-08 21:57:08 +00:00
_LOGGER.exception("Failed to call %s.is_on for %s",
module, entity_id)
return False
def turn_on(hass, entity_id=None, **service_data):
2016-03-08 16:55:57 +00:00
"""Turn specified entity on if possible."""
if entity_id is not None:
service_data[ATTR_ENTITY_ID] = entity_id
hass.services.call(ha.DOMAIN, SERVICE_TURN_ON, service_data)
def turn_off(hass, entity_id=None, **service_data):
2016-03-08 16:55:57 +00:00
"""Turn specified entity off."""
if entity_id is not None:
service_data[ATTR_ENTITY_ID] = entity_id
hass.services.call(ha.DOMAIN, SERVICE_TURN_OFF, service_data)
def toggle(hass, entity_id=None, **service_data):
2016-03-08 16:55:57 +00:00
"""Toggle specified entity."""
if entity_id is not None:
service_data[ATTR_ENTITY_ID] = entity_id
hass.services.call(ha.DOMAIN, SERVICE_TOGGLE, service_data)
def reload_core_config(hass):
"""Reload the core config."""
hass.services.call(ha.DOMAIN, SERVICE_RELOAD_CORE_CONFIG)
def setup(hass, config):
2016-03-08 16:55:57 +00:00
"""Setup general services related to Home Assistant."""
2014-04-13 19:59:45 +00:00
def handle_turn_service(service):
2016-03-08 16:55:57 +00:00
"""Method to handle calls to homeassistant.turn_on/off."""
entity_ids = extract_entity_ids(hass, service)
2014-04-13 19:59:45 +00:00
# Generic turn on/off method requires entity id
if not entity_ids:
2014-11-08 21:57:08 +00:00
_LOGGER.error(
"homeassistant/%s cannot be called without entity_id",
service.service)
2014-04-13 19:59:45 +00:00
return
2014-04-13 19:59:45 +00:00
# Group entity_ids by domain. groupby requires sorted data.
by_domain = it.groupby(sorted(entity_ids),
2016-08-09 03:21:40 +00:00
lambda item: ha.split_entity_id(item)[0])
2014-04-13 19:59:45 +00:00
for domain, ent_ids in by_domain:
# We want to block for all calls and only return when all calls
# have been processed. If a service does not exist it causes a 10
# second delay while we're blocking waiting for a response.
# But services can be registered on other HA instances that are
# listening to the bus too. So as a in between solution, we'll
# block only if the service is defined in the current HA instance.
blocking = hass.services.has_service(domain, service.service)
2014-04-13 19:59:45 +00:00
# Create a new dict for this call
data = dict(service.data)
# ent_ids is a generator, convert it to a list.
data[ATTR_ENTITY_ID] = list(ent_ids)
hass.services.call(domain, service.service, data, blocking)
hass.services.register(ha.DOMAIN, SERVICE_TURN_OFF, handle_turn_service)
hass.services.register(ha.DOMAIN, SERVICE_TURN_ON, handle_turn_service)
hass.services.register(ha.DOMAIN, SERVICE_TOGGLE, handle_turn_service)
def handle_reload_config(call):
"""Service handler for reloading core config."""
from homeassistant.exceptions import HomeAssistantError
from homeassistant import config as conf_util
try:
path = conf_util.find_config_file(hass.config.config_dir)
conf = conf_util.load_yaml_config_file(path)
except HomeAssistantError as err:
_LOGGER.error(err)
return
conf_util.process_ha_core_config(hass, conf.get(ha.DOMAIN) or {})
hass.services.register(ha.DOMAIN, SERVICE_RELOAD_CORE_CONFIG,
handle_reload_config)
return True