core/homeassistant/components/light/__init__.py

279 lines
8.7 KiB
Python
Raw Normal View History

"""
homeassistant.components.light
~~~~~~~~~~~~~~~~~~~~~~~~~~~~
Provides functionality to interact with lights.
2014-03-26 07:08:50 +00:00
It offers the following services:
TURN_OFF - Turns one or multiple lights off.
Supports following parameters:
- transition
Integer that represents the time the light should take to transition to
the new state.
- entity_id
String or list of strings that point at entity_ids of lights.
TURN_ON - Turns one or multiple lights on and change attributes.
Supports following parameters:
- transition
Integer that represents the time the light should take to transition to
the new state.
- entity_id
String or list of strings that point at entity_ids of lights.
- profile
String with the name of one of the built-in profiles (relax, energize,
concentrate, reading) or one of the custom profiles defined in
light_profiles.csv in the current working directory.
Light profiles define a xy color and a brightness.
If a profile is given and a brightness or xy color then the profile values
will be overwritten.
- xy_color
A list containing two floats representing the xy color you want the light
to be.
- rgb_color
A list containing three integers representing the xy color you want the
light to be.
- brightness
Integer between 0 and 255 representing how bright you want the light to be.
"""
import logging
2014-03-26 07:08:50 +00:00
import os
import csv
import homeassistant.util as util
from homeassistant.const import (
STATE_ON, SERVICE_TURN_ON, SERVICE_TURN_OFF, ATTR_ENTITY_ID)
from homeassistant.helpers import (
extract_entity_ids, platform_devices_from_config)
from homeassistant.components import group
DOMAIN = "light"
DEPENDENCIES = []
GROUP_NAME_ALL_LIGHTS = 'all_lights'
ENTITY_ID_ALL_LIGHTS = group.ENTITY_ID_FORMAT.format(
GROUP_NAME_ALL_LIGHTS)
ENTITY_ID_FORMAT = DOMAIN + ".{}"
# integer that represents transition time in seconds to make change
ATTR_TRANSITION = "transition"
# lists holding color values
ATTR_RGB_COLOR = "rgb_color"
ATTR_XY_COLOR = "xy_color"
# int with value 0 .. 255 representing brightness of the light
ATTR_BRIGHTNESS = "brightness"
2014-03-26 07:08:50 +00:00
# String representing a profile (built-in ones or external defined)
ATTR_PROFILE = "profile"
# If the light should flash, can be FLASH_SHORT or FLASH_LONG
ATTR_FLASH = "flash"
FLASH_SHORT = "short"
FLASH_LONG = "long"
2014-03-26 07:08:50 +00:00
LIGHT_PROFILES_FILE = "light_profiles.csv"
2014-11-09 23:12:23 +00:00
_LOGGER = logging.getLogger(__name__)
def is_on(hass, entity_id=None):
""" Returns if the lights are on based on the statemachine. """
entity_id = entity_id or ENTITY_ID_ALL_LIGHTS
return hass.states.is_state(entity_id, STATE_ON)
# pylint: disable=too-many-arguments
def turn_on(hass, entity_id=None, transition=None, brightness=None,
rgb_color=None, xy_color=None, profile=None, flash=None):
""" Turns all or specified light on. """
data = {
key: value for key, value in [
(ATTR_ENTITY_ID, entity_id),
(ATTR_PROFILE, profile),
(ATTR_TRANSITION, transition),
(ATTR_BRIGHTNESS, brightness),
(ATTR_RGB_COLOR, rgb_color),
(ATTR_XY_COLOR, xy_color),
(ATTR_FLASH, flash),
] if value is not None
}
hass.services.call(DOMAIN, SERVICE_TURN_ON, data)
def turn_off(hass, entity_id=None, transition=None):
""" Turns all or specified light off. """
data = {
key: value for key, value in [
(ATTR_ENTITY_ID, entity_id),
(ATTR_TRANSITION, transition),
] if value is not None
}
hass.services.call(DOMAIN, SERVICE_TURN_OFF, data)
2014-03-26 07:08:50 +00:00
# pylint: disable=too-many-branches, too-many-locals
def setup(hass, config):
""" Exposes light control via statemachine and services. """
2014-11-26 07:16:07 +00:00
# Load built-in profiles and custom profiles
profile_paths = [os.path.join(os.path.dirname(__file__),
LIGHT_PROFILES_FILE),
hass.get_config_path(LIGHT_PROFILES_FILE)]
profiles = {}
for profile_path in profile_paths:
if os.path.isfile(profile_path):
with open(profile_path) as inp:
reader = csv.reader(inp)
# Skip the header
next(reader, None)
try:
for profile_id, color_x, color_y, brightness in reader:
profiles[profile_id] = (float(color_x), float(color_y),
int(brightness))
except ValueError:
# ValueError if not 4 values per row
# ValueError if convert to float/int failed
_LOGGER.error(
"Error parsing light profiles from %s", profile_path)
return False
lights = platform_devices_from_config(
config, DOMAIN, hass, ENTITY_ID_FORMAT, _LOGGER)
if not lights:
2014-11-09 23:12:23 +00:00
return False
2014-11-09 23:12:23 +00:00
# pylint: disable=unused-argument
def update_lights_state(now):
""" Update the states of all the lights. """
for light in lights.values():
2014-11-09 23:12:23 +00:00
light.update_ha_state(hass)
update_lights_state(None)
# Track all lights in a group
2015-01-09 04:02:34 +00:00
group.Group(hass, GROUP_NAME_ALL_LIGHTS, lights.keys(), False)
2014-11-09 23:12:23 +00:00
def handle_light_service(service):
""" Hande a turn light on or off service call. """
# Get and validate data
dat = service.data
# Convert the entity ids to valid light ids
target_lights = [lights[entity_id] for entity_id
in extract_entity_ids(hass, service)
if entity_id in lights]
if not target_lights:
target_lights = lights.values()
2014-11-26 05:28:43 +00:00
params = {}
2014-03-26 07:08:50 +00:00
transition = util.convert(dat.get(ATTR_TRANSITION), int)
2014-11-26 05:28:43 +00:00
if transition is not None:
params[ATTR_TRANSITION] = transition
if service.service == SERVICE_TURN_OFF:
for light in target_lights:
2014-11-26 05:28:43 +00:00
# pylint: disable=star-args
light.turn_off(**params)
else:
# Processing extra data for turn light on request
2014-03-26 07:08:50 +00:00
# We process the profile first so that we get the desired
# behavior that extra service data attributes overwrite
# profile values
profile = profiles.get(dat.get(ATTR_PROFILE))
if profile:
2014-11-26 05:28:43 +00:00
*params[ATTR_XY_COLOR], params[ATTR_BRIGHTNESS] = profile
2014-03-26 07:08:50 +00:00
if ATTR_BRIGHTNESS in dat:
2014-11-26 05:28:43 +00:00
# We pass in the old value as the default parameter if parsing
# of the new one goes wrong.
params[ATTR_BRIGHTNESS] = util.convert(
dat.get(ATTR_BRIGHTNESS), int, params.get(ATTR_BRIGHTNESS))
2014-03-26 07:08:50 +00:00
if ATTR_XY_COLOR in dat:
try:
# xy_color should be a list containing 2 floats
2014-11-26 05:28:43 +00:00
xycolor = dat.get(ATTR_XY_COLOR)
2014-11-26 05:38:47 +00:00
# Without this check, a xycolor with value '99' would work
if not isinstance(xycolor, str):
2014-11-26 05:28:43 +00:00
params[ATTR_XY_COLOR] = [float(val) for val in xycolor]
except (TypeError, ValueError):
2014-04-15 06:48:00 +00:00
# TypeError if xy_color is not iterable
# ValueError if value could not be converted to float
pass
2014-03-26 07:08:50 +00:00
if ATTR_RGB_COLOR in dat:
try:
# rgb_color should be a list containing 3 ints
2014-04-15 06:48:00 +00:00
rgb_color = dat.get(ATTR_RGB_COLOR)
if len(rgb_color) == 3:
2014-11-26 05:28:43 +00:00
params[ATTR_XY_COLOR] = \
util.color_RGB_to_xy(int(rgb_color[0]),
int(rgb_color[1]),
int(rgb_color[2]))
except (TypeError, ValueError):
2014-04-15 06:48:00 +00:00
# TypeError if rgb_color is not iterable
2014-03-26 07:08:50 +00:00
# ValueError if not all values can be converted to int
pass
if ATTR_FLASH in dat:
if dat[ATTR_FLASH] == FLASH_SHORT:
params[ATTR_FLASH] = FLASH_SHORT
elif dat[ATTR_FLASH] == FLASH_LONG:
params[ATTR_FLASH] = FLASH_LONG
for light in target_lights:
2014-11-26 05:28:43 +00:00
# pylint: disable=star-args
light.turn_on(**params)
for light in target_lights:
2014-11-09 23:12:23 +00:00
light.update_ha_state(hass, True)
# Update light state every 30 seconds
hass.track_time_change(update_lights_state, second=[0, 30])
# Listen for light on and light off service calls
hass.services.register(DOMAIN, SERVICE_TURN_ON,
handle_light_service)
hass.services.register(DOMAIN, SERVICE_TURN_OFF,
handle_light_service)
return True