2016-08-25 12:47:07 +00:00
|
|
|
"""
|
|
|
|
Provides functionality to interact with fans.
|
|
|
|
|
|
|
|
For more details about this component, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/fan/
|
|
|
|
"""
|
2017-02-02 20:07:00 +00:00
|
|
|
import asyncio
|
2017-01-05 23:16:12 +00:00
|
|
|
from datetime import timedelta
|
2017-02-02 20:07:00 +00:00
|
|
|
import functools as ft
|
2016-08-25 12:47:07 +00:00
|
|
|
import logging
|
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.components import group
|
2016-08-27 20:53:12 +00:00
|
|
|
from homeassistant.const import (SERVICE_TURN_ON, SERVICE_TOGGLE,
|
|
|
|
SERVICE_TURN_OFF, ATTR_ENTITY_ID,
|
|
|
|
STATE_UNKNOWN)
|
2017-07-16 17:14:46 +00:00
|
|
|
from homeassistant.loader import bind_hass
|
2016-08-27 20:53:12 +00:00
|
|
|
from homeassistant.helpers.entity import ToggleEntity
|
2016-08-25 12:47:07 +00:00
|
|
|
from homeassistant.helpers.entity_component import EntityComponent
|
|
|
|
from homeassistant.helpers.config_validation import PLATFORM_SCHEMA # noqa
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
|
2017-04-30 05:04:49 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
2016-08-25 12:47:07 +00:00
|
|
|
|
|
|
|
DOMAIN = 'fan'
|
2017-06-15 22:52:28 +00:00
|
|
|
DEPENDENCIES = ['group']
|
2017-04-30 05:04:49 +00:00
|
|
|
SCAN_INTERVAL = timedelta(seconds=30)
|
2017-02-02 20:07:00 +00:00
|
|
|
|
2016-08-25 12:47:07 +00:00
|
|
|
GROUP_NAME_ALL_FANS = 'all fans'
|
|
|
|
ENTITY_ID_ALL_FANS = group.ENTITY_ID_FORMAT.format(GROUP_NAME_ALL_FANS)
|
|
|
|
|
|
|
|
ENTITY_ID_FORMAT = DOMAIN + '.{}'
|
|
|
|
|
|
|
|
# Bitfield of features supported by the fan entity
|
|
|
|
SUPPORT_SET_SPEED = 1
|
|
|
|
SUPPORT_OSCILLATE = 2
|
2017-01-14 06:08:13 +00:00
|
|
|
SUPPORT_DIRECTION = 4
|
2016-08-25 12:47:07 +00:00
|
|
|
|
|
|
|
SERVICE_SET_SPEED = 'set_speed'
|
|
|
|
SERVICE_OSCILLATE = 'oscillate'
|
2017-01-14 06:08:13 +00:00
|
|
|
SERVICE_SET_DIRECTION = 'set_direction'
|
2016-08-25 12:47:07 +00:00
|
|
|
|
|
|
|
SPEED_OFF = 'off'
|
|
|
|
SPEED_LOW = 'low'
|
2016-09-04 10:15:55 +00:00
|
|
|
SPEED_MEDIUM = 'medium'
|
2016-08-25 12:47:07 +00:00
|
|
|
SPEED_HIGH = 'high'
|
|
|
|
|
2017-01-14 06:08:13 +00:00
|
|
|
DIRECTION_FORWARD = 'forward'
|
|
|
|
DIRECTION_REVERSE = 'reverse'
|
|
|
|
|
2016-08-25 12:47:07 +00:00
|
|
|
ATTR_SPEED = 'speed'
|
|
|
|
ATTR_SPEED_LIST = 'speed_list'
|
2016-08-27 20:53:12 +00:00
|
|
|
ATTR_OSCILLATING = 'oscillating'
|
2017-01-14 06:08:13 +00:00
|
|
|
ATTR_DIRECTION = 'direction'
|
2016-08-25 12:47:07 +00:00
|
|
|
|
|
|
|
PROP_TO_ATTR = {
|
|
|
|
'speed': ATTR_SPEED,
|
|
|
|
'speed_list': ATTR_SPEED_LIST,
|
2016-08-27 20:53:12 +00:00
|
|
|
'oscillating': ATTR_OSCILLATING,
|
2017-01-14 06:08:13 +00:00
|
|
|
'direction': ATTR_DIRECTION,
|
2016-08-25 12:47:07 +00:00
|
|
|
} # type: dict
|
|
|
|
|
|
|
|
FAN_SET_SPEED_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(ATTR_ENTITY_ID): cv.entity_ids,
|
|
|
|
vol.Required(ATTR_SPEED): cv.string
|
|
|
|
}) # type: dict
|
|
|
|
|
|
|
|
FAN_TURN_ON_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(ATTR_ENTITY_ID): cv.entity_ids,
|
|
|
|
vol.Optional(ATTR_SPEED): cv.string
|
|
|
|
}) # type: dict
|
|
|
|
|
|
|
|
FAN_TURN_OFF_SCHEMA = vol.Schema({
|
2017-06-13 15:28:05 +00:00
|
|
|
vol.Optional(ATTR_ENTITY_ID): cv.entity_ids
|
2016-08-25 12:47:07 +00:00
|
|
|
}) # type: dict
|
|
|
|
|
|
|
|
FAN_OSCILLATE_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(ATTR_ENTITY_ID): cv.entity_ids,
|
2016-08-27 20:53:12 +00:00
|
|
|
vol.Required(ATTR_OSCILLATING): cv.boolean
|
2016-08-25 12:47:07 +00:00
|
|
|
}) # type: dict
|
|
|
|
|
2016-08-27 20:53:12 +00:00
|
|
|
FAN_TOGGLE_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(ATTR_ENTITY_ID): cv.entity_ids
|
|
|
|
})
|
|
|
|
|
2017-01-14 06:08:13 +00:00
|
|
|
FAN_SET_DIRECTION_SCHEMA = vol.Schema({
|
|
|
|
vol.Required(ATTR_ENTITY_ID): cv.entity_ids,
|
|
|
|
vol.Optional(ATTR_DIRECTION): cv.string
|
|
|
|
}) # type: dict
|
|
|
|
|
2017-02-02 20:07:00 +00:00
|
|
|
SERVICE_TO_METHOD = {
|
|
|
|
SERVICE_TURN_ON: {
|
|
|
|
'method': 'async_turn_on',
|
|
|
|
'schema': FAN_TURN_ON_SCHEMA,
|
|
|
|
},
|
|
|
|
SERVICE_TURN_OFF: {
|
|
|
|
'method': 'async_turn_off',
|
|
|
|
'schema': FAN_TURN_OFF_SCHEMA,
|
|
|
|
},
|
|
|
|
SERVICE_TOGGLE: {
|
|
|
|
'method': 'async_toggle',
|
|
|
|
'schema': FAN_TOGGLE_SCHEMA,
|
|
|
|
},
|
|
|
|
SERVICE_SET_SPEED: {
|
|
|
|
'method': 'async_set_speed',
|
|
|
|
'schema': FAN_SET_SPEED_SCHEMA,
|
|
|
|
},
|
|
|
|
SERVICE_OSCILLATE: {
|
|
|
|
'method': 'async_oscillate',
|
|
|
|
'schema': FAN_OSCILLATE_SCHEMA,
|
|
|
|
},
|
|
|
|
SERVICE_SET_DIRECTION: {
|
|
|
|
'method': 'async_set_direction',
|
|
|
|
'schema': FAN_SET_DIRECTION_SCHEMA,
|
|
|
|
},
|
|
|
|
}
|
2016-08-25 12:47:07 +00:00
|
|
|
|
|
|
|
|
2017-07-16 17:14:46 +00:00
|
|
|
@bind_hass
|
2016-08-25 12:47:07 +00:00
|
|
|
def is_on(hass, entity_id: str=None) -> bool:
|
|
|
|
"""Return if the fans are on based on the statemachine."""
|
|
|
|
entity_id = entity_id or ENTITY_ID_ALL_FANS
|
2016-08-27 20:53:12 +00:00
|
|
|
state = hass.states.get(entity_id)
|
|
|
|
return state.attributes[ATTR_SPEED] not in [SPEED_OFF, STATE_UNKNOWN]
|
2016-08-25 12:47:07 +00:00
|
|
|
|
|
|
|
|
2017-07-16 17:14:46 +00:00
|
|
|
@bind_hass
|
2016-08-25 12:47:07 +00:00
|
|
|
def turn_on(hass, entity_id: str=None, speed: str=None) -> None:
|
|
|
|
"""Turn all or specified fan on."""
|
|
|
|
data = {
|
|
|
|
key: value for key, value in [
|
|
|
|
(ATTR_ENTITY_ID, entity_id),
|
|
|
|
(ATTR_SPEED, speed),
|
|
|
|
] if value is not None
|
|
|
|
}
|
|
|
|
|
|
|
|
hass.services.call(DOMAIN, SERVICE_TURN_ON, data)
|
|
|
|
|
|
|
|
|
2017-07-16 17:14:46 +00:00
|
|
|
@bind_hass
|
2016-08-25 12:47:07 +00:00
|
|
|
def turn_off(hass, entity_id: str=None) -> None:
|
|
|
|
"""Turn all or specified fan off."""
|
2017-06-13 15:28:05 +00:00
|
|
|
data = {ATTR_ENTITY_ID: entity_id} if entity_id else {}
|
2016-08-25 12:47:07 +00:00
|
|
|
|
|
|
|
hass.services.call(DOMAIN, SERVICE_TURN_OFF, data)
|
|
|
|
|
|
|
|
|
2017-07-16 17:14:46 +00:00
|
|
|
@bind_hass
|
2016-08-27 20:53:12 +00:00
|
|
|
def toggle(hass, entity_id: str=None) -> None:
|
|
|
|
"""Toggle all or specified fans."""
|
|
|
|
data = {
|
|
|
|
ATTR_ENTITY_ID: entity_id
|
|
|
|
}
|
|
|
|
|
|
|
|
hass.services.call(DOMAIN, SERVICE_TOGGLE, data)
|
|
|
|
|
|
|
|
|
2017-07-16 17:14:46 +00:00
|
|
|
@bind_hass
|
2016-08-25 12:47:07 +00:00
|
|
|
def oscillate(hass, entity_id: str=None, should_oscillate: bool=True) -> None:
|
|
|
|
"""Set oscillation on all or specified fan."""
|
|
|
|
data = {
|
|
|
|
key: value for key, value in [
|
|
|
|
(ATTR_ENTITY_ID, entity_id),
|
2016-08-27 20:53:12 +00:00
|
|
|
(ATTR_OSCILLATING, should_oscillate),
|
2016-08-25 12:47:07 +00:00
|
|
|
] if value is not None
|
|
|
|
}
|
|
|
|
|
|
|
|
hass.services.call(DOMAIN, SERVICE_OSCILLATE, data)
|
|
|
|
|
|
|
|
|
2017-07-16 17:14:46 +00:00
|
|
|
@bind_hass
|
2016-08-25 12:47:07 +00:00
|
|
|
def set_speed(hass, entity_id: str=None, speed: str=None) -> None:
|
|
|
|
"""Set speed for all or specified fan."""
|
|
|
|
data = {
|
|
|
|
key: value for key, value in [
|
|
|
|
(ATTR_ENTITY_ID, entity_id),
|
|
|
|
(ATTR_SPEED, speed),
|
|
|
|
] if value is not None
|
|
|
|
}
|
|
|
|
|
|
|
|
hass.services.call(DOMAIN, SERVICE_SET_SPEED, data)
|
|
|
|
|
|
|
|
|
2017-07-16 17:14:46 +00:00
|
|
|
@bind_hass
|
2017-01-14 06:08:13 +00:00
|
|
|
def set_direction(hass, entity_id: str=None, direction: str=None) -> None:
|
|
|
|
"""Set direction for all or specified fan."""
|
|
|
|
data = {
|
|
|
|
key: value for key, value in [
|
|
|
|
(ATTR_ENTITY_ID, entity_id),
|
|
|
|
(ATTR_DIRECTION, direction),
|
|
|
|
] if value is not None
|
|
|
|
}
|
|
|
|
|
|
|
|
hass.services.call(DOMAIN, SERVICE_SET_DIRECTION, data)
|
|
|
|
|
|
|
|
|
2017-02-02 20:07:00 +00:00
|
|
|
@asyncio.coroutine
|
|
|
|
def async_setup(hass, config: dict):
|
2016-08-25 12:47:07 +00:00
|
|
|
"""Expose fan control via statemachine and services."""
|
|
|
|
component = EntityComponent(
|
|
|
|
_LOGGER, DOMAIN, hass, SCAN_INTERVAL, GROUP_NAME_ALL_FANS)
|
|
|
|
|
2017-02-02 20:07:00 +00:00
|
|
|
yield from component.async_setup(config)
|
|
|
|
|
|
|
|
@asyncio.coroutine
|
|
|
|
def async_handle_fan_service(service):
|
2018-01-29 22:37:19 +00:00
|
|
|
"""Handle service call for fans."""
|
2017-02-02 20:07:00 +00:00
|
|
|
method = SERVICE_TO_METHOD.get(service.service)
|
2016-08-25 12:47:07 +00:00
|
|
|
params = service.data.copy()
|
|
|
|
|
|
|
|
# Convert the entity ids to valid fan ids
|
2017-02-02 20:07:00 +00:00
|
|
|
target_fans = component.async_extract_from_service(service)
|
2016-08-25 12:47:07 +00:00
|
|
|
params.pop(ATTR_ENTITY_ID, None)
|
|
|
|
|
2017-02-02 20:07:00 +00:00
|
|
|
update_tasks = []
|
|
|
|
for fan in target_fans:
|
2017-10-19 08:56:25 +00:00
|
|
|
yield from getattr(fan, method['method'])(**params)
|
2017-02-02 20:07:00 +00:00
|
|
|
if not fan.should_poll:
|
|
|
|
continue
|
2017-10-19 08:56:25 +00:00
|
|
|
update_tasks.append(fan.async_update_ha_state(True))
|
2016-08-25 12:47:07 +00:00
|
|
|
|
2017-02-02 20:07:00 +00:00
|
|
|
if update_tasks:
|
|
|
|
yield from asyncio.wait(update_tasks, loop=hass.loop)
|
2016-08-25 12:47:07 +00:00
|
|
|
|
2017-02-02 20:07:00 +00:00
|
|
|
for service_name in SERVICE_TO_METHOD:
|
|
|
|
schema = SERVICE_TO_METHOD[service_name].get('schema')
|
|
|
|
hass.services.async_register(
|
2018-01-07 22:54:16 +00:00
|
|
|
DOMAIN, service_name, async_handle_fan_service, schema=schema)
|
2017-01-14 06:08:13 +00:00
|
|
|
|
2016-08-25 12:47:07 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
2016-08-27 20:53:12 +00:00
|
|
|
class FanEntity(ToggleEntity):
|
2016-08-25 12:47:07 +00:00
|
|
|
"""Representation of a fan."""
|
|
|
|
|
2016-08-27 20:53:12 +00:00
|
|
|
def set_speed(self: ToggleEntity, speed: str) -> None:
|
2016-08-25 12:47:07 +00:00
|
|
|
"""Set the speed of the fan."""
|
2017-01-14 06:08:13 +00:00
|
|
|
raise NotImplementedError()
|
|
|
|
|
2017-02-02 20:07:00 +00:00
|
|
|
def async_set_speed(self: ToggleEntity, speed: str):
|
|
|
|
"""Set the speed of the fan.
|
|
|
|
|
|
|
|
This method must be run in the event loop and returns a coroutine.
|
|
|
|
"""
|
|
|
|
if speed is SPEED_OFF:
|
|
|
|
return self.async_turn_off()
|
2017-05-26 15:28:07 +00:00
|
|
|
return self.hass.async_add_job(self.set_speed, speed)
|
2017-02-02 20:07:00 +00:00
|
|
|
|
2017-01-14 06:08:13 +00:00
|
|
|
def set_direction(self: ToggleEntity, direction: str) -> None:
|
|
|
|
"""Set the direction of the fan."""
|
|
|
|
raise NotImplementedError()
|
2016-08-25 12:47:07 +00:00
|
|
|
|
2017-02-02 20:07:00 +00:00
|
|
|
def async_set_direction(self: ToggleEntity, direction: str):
|
|
|
|
"""Set the direction of the fan.
|
|
|
|
|
|
|
|
This method must be run in the event loop and returns a coroutine.
|
|
|
|
"""
|
2017-05-26 15:28:07 +00:00
|
|
|
return self.hass.async_add_job(self.set_direction, direction)
|
2017-02-02 20:07:00 +00:00
|
|
|
|
2016-08-27 20:53:12 +00:00
|
|
|
def turn_on(self: ToggleEntity, speed: str=None, **kwargs) -> None:
|
2016-08-25 12:47:07 +00:00
|
|
|
"""Turn on the fan."""
|
2016-08-27 20:53:12 +00:00
|
|
|
raise NotImplementedError()
|
2016-08-25 12:47:07 +00:00
|
|
|
|
2017-02-02 20:07:00 +00:00
|
|
|
def async_turn_on(self: ToggleEntity, speed: str=None, **kwargs):
|
|
|
|
"""Turn on the fan.
|
|
|
|
|
|
|
|
This method must be run in the event loop and returns a coroutine.
|
|
|
|
"""
|
|
|
|
if speed is SPEED_OFF:
|
|
|
|
return self.async_turn_off()
|
2017-05-26 15:28:07 +00:00
|
|
|
return self.hass.async_add_job(
|
|
|
|
ft.partial(self.turn_on, speed, **kwargs))
|
2016-08-25 12:47:07 +00:00
|
|
|
|
2016-08-27 20:53:12 +00:00
|
|
|
def oscillate(self: ToggleEntity, oscillating: bool) -> None:
|
2016-08-25 12:47:07 +00:00
|
|
|
"""Oscillate the fan."""
|
|
|
|
pass
|
|
|
|
|
2017-02-02 20:07:00 +00:00
|
|
|
def async_oscillate(self: ToggleEntity, oscillating: bool):
|
|
|
|
"""Oscillate the fan.
|
|
|
|
|
|
|
|
This method must be run in the event loop and returns a coroutine.
|
|
|
|
"""
|
2017-05-26 15:28:07 +00:00
|
|
|
return self.hass.async_add_job(self.oscillate, oscillating)
|
2017-02-02 20:07:00 +00:00
|
|
|
|
2016-08-25 12:47:07 +00:00
|
|
|
@property
|
2016-08-27 20:53:12 +00:00
|
|
|
def is_on(self):
|
|
|
|
"""Return true if the entity is on."""
|
2017-01-14 06:08:13 +00:00
|
|
|
return self.speed not in [SPEED_OFF, STATE_UNKNOWN]
|
|
|
|
|
|
|
|
@property
|
|
|
|
def speed(self) -> str:
|
|
|
|
"""Return the current speed."""
|
|
|
|
return None
|
2016-08-27 20:53:12 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def speed_list(self: ToggleEntity) -> list:
|
2016-08-25 12:47:07 +00:00
|
|
|
"""Get the list of available speeds."""
|
|
|
|
return []
|
|
|
|
|
2017-01-14 06:08:13 +00:00
|
|
|
@property
|
|
|
|
def current_direction(self) -> str:
|
|
|
|
"""Return the current direction of the fan."""
|
|
|
|
return None
|
|
|
|
|
2016-08-25 12:47:07 +00:00
|
|
|
@property
|
2016-08-27 20:53:12 +00:00
|
|
|
def state_attributes(self: ToggleEntity) -> dict:
|
2016-08-25 12:47:07 +00:00
|
|
|
"""Return optional state attributes."""
|
|
|
|
data = {} # type: dict
|
|
|
|
|
|
|
|
for prop, attr in PROP_TO_ATTR.items():
|
2016-08-27 20:53:12 +00:00
|
|
|
if not hasattr(self, prop):
|
|
|
|
continue
|
|
|
|
|
2016-08-25 12:47:07 +00:00
|
|
|
value = getattr(self, prop)
|
|
|
|
if value is not None:
|
|
|
|
data[attr] = value
|
|
|
|
|
|
|
|
return data
|
|
|
|
|
|
|
|
@property
|
2016-08-27 20:53:12 +00:00
|
|
|
def supported_features(self: ToggleEntity) -> int:
|
2016-08-25 12:47:07 +00:00
|
|
|
"""Flag supported features."""
|
|
|
|
return 0
|