2019-04-03 15:40:03 +00:00
|
|
|
"""Support for Qwikswitch devices."""
|
2016-05-13 04:39:30 +00:00
|
|
|
import logging
|
2017-01-17 22:40:34 +00:00
|
|
|
|
2016-07-24 00:03:29 +00:00
|
|
|
import voluptuous as vol
|
2016-05-13 04:39:30 +00:00
|
|
|
|
2018-04-21 06:34:42 +00:00
|
|
|
from homeassistant.components.binary_sensor import DEVICE_CLASSES_SCHEMA
|
|
|
|
from homeassistant.components.light import ATTR_BRIGHTNESS
|
2017-01-17 22:40:34 +00:00
|
|
|
from homeassistant.const import (
|
2018-04-21 06:34:42 +00:00
|
|
|
CONF_SENSORS, CONF_SWITCHES, CONF_URL, EVENT_HOMEASSISTANT_START,
|
|
|
|
EVENT_HOMEASSISTANT_STOP)
|
2018-03-25 21:32:13 +00:00
|
|
|
from homeassistant.core import callback
|
|
|
|
from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
2018-04-21 06:34:42 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2016-07-24 00:03:29 +00:00
|
|
|
from homeassistant.helpers.discovery import load_platform
|
2018-03-29 21:29:46 +00:00
|
|
|
from homeassistant.helpers.entity import Entity
|
2016-07-24 00:03:29 +00:00
|
|
|
|
2016-05-13 04:39:30 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2017-01-17 22:40:34 +00:00
|
|
|
DOMAIN = 'qwikswitch'
|
|
|
|
|
|
|
|
CONF_DIMMER_ADJUST = 'dimmer_adjust'
|
|
|
|
CONF_BUTTON_EVENTS = 'button_events'
|
2016-07-24 00:03:29 +00:00
|
|
|
CV_DIM_VALUE = vol.All(vol.Coerce(float), vol.Range(min=1, max=3))
|
2017-01-17 22:40:34 +00:00
|
|
|
|
2018-04-21 06:34:42 +00:00
|
|
|
|
2016-07-24 00:03:29 +00:00
|
|
|
CONFIG_SCHEMA = vol.Schema({
|
|
|
|
DOMAIN: vol.Schema({
|
2017-01-17 22:40:34 +00:00
|
|
|
vol.Required(CONF_URL, default='http://127.0.0.1:2020'):
|
|
|
|
vol.Coerce(str),
|
|
|
|
vol.Optional(CONF_DIMMER_ADJUST, default=1): CV_DIM_VALUE,
|
2018-03-29 21:29:46 +00:00
|
|
|
vol.Optional(CONF_BUTTON_EVENTS, default=[]): cv.ensure_list_csv,
|
2018-04-08 19:59:19 +00:00
|
|
|
vol.Optional(CONF_SENSORS, default=[]): vol.All(
|
|
|
|
cv.ensure_list, [vol.Schema({
|
|
|
|
vol.Required('id'): str,
|
|
|
|
vol.Optional('channel', default=1): int,
|
|
|
|
vol.Required('name'): str,
|
|
|
|
vol.Required('type'): str,
|
2018-04-21 06:34:42 +00:00
|
|
|
vol.Optional('class'): DEVICE_CLASSES_SCHEMA,
|
|
|
|
vol.Optional('invert'): bool
|
2018-04-08 19:59:19 +00:00
|
|
|
})]),
|
2018-03-29 21:29:46 +00:00
|
|
|
vol.Optional(CONF_SWITCHES, default=[]): vol.All(
|
|
|
|
cv.ensure_list, [str])
|
2016-07-24 00:03:29 +00:00
|
|
|
})}, extra=vol.ALLOW_EXTRA)
|
|
|
|
|
2016-05-13 04:39:30 +00:00
|
|
|
|
2018-04-08 19:59:19 +00:00
|
|
|
class QSEntity(Entity):
|
|
|
|
"""Qwikswitch Entity base."""
|
|
|
|
|
|
|
|
def __init__(self, qsid, name):
|
|
|
|
"""Initialize the QSEntity."""
|
|
|
|
self._name = name
|
|
|
|
self.qsid = qsid
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the sensor."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def poll(self):
|
|
|
|
"""QS sensors gets packets in update_packet."""
|
|
|
|
return False
|
|
|
|
|
2018-04-09 23:24:06 +00:00
|
|
|
@property
|
|
|
|
def unique_id(self):
|
|
|
|
"""Return a unique identifier for this sensor."""
|
|
|
|
return "qs{}".format(self.qsid)
|
|
|
|
|
|
|
|
@callback
|
2018-04-08 19:59:19 +00:00
|
|
|
def update_packet(self, packet):
|
|
|
|
"""Receive update packet from QSUSB. Match dispather_send signature."""
|
|
|
|
self.async_schedule_update_ha_state()
|
|
|
|
|
|
|
|
async def async_added_to_hass(self):
|
|
|
|
"""Listen for updates from QSUSb via dispatcher."""
|
|
|
|
self.hass.helpers.dispatcher.async_dispatcher_connect(
|
|
|
|
self.qsid, self.update_packet)
|
2016-05-13 04:39:30 +00:00
|
|
|
|
2018-04-08 19:59:19 +00:00
|
|
|
|
|
|
|
class QSToggleEntity(QSEntity):
|
|
|
|
"""Representation of a Qwikswitch Toggle Entity.
|
2016-05-13 04:39:30 +00:00
|
|
|
|
|
|
|
Implemented:
|
|
|
|
- QSLight extends QSToggleEntity and Light[2] (ToggleEntity[1])
|
|
|
|
- QSSwitch extends QSToggleEntity and SwitchDevice[3] (ToggleEntity[1])
|
|
|
|
|
|
|
|
[1] /helpers/entity.py
|
|
|
|
[2] /components/light/__init__.py
|
|
|
|
[3] /components/switch/__init__.py
|
|
|
|
"""
|
|
|
|
|
2018-03-25 21:32:13 +00:00
|
|
|
def __init__(self, qsid, qsusb):
|
2016-05-14 21:21:05 +00:00
|
|
|
"""Initialize the ToggleEntity."""
|
2018-08-26 19:25:39 +00:00
|
|
|
self.device = qsusb.devices[qsid]
|
|
|
|
super().__init__(qsid, self.device.name)
|
2016-05-13 04:39:30 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self):
|
2016-05-21 14:59:52 +00:00
|
|
|
"""Check if device is on (non-zero)."""
|
2018-08-26 19:25:39 +00:00
|
|
|
return self.device.value > 0
|
2016-05-13 04:39:30 +00:00
|
|
|
|
2018-03-29 21:29:46 +00:00
|
|
|
async def async_turn_on(self, **kwargs):
|
2016-05-13 04:39:30 +00:00
|
|
|
"""Turn the device on."""
|
2018-03-25 21:32:13 +00:00
|
|
|
new = kwargs.get(ATTR_BRIGHTNESS, 255)
|
2018-04-08 19:59:19 +00:00
|
|
|
self.hass.data[DOMAIN].devices.set_value(self.qsid, new)
|
2018-03-25 21:32:13 +00:00
|
|
|
|
2018-03-29 21:29:46 +00:00
|
|
|
async def async_turn_off(self, **_):
|
2016-05-13 04:39:30 +00:00
|
|
|
"""Turn the device off."""
|
2018-04-08 19:59:19 +00:00
|
|
|
self.hass.data[DOMAIN].devices.set_value(self.qsid, 0)
|
2016-07-24 00:03:29 +00:00
|
|
|
|
|
|
|
|
2018-03-29 21:29:46 +00:00
|
|
|
async def async_setup(hass, config):
|
|
|
|
"""Qwiskswitch component setup."""
|
|
|
|
from pyqwikswitch.async_ import QSUsb
|
2019-04-02 03:57:25 +00:00
|
|
|
from pyqwikswitch.qwikswitch import (
|
|
|
|
CMD_BUTTONS, QS_CMD, QS_ID, QSType, SENSORS)
|
2016-05-14 21:21:05 +00:00
|
|
|
|
2018-03-29 21:29:46 +00:00
|
|
|
# Add cmd's to in /&listen packets will fire events
|
2016-05-14 21:21:05 +00:00
|
|
|
# By default only buttons of type [TOGGLE,SCENE EXE,LEVEL]
|
2018-03-29 21:29:46 +00:00
|
|
|
cmd_buttons = set(CMD_BUTTONS)
|
|
|
|
for btn in config[DOMAIN][CONF_BUTTON_EVENTS]:
|
|
|
|
cmd_buttons.add(btn)
|
2016-05-13 04:39:30 +00:00
|
|
|
|
2017-01-17 22:40:34 +00:00
|
|
|
url = config[DOMAIN][CONF_URL]
|
|
|
|
dimmer_adjust = config[DOMAIN][CONF_DIMMER_ADJUST]
|
2018-04-08 19:59:19 +00:00
|
|
|
sensors = config[DOMAIN][CONF_SENSORS]
|
|
|
|
switches = config[DOMAIN][CONF_SWITCHES]
|
2016-05-13 04:39:30 +00:00
|
|
|
|
2018-03-29 21:29:46 +00:00
|
|
|
def callback_value_changed(_qsd, qsid, _val):
|
|
|
|
"""Update entity values based on device change."""
|
|
|
|
_LOGGER.debug("Dispatch %s (update from devices)", qsid)
|
|
|
|
hass.helpers.dispatcher.async_dispatcher_send(qsid, None)
|
2018-03-25 21:32:13 +00:00
|
|
|
|
|
|
|
session = async_get_clientsession(hass)
|
|
|
|
qsusb = QSUsb(url=url, dim_adj=dimmer_adjust, session=session,
|
|
|
|
callback_value_changed=callback_value_changed)
|
2016-05-14 21:21:05 +00:00
|
|
|
|
2018-03-29 21:29:46 +00:00
|
|
|
# Discover all devices in QSUSB
|
|
|
|
if not await qsusb.update_from_devices():
|
|
|
|
return False
|
2016-07-24 00:03:29 +00:00
|
|
|
|
2018-03-29 21:29:46 +00:00
|
|
|
hass.data[DOMAIN] = qsusb
|
2016-07-24 00:03:29 +00:00
|
|
|
|
2018-04-21 06:34:42 +00:00
|
|
|
comps = {'switch': [], 'light': [], 'sensor': [], 'binary_sensor': []}
|
|
|
|
|
|
|
|
try:
|
2018-05-14 06:40:25 +00:00
|
|
|
sensor_ids = []
|
2018-04-21 06:34:42 +00:00
|
|
|
for sens in sensors:
|
|
|
|
_, _type = SENSORS[sens['type']]
|
2018-05-14 06:40:25 +00:00
|
|
|
sensor_ids.append(sens['id'])
|
2018-04-21 06:34:42 +00:00
|
|
|
if _type is bool:
|
|
|
|
comps['binary_sensor'].append(sens)
|
|
|
|
continue
|
|
|
|
comps['sensor'].append(sens)
|
|
|
|
for _key in ('invert', 'class'):
|
|
|
|
if _key in sens:
|
|
|
|
_LOGGER.warning(
|
|
|
|
"%s should only be used for binary_sensors: %s",
|
|
|
|
_key, sens)
|
|
|
|
|
|
|
|
except KeyError:
|
|
|
|
_LOGGER.warning("Sensor validation failed")
|
|
|
|
|
2018-04-08 19:59:19 +00:00
|
|
|
for qsid, dev in qsusb.devices.items():
|
|
|
|
if qsid in switches:
|
|
|
|
if dev.qstype != QSType.relay:
|
2018-03-29 21:29:46 +00:00
|
|
|
_LOGGER.warning(
|
2018-04-08 19:59:19 +00:00
|
|
|
"You specified a switch that is not a relay %s", qsid)
|
2018-03-29 21:29:46 +00:00
|
|
|
continue
|
2018-04-21 06:34:42 +00:00
|
|
|
comps['switch'].append(qsid)
|
2018-04-08 19:59:19 +00:00
|
|
|
elif dev.qstype in (QSType.relay, QSType.dimmer):
|
2018-04-21 06:34:42 +00:00
|
|
|
comps['light'].append(qsid)
|
2016-07-24 00:03:29 +00:00
|
|
|
else:
|
2018-04-08 19:59:19 +00:00
|
|
|
_LOGGER.warning("Ignored unknown QSUSB device: %s", dev)
|
2018-03-25 21:32:13 +00:00
|
|
|
continue
|
2016-07-24 00:03:29 +00:00
|
|
|
|
|
|
|
# Load platforms
|
2018-04-21 06:34:42 +00:00
|
|
|
for comp_name, comp_conf in comps.items():
|
2018-03-29 21:29:46 +00:00
|
|
|
if comp_conf:
|
|
|
|
load_platform(hass, comp_name, DOMAIN, {DOMAIN: comp_conf}, config)
|
2016-05-13 04:39:30 +00:00
|
|
|
|
2018-04-08 19:59:19 +00:00
|
|
|
def callback_qs_listen(qspacket):
|
2016-05-21 14:59:52 +00:00
|
|
|
"""Typically a button press or update signal."""
|
2016-05-13 04:39:30 +00:00
|
|
|
# If button pressed, fire a hass event
|
2018-04-08 19:59:19 +00:00
|
|
|
if QS_ID in qspacket:
|
|
|
|
if qspacket.get(QS_CMD, '') in cmd_buttons:
|
2018-03-29 21:29:46 +00:00
|
|
|
hass.bus.async_fire(
|
2018-04-08 19:59:19 +00:00
|
|
|
'qwikswitch.button.{}'.format(qspacket[QS_ID]), qspacket)
|
2018-03-29 21:29:46 +00:00
|
|
|
return
|
|
|
|
|
2018-05-14 06:40:25 +00:00
|
|
|
if qspacket[QS_ID] in sensor_ids:
|
2018-04-08 19:59:19 +00:00
|
|
|
_LOGGER.debug("Dispatch %s ((%s))", qspacket[QS_ID], qspacket)
|
2018-03-29 21:29:46 +00:00
|
|
|
hass.helpers.dispatcher.async_dispatcher_send(
|
2018-04-08 19:59:19 +00:00
|
|
|
qspacket[QS_ID], qspacket)
|
2016-05-13 04:39:30 +00:00
|
|
|
|
|
|
|
# Update all ha_objects
|
2018-03-25 21:32:13 +00:00
|
|
|
hass.async_add_job(qsusb.update_from_devices)
|
2016-05-13 04:39:30 +00:00
|
|
|
|
2018-03-25 21:32:13 +00:00
|
|
|
@callback
|
2018-03-29 21:29:46 +00:00
|
|
|
def async_start(_):
|
2016-07-24 00:03:29 +00:00
|
|
|
"""Start listening."""
|
2018-03-25 21:32:13 +00:00
|
|
|
hass.async_add_job(qsusb.listen, callback_qs_listen)
|
|
|
|
|
|
|
|
hass.bus.async_listen_once(EVENT_HOMEASSISTANT_START, async_start)
|
2016-07-24 00:03:29 +00:00
|
|
|
|
2018-03-29 21:29:46 +00:00
|
|
|
@callback
|
|
|
|
def async_stop(_):
|
2018-04-21 06:34:42 +00:00
|
|
|
"""Stop the listener."""
|
2018-03-29 21:29:46 +00:00
|
|
|
hass.data[DOMAIN].stop()
|
|
|
|
|
|
|
|
hass.bus.async_listen(EVENT_HOMEASSISTANT_STOP, async_stop)
|
|
|
|
|
2016-05-13 04:39:30 +00:00
|
|
|
return True
|