2020-10-21 19:32:32 +02:00
|
|
|
#!/usr/bin/env python3
|
2020-09-16 09:41:28 +02:00
|
|
|
"""The python-broadlink library."""
|
2019-05-27 20:57:32 +02:00
|
|
|
import socket
|
2020-09-24 07:36:12 +02:00
|
|
|
from typing import Generator, List, Union, Tuple
|
2019-05-19 17:54:14 +02:00
|
|
|
|
2020-09-17 05:41:32 +02:00
|
|
|
from .alarm import S1C
|
|
|
|
from .climate import hysen
|
2020-09-20 06:34:31 +02:00
|
|
|
from .cover import dooya
|
2020-09-23 07:43:56 +02:00
|
|
|
from .device import device, scan
|
|
|
|
from .exceptions import exception
|
2020-09-17 05:41:32 +02:00
|
|
|
from .light import lb1
|
2020-09-24 07:36:12 +02:00
|
|
|
from .remote import rm, rm4
|
2020-09-17 05:41:32 +02:00
|
|
|
from .sensor import a1
|
2020-11-04 03:15:34 +01:00
|
|
|
from .switch import bg1, mp1, sp1, sp2, sp4, sp4b
|
2018-04-30 23:06:19 +02:00
|
|
|
|
2020-06-16 21:19:32 +02:00
|
|
|
|
2020-09-24 07:36:12 +02:00
|
|
|
SUPPORTED_TYPES = {
|
|
|
|
0x0000: (sp1, "SP1", "Broadlink"),
|
|
|
|
0x2711: (sp2, "SP2", "Broadlink"),
|
|
|
|
0x2716: (sp2, "NEO PRO", "Ankuoo"),
|
|
|
|
0x2717: (sp2, "NEO", "Ankuoo"),
|
|
|
|
0x2719: (sp2, "SP2-compatible", "Honeywell"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x271A: (sp2, "SP2-compatible", "Honeywell"),
|
2020-11-25 08:26:32 +01:00
|
|
|
0x271D: (sp2, "Ego", "Efergy"),
|
2020-09-24 07:36:12 +02:00
|
|
|
0x2720: (sp2, "SP mini", "Broadlink"),
|
|
|
|
0x2728: (sp2, "SP2-compatible", "URANT"),
|
|
|
|
0x2733: (sp2, "SP3", "Broadlink"),
|
|
|
|
0x2736: (sp2, "SP mini+", "Broadlink"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x273E: (sp2, "SP mini", "Broadlink"),
|
2020-09-24 07:36:12 +02:00
|
|
|
0x7530: (sp2, "SP2", "Broadlink (OEM)"),
|
|
|
|
0x7539: (sp2, "SP2-IL", "Broadlink (OEM)"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x753E: (sp2, "SP mini 3", "Broadlink"),
|
2020-09-24 07:36:12 +02:00
|
|
|
0x7540: (sp2, "MP2", "Broadlink"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x7544: (sp2, "SP2-CL", "Broadlink"),
|
2020-09-24 07:36:12 +02:00
|
|
|
0x7546: (sp2, "SP2-UK/BR/IN", "Broadlink (OEM)"),
|
|
|
|
0x7547: (sp2, "SC1", "Broadlink"),
|
|
|
|
0x7918: (sp2, "SP2", "Broadlink (OEM)"),
|
|
|
|
0x7919: (sp2, "SP2-compatible", "Honeywell"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x791A: (sp2, "SP2-compatible", "Honeywell"),
|
|
|
|
0x7D00: (sp2, "SP3-EU", "Broadlink (OEM)"),
|
|
|
|
0x7D0D: (sp2, "SP mini 3", "Broadlink (OEM)"),
|
2020-09-24 07:36:12 +02:00
|
|
|
0x9479: (sp2, "SP3S-US", "Broadlink"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x947A: (sp2, "SP3S-EU", "Broadlink"),
|
|
|
|
0x756C: (sp4, "SP4M", "Broadlink"),
|
2020-11-22 19:19:21 +01:00
|
|
|
0x756F: (sp4, "MCB1", "Broadlink"),
|
2020-11-04 03:15:34 +01:00
|
|
|
0x7579: (sp4, "SP4L-EU", "Broadlink"),
|
|
|
|
0x7583: (sp4, "SP mini 3", "Broadlink"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x7D11: (sp4, "SP mini 3", "Broadlink"),
|
2020-12-04 00:02:11 +01:00
|
|
|
0xA56A: (sp4, "MCB1", "Broadlink"),
|
2020-12-08 20:54:11 +01:00
|
|
|
0xA589: (sp4, "SP4L-UK", "Broadlink"),
|
2020-12-04 01:24:47 +01:00
|
|
|
0x6113: (sp4b, "SCB1E", "Broadlink"),
|
2020-12-04 16:42:16 +01:00
|
|
|
0x618B: (sp4b, "SP4L-EU", "Broadlink"),
|
2020-12-27 05:42:48 +01:00
|
|
|
0x6489: (sp4b, "SP4L-AU", "Broadlink"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x648B: (sp4b, "SP4M-US", "Broadlink"),
|
2020-09-24 07:36:12 +02:00
|
|
|
0x2712: (rm, "RM pro/pro+", "Broadlink"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x272A: (rm, "RM pro", "Broadlink"),
|
2020-09-24 07:36:12 +02:00
|
|
|
0x2737: (rm, "RM mini 3", "Broadlink"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x273D: (rm, "RM pro", "Broadlink"),
|
|
|
|
0x277C: (rm, "RM home", "Broadlink"),
|
2020-09-24 07:36:12 +02:00
|
|
|
0x2783: (rm, "RM home", "Broadlink"),
|
|
|
|
0x2787: (rm, "RM pro", "Broadlink"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x278B: (rm, "RM plus", "Broadlink"),
|
|
|
|
0x278F: (rm, "RM mini", "Broadlink"),
|
2020-09-24 07:36:12 +02:00
|
|
|
0x2797: (rm, "RM pro+", "Broadlink"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x279D: (rm, "RM pro+", "Broadlink"),
|
|
|
|
0x27A1: (rm, "RM plus", "Broadlink"),
|
|
|
|
0x27A6: (rm, "RM plus", "Broadlink"),
|
|
|
|
0x27A9: (rm, "RM pro+", "Broadlink"),
|
|
|
|
0x27C2: (rm, "RM mini 3", "Broadlink"),
|
|
|
|
0x27C3: (rm, "RM pro+", "Broadlink"),
|
|
|
|
0x27C7: (rm, "RM mini 3", "Broadlink"),
|
|
|
|
0x27CC: (rm, "RM mini 3", "Broadlink"),
|
|
|
|
0x27CD: (rm, "RM mini 3", "Broadlink"),
|
|
|
|
0x27D0: (rm, "RM mini 3", "Broadlink"),
|
|
|
|
0x27D1: (rm, "RM mini 3", "Broadlink"),
|
2020-12-09 15:23:17 +01:00
|
|
|
0x27D3: (rm, "RM mini 3", "Broadlink"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x27DE: (rm, "RM mini 3", "Broadlink"),
|
|
|
|
0x51DA: (rm4, "RM4 mini", "Broadlink"),
|
|
|
|
0x5F36: (rm4, "RM mini 3", "Broadlink"),
|
2020-09-24 07:36:12 +02:00
|
|
|
0x6026: (rm4, "RM4 pro", "Broadlink"),
|
|
|
|
0x6070: (rm4, "RM4C mini", "Broadlink"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x610E: (rm4, "RM4 mini", "Broadlink"),
|
|
|
|
0x610F: (rm4, "RM4C mini", "Broadlink"),
|
|
|
|
0x61A2: (rm4, "RM4 pro", "Broadlink"),
|
|
|
|
0x62BC: (rm4, "RM4 mini", "Broadlink"),
|
|
|
|
0x62BE: (rm4, "RM4C mini", "Broadlink"),
|
2020-11-14 19:02:31 +01:00
|
|
|
0x6364: (rm4, "RM4S", "Broadlink"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x648D: (rm4, "RM4 mini", "Broadlink"),
|
|
|
|
0x649B: (rm4, "RM4 pro", "Broadlink"),
|
2020-11-23 17:38:02 +01:00
|
|
|
0x6508: (rm4, "RM mini 3", "Broadlink"),
|
2020-11-16 21:04:17 +01:00
|
|
|
0x6539: (rm4, "RM4C mini", "Broadlink"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x653A: (rm4, "RM4 mini", "Broadlink"),
|
2020-11-09 01:12:59 +01:00
|
|
|
0x653C: (rm4, "RM4 pro", "Broadlink"),
|
2020-09-24 07:36:12 +02:00
|
|
|
0x2714: (a1, "e-Sensor", "Broadlink"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x4EB5: (mp1, "MP1-1K4S", "Broadlink"),
|
|
|
|
0x4EF7: (mp1, "MP1-1K4S", "Broadlink (OEM)"),
|
|
|
|
0x4F1B: (mp1, "MP1-1K3S2U", "Broadlink (OEM)"),
|
|
|
|
0x4F65: (mp1, "MP1-1K3S2U", "Broadlink"),
|
2020-09-24 07:36:12 +02:00
|
|
|
0x5043: (lb1, "SB800TD", "Broadlink (OEM)"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x504E: (lb1, "LB1", "Broadlink"),
|
|
|
|
0x60C7: (lb1, "LB1", "Broadlink"),
|
|
|
|
0x60C8: (lb1, "LB1", "Broadlink"),
|
2020-09-24 07:36:12 +02:00
|
|
|
0x6112: (lb1, "LB1", "Broadlink"),
|
|
|
|
0x2722: (S1C, "S2KIT", "Broadlink"),
|
2020-11-05 20:25:33 +01:00
|
|
|
0x4EAD: (hysen, "HY02B05H", "Hysen"),
|
|
|
|
0x4E4D: (dooya, "DT360E-45/20", "Dooya"),
|
|
|
|
0x51E3: (bg1, "BG800/BG900", "BG Electrical"),
|
2020-09-24 07:36:12 +02:00
|
|
|
}
|
2019-05-19 17:54:14 +02:00
|
|
|
|
2020-07-19 02:53:00 +02:00
|
|
|
|
2020-09-17 02:35:09 +02:00
|
|
|
def gendevice(
|
2020-11-05 20:25:33 +01:00
|
|
|
dev_type: int,
|
|
|
|
host: Tuple[str, int],
|
|
|
|
mac: Union[bytes, str],
|
|
|
|
name: str = None,
|
|
|
|
is_locked: bool = None,
|
2020-09-17 08:19:24 +02:00
|
|
|
) -> device:
|
2020-07-19 02:53:00 +02:00
|
|
|
"""Generate a device."""
|
2020-06-16 21:19:32 +02:00
|
|
|
try:
|
2020-09-24 07:36:12 +02:00
|
|
|
dev_class, model, manufacturer = SUPPORTED_TYPES[dev_type]
|
2020-07-19 02:53:00 +02:00
|
|
|
|
2020-06-16 21:19:32 +02:00
|
|
|
except KeyError:
|
2020-07-31 07:10:21 +02:00
|
|
|
return device(host, mac, dev_type, name=name, is_locked=is_locked)
|
2020-06-16 21:19:32 +02:00
|
|
|
|
2020-07-19 02:53:00 +02:00
|
|
|
return dev_class(
|
|
|
|
host,
|
|
|
|
mac,
|
|
|
|
dev_type,
|
|
|
|
name=name,
|
|
|
|
model=model,
|
|
|
|
manufacturer=manufacturer,
|
2020-07-31 07:10:21 +02:00
|
|
|
is_locked=is_locked,
|
2020-07-19 02:53:00 +02:00
|
|
|
)
|
2019-05-19 17:54:14 +02:00
|
|
|
|
2016-11-19 23:22:08 +01:00
|
|
|
|
2020-09-23 07:43:56 +02:00
|
|
|
def hello(
|
2020-11-05 20:25:33 +01:00
|
|
|
host: str,
|
|
|
|
port: int = 80,
|
|
|
|
timeout: int = 10,
|
|
|
|
local_ip_address: str = None,
|
2020-09-23 07:43:56 +02:00
|
|
|
) -> device:
|
|
|
|
"""Direct device discovery.
|
|
|
|
|
|
|
|
Useful if the device is locked.
|
|
|
|
"""
|
|
|
|
try:
|
|
|
|
return next(xdiscover(timeout, local_ip_address, host, port))
|
|
|
|
except StopIteration:
|
|
|
|
raise exception(-4000) # Network timeout.
|
|
|
|
|
|
|
|
|
2020-08-10 21:28:09 +02:00
|
|
|
def discover(
|
2020-11-05 20:25:33 +01:00
|
|
|
timeout: int = 10,
|
|
|
|
local_ip_address: str = None,
|
|
|
|
discover_ip_address: str = "255.255.255.255",
|
|
|
|
discover_ip_port: int = 80,
|
2020-09-17 08:19:24 +02:00
|
|
|
) -> List[device]:
|
2020-09-16 09:41:28 +02:00
|
|
|
"""Discover devices connected to the local network."""
|
2020-11-05 20:25:33 +01:00
|
|
|
responses = scan(timeout, local_ip_address, discover_ip_address, discover_ip_port)
|
2020-09-23 07:43:56 +02:00
|
|
|
return [gendevice(*resp) for resp in responses]
|
|
|
|
|
|
|
|
|
|
|
|
def xdiscover(
|
2020-11-05 20:25:33 +01:00
|
|
|
timeout: int = 10,
|
|
|
|
local_ip_address: str = None,
|
|
|
|
discover_ip_address: str = "255.255.255.255",
|
|
|
|
discover_ip_port: int = 80,
|
2020-09-23 07:43:56 +02:00
|
|
|
) -> Generator[device, None, None]:
|
|
|
|
"""Discover devices connected to the local network.
|
|
|
|
|
|
|
|
This function returns a generator that yields devices instantly.
|
|
|
|
"""
|
2020-11-05 20:25:33 +01:00
|
|
|
responses = scan(timeout, local_ip_address, discover_ip_address, discover_ip_port)
|
2020-09-23 07:43:56 +02:00
|
|
|
for resp in responses:
|
|
|
|
yield gendevice(*resp)
|
2019-05-19 17:54:14 +02:00
|
|
|
|
|
|
|
|
2017-04-22 21:48:02 +02:00
|
|
|
# Setup a new Broadlink device via AP Mode. Review the README to see how to enter AP Mode.
|
|
|
|
# Only tested with Broadlink RM3 Mini (Blackbean)
|
2020-09-17 02:35:09 +02:00
|
|
|
def setup(ssid: str, password: str, security_mode: int) -> None:
|
2020-09-16 09:41:28 +02:00
|
|
|
"""Set up a new Broadlink device via AP mode."""
|
2019-05-19 17:54:14 +02:00
|
|
|
# Security mode options are (0 - none, 1 = WEP, 2 = WPA1, 3 = WPA2, 4 = WPA1/2)
|
|
|
|
payload = bytearray(0x88)
|
|
|
|
payload[0x26] = 0x14 # This seems to always be set to 14
|
|
|
|
# Add the SSID to the payload
|
|
|
|
ssid_start = 68
|
|
|
|
ssid_length = 0
|
|
|
|
for letter in ssid:
|
|
|
|
payload[(ssid_start + ssid_length)] = ord(letter)
|
|
|
|
ssid_length += 1
|
|
|
|
# Add the WiFi password to the payload
|
|
|
|
pass_start = 100
|
|
|
|
pass_length = 0
|
|
|
|
for letter in password:
|
|
|
|
payload[(pass_start + pass_length)] = ord(letter)
|
|
|
|
pass_length += 1
|
|
|
|
|
|
|
|
payload[0x84] = ssid_length # Character length of SSID
|
|
|
|
payload[0x85] = pass_length # Character length of password
|
2020-09-24 07:36:12 +02:00
|
|
|
payload[0x86] = security_mode # Type of encryption
|
2019-05-19 17:54:14 +02:00
|
|
|
|
2020-11-05 20:25:33 +01:00
|
|
|
checksum = sum(payload, 0xBEAF) & 0xFFFF
|
|
|
|
payload[0x20] = checksum & 0xFF # Checksum 1 position
|
2019-05-19 17:54:14 +02:00
|
|
|
payload[0x21] = checksum >> 8 # Checksum 2 position
|
2017-04-22 21:48:02 +02:00
|
|
|
|
2020-11-05 20:25:33 +01:00
|
|
|
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) # Internet # UDP
|
2019-05-19 17:54:14 +02:00
|
|
|
sock.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)
|
|
|
|
sock.setsockopt(socket.SOL_SOCKET, socket.SO_BROADCAST, 1)
|
2020-11-05 20:25:33 +01:00
|
|
|
sock.sendto(payload, ("255.255.255.255", 80))
|
2020-04-25 11:40:48 +02:00
|
|
|
sock.close()
|