2016-12-16 05:42:00 +00:00
|
|
|
"""
|
|
|
|
Support for Broadlink RM devices.
|
|
|
|
|
|
|
|
For more details about this platform, please refer to the documentation at
|
|
|
|
https://home-assistant.io/components/switch.broadlink/
|
|
|
|
"""
|
|
|
|
from datetime import timedelta
|
|
|
|
from base64 import b64encode, b64decode
|
|
|
|
import asyncio
|
|
|
|
import binascii
|
|
|
|
import logging
|
|
|
|
import socket
|
2017-05-02 16:18:47 +00:00
|
|
|
|
2016-12-16 05:42:00 +00:00
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.util.dt import utcnow
|
|
|
|
from homeassistant.components.switch import (SwitchDevice, PLATFORM_SCHEMA)
|
2017-05-02 16:18:47 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
CONF_FRIENDLY_NAME, CONF_SWITCHES, CONF_COMMAND_OFF, CONF_COMMAND_ON,
|
|
|
|
CONF_TIMEOUT, CONF_HOST, CONF_MAC, CONF_TYPE)
|
2016-12-16 05:42:00 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
|
2017-07-13 12:42:30 +00:00
|
|
|
REQUIREMENTS = ['broadlink==0.5']
|
2016-12-16 05:42:00 +00:00
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2017-05-02 16:18:47 +00:00
|
|
|
DOMAIN = 'broadlink'
|
2016-12-16 05:42:00 +00:00
|
|
|
DEFAULT_NAME = 'Broadlink switch'
|
|
|
|
DEFAULT_TIMEOUT = 10
|
2017-02-06 06:53:58 +00:00
|
|
|
DEFAULT_RETRY = 3
|
2017-05-02 16:18:47 +00:00
|
|
|
SERVICE_LEARN = 'learn_command'
|
|
|
|
SERVICE_SEND = 'send_packet'
|
2016-12-16 05:42:00 +00:00
|
|
|
|
2017-05-02 16:18:47 +00:00
|
|
|
RM_TYPES = ['rm', 'rm2', 'rm_mini', 'rm_pro_phicomm', 'rm2_home_plus',
|
|
|
|
'rm2_home_plus_gdt', 'rm2_pro_plus', 'rm2_pro_plus2',
|
|
|
|
'rm2_pro_plus_bl', 'rm_mini_shate']
|
|
|
|
SP1_TYPES = ['sp1']
|
|
|
|
SP2_TYPES = ['sp2', 'honeywell_sp2', 'sp3', 'spmini2', 'spminiplus']
|
2017-01-03 22:45:11 +00:00
|
|
|
|
|
|
|
SWITCH_TYPES = RM_TYPES + SP1_TYPES + SP2_TYPES
|
2016-12-19 01:59:08 +00:00
|
|
|
|
2016-12-16 05:42:00 +00:00
|
|
|
SWITCH_SCHEMA = vol.Schema({
|
|
|
|
vol.Optional(CONF_COMMAND_OFF, default=None): cv.string,
|
|
|
|
vol.Optional(CONF_COMMAND_ON, default=None): cv.string,
|
2017-06-02 05:58:57 +00:00
|
|
|
vol.Optional(CONF_FRIENDLY_NAME): cv.string,
|
2016-12-16 05:42:00 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
2017-01-03 22:45:11 +00:00
|
|
|
vol.Optional(CONF_SWITCHES, default={}):
|
|
|
|
vol.Schema({cv.slug: SWITCH_SCHEMA}),
|
2016-12-16 05:42:00 +00:00
|
|
|
vol.Required(CONF_HOST): cv.string,
|
|
|
|
vol.Required(CONF_MAC): cv.string,
|
2017-01-03 22:45:11 +00:00
|
|
|
vol.Optional(CONF_FRIENDLY_NAME, default=DEFAULT_NAME): cv.string,
|
|
|
|
vol.Optional(CONF_TYPE, default=SWITCH_TYPES[0]): vol.In(SWITCH_TYPES),
|
2016-12-16 05:42:00 +00:00
|
|
|
vol.Optional(CONF_TIMEOUT, default=DEFAULT_TIMEOUT): cv.positive_int
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
def setup_platform(hass, config, add_devices, discovery_info=None):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Set up Broadlink switches."""
|
2016-12-16 05:42:00 +00:00
|
|
|
import broadlink
|
|
|
|
devices = config.get(CONF_SWITCHES, {})
|
|
|
|
ip_addr = config.get(CONF_HOST)
|
2017-01-03 22:45:11 +00:00
|
|
|
friendly_name = config.get(CONF_FRIENDLY_NAME)
|
2016-12-16 05:42:00 +00:00
|
|
|
mac_addr = binascii.unhexlify(
|
|
|
|
config.get(CONF_MAC).encode().replace(b':', b''))
|
2017-01-03 22:45:11 +00:00
|
|
|
switch_type = config.get(CONF_TYPE)
|
2016-12-19 01:59:08 +00:00
|
|
|
|
2016-12-16 05:42:00 +00:00
|
|
|
@asyncio.coroutine
|
|
|
|
def _learn_command(call):
|
|
|
|
try:
|
2017-05-26 15:28:07 +00:00
|
|
|
auth = yield from hass.async_add_job(broadlink_device.auth)
|
2016-12-16 05:42:00 +00:00
|
|
|
except socket.timeout:
|
2017-05-02 16:18:47 +00:00
|
|
|
_LOGGER.error("Failed to connect to device, timeout")
|
2017-01-03 22:45:11 +00:00
|
|
|
return
|
|
|
|
if not auth:
|
2017-05-02 16:18:47 +00:00
|
|
|
_LOGGER.error("Failed to connect to device")
|
2016-12-16 05:42:00 +00:00
|
|
|
return
|
2017-01-03 22:45:11 +00:00
|
|
|
|
2017-05-26 15:28:07 +00:00
|
|
|
yield from hass.async_add_job(broadlink_device.enter_learning)
|
2016-12-16 05:42:00 +00:00
|
|
|
|
|
|
|
_LOGGER.info("Press the key you want HASS to learn")
|
|
|
|
start_time = utcnow()
|
|
|
|
while (utcnow() - start_time) < timedelta(seconds=20):
|
2017-05-26 15:28:07 +00:00
|
|
|
packet = yield from hass.async_add_job(
|
|
|
|
broadlink_device.check_data)
|
2016-12-16 05:42:00 +00:00
|
|
|
if packet:
|
2017-05-02 16:18:47 +00:00
|
|
|
log_msg = "Recieved packet is: {}".\
|
2016-12-16 05:42:00 +00:00
|
|
|
format(b64encode(packet).decode('utf8'))
|
|
|
|
_LOGGER.info(log_msg)
|
2017-07-16 19:39:38 +00:00
|
|
|
hass.components.persistent_notification.async_create(
|
|
|
|
log_msg, title='Broadlink switch')
|
2016-12-16 05:42:00 +00:00
|
|
|
return
|
|
|
|
yield from asyncio.sleep(1, loop=hass.loop)
|
2017-05-02 16:18:47 +00:00
|
|
|
_LOGGER.error("Did not received any signal")
|
2017-07-16 19:39:38 +00:00
|
|
|
hass.components.persistent_notification.async_create(
|
|
|
|
"Did not received any signal", title='Broadlink switch')
|
2016-12-20 20:05:54 +00:00
|
|
|
|
2017-02-06 06:53:58 +00:00
|
|
|
@asyncio.coroutine
|
|
|
|
def _send_packet(call):
|
|
|
|
packets = call.data.get('packet', [])
|
|
|
|
for packet in packets:
|
|
|
|
for retry in range(DEFAULT_RETRY):
|
|
|
|
try:
|
|
|
|
payload = b64decode(packet)
|
2017-05-26 15:28:07 +00:00
|
|
|
yield from hass.async_add_job(
|
|
|
|
broadlink_device.send_data, payload)
|
2017-02-06 06:53:58 +00:00
|
|
|
break
|
|
|
|
except (socket.timeout, ValueError):
|
|
|
|
try:
|
2017-05-26 15:28:07 +00:00
|
|
|
yield from hass.async_add_job(
|
|
|
|
broadlink_device.auth)
|
2017-02-06 06:53:58 +00:00
|
|
|
except socket.timeout:
|
|
|
|
if retry == DEFAULT_RETRY-1:
|
2017-05-02 16:18:47 +00:00
|
|
|
_LOGGER.error("Failed to send packet to device")
|
2017-02-06 06:53:58 +00:00
|
|
|
|
2017-01-03 22:45:11 +00:00
|
|
|
if switch_type in RM_TYPES:
|
2016-12-20 20:05:54 +00:00
|
|
|
broadlink_device = broadlink.rm((ip_addr, 80), mac_addr)
|
2017-02-06 06:53:58 +00:00
|
|
|
hass.services.register(DOMAIN, SERVICE_LEARN + '_' +
|
|
|
|
ip_addr.replace('.', '_'), _learn_command)
|
|
|
|
hass.services.register(DOMAIN, SERVICE_SEND + '_' +
|
|
|
|
ip_addr.replace('.', '_'), _send_packet)
|
2017-01-03 22:45:11 +00:00
|
|
|
switches = []
|
|
|
|
for object_id, device_config in devices.items():
|
|
|
|
switches.append(
|
|
|
|
BroadlinkRMSwitch(
|
|
|
|
device_config.get(CONF_FRIENDLY_NAME, object_id),
|
|
|
|
broadlink_device,
|
|
|
|
device_config.get(CONF_COMMAND_ON),
|
|
|
|
device_config.get(CONF_COMMAND_OFF)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
elif switch_type in SP1_TYPES:
|
2016-12-20 20:05:54 +00:00
|
|
|
broadlink_device = broadlink.sp1((ip_addr, 80), mac_addr)
|
2017-01-03 22:45:11 +00:00
|
|
|
switches = [BroadlinkSP1Switch(friendly_name, broadlink_device)]
|
|
|
|
elif switch_type in SP2_TYPES:
|
2016-12-20 20:05:54 +00:00
|
|
|
broadlink_device = broadlink.sp2((ip_addr, 80), mac_addr)
|
2017-01-03 22:45:11 +00:00
|
|
|
switches = [BroadlinkSP2Switch(friendly_name, broadlink_device)]
|
2016-12-20 20:05:54 +00:00
|
|
|
|
|
|
|
broadlink_device.timeout = config.get(CONF_TIMEOUT)
|
|
|
|
try:
|
|
|
|
broadlink_device.auth()
|
|
|
|
except socket.timeout:
|
2017-05-02 16:18:47 +00:00
|
|
|
_LOGGER.error("Failed to connect to device")
|
2016-12-16 05:42:00 +00:00
|
|
|
|
|
|
|
add_devices(switches)
|
|
|
|
|
|
|
|
|
2016-12-19 01:59:08 +00:00
|
|
|
class BroadlinkRMSwitch(SwitchDevice):
|
2016-12-16 05:42:00 +00:00
|
|
|
"""Representation of an Broadlink switch."""
|
|
|
|
|
2017-01-03 22:45:11 +00:00
|
|
|
def __init__(self, friendly_name, device, command_on, command_off):
|
2016-12-16 05:42:00 +00:00
|
|
|
"""Initialize the switch."""
|
|
|
|
self._name = friendly_name
|
|
|
|
self._state = False
|
|
|
|
self._command_on = b64decode(command_on) if command_on else None
|
|
|
|
self._command_off = b64decode(command_off) if command_off else None
|
|
|
|
self._device = device
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the switch."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def assumed_state(self):
|
|
|
|
"""Return true if unable to access real state of entity."""
|
|
|
|
return True
|
|
|
|
|
2016-12-19 01:59:08 +00:00
|
|
|
@property
|
|
|
|
def should_poll(self):
|
2017-05-02 20:47:20 +00:00
|
|
|
"""Return the polling state."""
|
2016-12-19 01:59:08 +00:00
|
|
|
return False
|
|
|
|
|
2016-12-16 05:42:00 +00:00
|
|
|
@property
|
|
|
|
def is_on(self):
|
|
|
|
"""Return true if device is on."""
|
|
|
|
return self._state
|
|
|
|
|
|
|
|
def turn_on(self, **kwargs):
|
|
|
|
"""Turn the device on."""
|
|
|
|
if self._sendpacket(self._command_on):
|
|
|
|
self._state = True
|
2017-03-04 23:10:36 +00:00
|
|
|
self.schedule_update_ha_state()
|
2016-12-16 05:42:00 +00:00
|
|
|
|
|
|
|
def turn_off(self, **kwargs):
|
|
|
|
"""Turn the device off."""
|
|
|
|
if self._sendpacket(self._command_off):
|
|
|
|
self._state = False
|
2017-03-04 23:10:36 +00:00
|
|
|
self.schedule_update_ha_state()
|
2016-12-16 05:42:00 +00:00
|
|
|
|
|
|
|
def _sendpacket(self, packet, retry=2):
|
|
|
|
"""Send packet to device."""
|
|
|
|
if packet is None:
|
2017-05-02 16:18:47 +00:00
|
|
|
_LOGGER.debug("Empty packet")
|
2016-12-16 05:42:00 +00:00
|
|
|
return True
|
|
|
|
try:
|
|
|
|
self._device.send_data(packet)
|
2016-12-20 20:16:18 +00:00
|
|
|
except (socket.timeout, ValueError) as error:
|
2016-12-16 05:42:00 +00:00
|
|
|
if retry < 1:
|
|
|
|
_LOGGER.error(error)
|
|
|
|
return False
|
2017-01-03 22:45:11 +00:00
|
|
|
if not self._auth():
|
|
|
|
return False
|
2017-02-22 21:11:49 +00:00
|
|
|
return self._sendpacket(packet, retry-1)
|
2016-12-16 05:42:00 +00:00
|
|
|
return True
|
2016-12-19 01:59:08 +00:00
|
|
|
|
2017-01-03 22:45:11 +00:00
|
|
|
def _auth(self, retry=2):
|
|
|
|
try:
|
|
|
|
auth = self._device.auth()
|
|
|
|
except socket.timeout:
|
|
|
|
auth = False
|
|
|
|
if not auth and retry > 0:
|
2017-02-22 21:11:49 +00:00
|
|
|
return self._auth(retry-1)
|
2017-01-03 22:45:11 +00:00
|
|
|
return auth
|
|
|
|
|
2016-12-19 01:59:08 +00:00
|
|
|
|
|
|
|
class BroadlinkSP1Switch(BroadlinkRMSwitch):
|
|
|
|
"""Representation of an Broadlink switch."""
|
|
|
|
|
2017-01-03 22:45:11 +00:00
|
|
|
def __init__(self, friendly_name, device):
|
2016-12-19 01:59:08 +00:00
|
|
|
"""Initialize the switch."""
|
2017-01-03 22:45:11 +00:00
|
|
|
super().__init__(friendly_name, device, None, None)
|
2016-12-19 01:59:08 +00:00
|
|
|
self._command_on = 1
|
|
|
|
self._command_off = 0
|
|
|
|
|
|
|
|
def _sendpacket(self, packet, retry=2):
|
|
|
|
"""Send packet to device."""
|
|
|
|
try:
|
|
|
|
self._device.set_power(packet)
|
2016-12-20 20:05:54 +00:00
|
|
|
except (socket.timeout, ValueError) as error:
|
2016-12-19 01:59:08 +00:00
|
|
|
if retry < 1:
|
|
|
|
_LOGGER.error(error)
|
|
|
|
return False
|
2017-01-03 22:45:11 +00:00
|
|
|
if not self._auth():
|
|
|
|
return False
|
2017-02-22 21:11:49 +00:00
|
|
|
return self._sendpacket(packet, retry-1)
|
2016-12-19 01:59:08 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
class BroadlinkSP2Switch(BroadlinkSP1Switch):
|
|
|
|
"""Representation of an Broadlink switch."""
|
|
|
|
|
|
|
|
@property
|
|
|
|
def assumed_state(self):
|
|
|
|
"""Return true if unable to access real state of entity."""
|
|
|
|
return False
|
|
|
|
|
|
|
|
@property
|
|
|
|
def should_poll(self):
|
2017-05-02 20:47:20 +00:00
|
|
|
"""Return the polling state."""
|
2016-12-19 01:59:08 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
def update(self):
|
|
|
|
"""Synchronize state with switch."""
|
|
|
|
self._update()
|
|
|
|
|
|
|
|
def _update(self, retry=2):
|
|
|
|
try:
|
|
|
|
state = self._device.check_power()
|
2016-12-20 20:05:54 +00:00
|
|
|
except (socket.timeout, ValueError) as error:
|
2016-12-19 01:59:08 +00:00
|
|
|
if retry < 1:
|
|
|
|
_LOGGER.error(error)
|
|
|
|
return
|
2017-01-03 22:45:11 +00:00
|
|
|
if not self._auth():
|
|
|
|
return
|
2017-02-22 21:11:49 +00:00
|
|
|
return self._update(retry-1)
|
2016-12-19 01:59:08 +00:00
|
|
|
if state is None and retry > 0:
|
2017-02-22 21:11:49 +00:00
|
|
|
return self._update(retry-1)
|
2016-12-19 01:59:08 +00:00
|
|
|
self._state = state
|