mirror of
https://github.com/mjg59/python-broadlink.git
synced 2024-11-14 19:40:11 +01:00
247be74c33
* Move IR duration<->Broadlink conversion down from CLI * Fix --learn base64 to not crash with --durations Also remove its b'...' wrapping. * Fix IR/RF conversions --------- Co-authored-by: William Grant <me@williamgrant.id.au>
222 lines
7.1 KiB
Python
Executable File
222 lines
7.1 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
import argparse
|
|
import base64
|
|
import time
|
|
import typing as t
|
|
|
|
import broadlink
|
|
from broadlink.const import DEFAULT_PORT
|
|
from broadlink.exceptions import ReadError, StorageError
|
|
from broadlink.remote import data_to_pulses, pulses_to_data
|
|
|
|
TIMEOUT = 30
|
|
|
|
|
|
def auto_int(x):
|
|
return int(x, 0)
|
|
|
|
|
|
def format_pulses(pulses: t.List[int]) -> str:
|
|
"""Format pulses."""
|
|
return " ".join(
|
|
f"+{pulse}" if i % 2 == 0 else f"-{pulse}"
|
|
for i, pulse in enumerate(pulses)
|
|
)
|
|
|
|
|
|
def parse_pulses(data: t.List[str]) -> t.List[int]:
|
|
"""Parse pulses."""
|
|
return [abs(int(s)) for s in data]
|
|
|
|
|
|
parser = argparse.ArgumentParser(fromfile_prefix_chars='@')
|
|
parser.add_argument("--device", help="device definition as 'type host mac'")
|
|
parser.add_argument("--type", type=auto_int, default=0x2712, help="type of device")
|
|
parser.add_argument("--host", help="host address")
|
|
parser.add_argument("--mac", help="mac address (hex reverse), as used by python-broadlink library")
|
|
parser.add_argument("--temperature", action="store_true", help="request temperature from device")
|
|
parser.add_argument("--humidity", action="store_true", help="request humidity from device")
|
|
parser.add_argument("--energy", action="store_true", help="request energy consumption from device")
|
|
parser.add_argument("--check", action="store_true", help="check current power state")
|
|
parser.add_argument("--checknl", action="store_true", help="check current nightlight state")
|
|
parser.add_argument("--turnon", action="store_true", help="turn on device")
|
|
parser.add_argument("--turnoff", action="store_true", help="turn off device")
|
|
parser.add_argument("--turnnlon", action="store_true", help="turn on nightlight on the device")
|
|
parser.add_argument("--turnnloff", action="store_true", help="turn off nightlight on the device")
|
|
parser.add_argument("--switch", action="store_true", help="switch state from on to off and off to on")
|
|
parser.add_argument("--send", action="store_true", help="send command")
|
|
parser.add_argument("--sensors", action="store_true", help="check all sensors")
|
|
parser.add_argument("--learn", action="store_true", help="learn command")
|
|
parser.add_argument("--rflearn", action="store_true", help="rf scan learning")
|
|
parser.add_argument("--frequency", type=float, help="specify radiofrequency for learning")
|
|
parser.add_argument("--learnfile", help="save learned command to a specified file")
|
|
parser.add_argument("--durations", action="store_true",
|
|
help="use durations in micro seconds instead of the Broadlink format")
|
|
parser.add_argument("--convert", action="store_true", help="convert input data to durations")
|
|
parser.add_argument("--joinwifi", nargs=2, help="Args are SSID PASSPHRASE to configure Broadlink device with")
|
|
parser.add_argument("data", nargs='*', help="Data to send or convert")
|
|
args = parser.parse_args()
|
|
|
|
if args.device:
|
|
values = args.device.split()
|
|
devtype = int(values[0], 0)
|
|
host = values[1]
|
|
mac = bytearray.fromhex(values[2])
|
|
elif args.mac:
|
|
devtype = args.type
|
|
host = args.host
|
|
mac = bytearray.fromhex(args.mac)
|
|
|
|
if args.host or args.device:
|
|
dev = broadlink.gendevice(devtype, (host, DEFAULT_PORT), mac)
|
|
dev.auth()
|
|
|
|
if args.joinwifi:
|
|
broadlink.setup(args.joinwifi[0], args.joinwifi[1], 4)
|
|
|
|
if args.convert:
|
|
data = bytearray.fromhex(''.join(args.data))
|
|
pulses = data_to_pulses(data)
|
|
print(format_pulses(pulses))
|
|
if args.temperature:
|
|
print(dev.check_temperature())
|
|
if args.humidity:
|
|
print(dev.check_humidity())
|
|
if args.energy:
|
|
print(dev.get_energy())
|
|
if args.sensors:
|
|
data = dev.check_sensors()
|
|
for key in data:
|
|
print("{} {}".format(key, data[key]))
|
|
if args.send:
|
|
data = (
|
|
pulses_to_data(parse_pulses(args.data))
|
|
if args.durations
|
|
else bytes.fromhex(''.join(args.data))
|
|
)
|
|
dev.send_data(data)
|
|
if args.learn or (args.learnfile and not args.rflearn):
|
|
dev.enter_learning()
|
|
print("Learning...")
|
|
start = time.time()
|
|
while time.time() - start < TIMEOUT:
|
|
time.sleep(1)
|
|
try:
|
|
data = dev.check_data()
|
|
except (ReadError, StorageError):
|
|
continue
|
|
else:
|
|
break
|
|
else:
|
|
print("No data received...")
|
|
exit(1)
|
|
|
|
print("Packet found!")
|
|
raw_fmt = data.hex()
|
|
base64_fmt = base64.b64encode(data).decode('ascii')
|
|
pulse_fmt = format_pulses(data_to_pulses(data))
|
|
|
|
print("Raw:", raw_fmt)
|
|
print("Base64:", base64_fmt)
|
|
print("Pulses:", pulse_fmt)
|
|
|
|
if args.learnfile:
|
|
print("Saving to {}".format(args.learnfile))
|
|
with open(args.learnfile, "w") as text_file:
|
|
text_file.write(pulse_fmt if args.durations else raw_fmt)
|
|
if args.check:
|
|
if dev.check_power():
|
|
print('* ON *')
|
|
else:
|
|
print('* OFF *')
|
|
if args.checknl:
|
|
if dev.check_nightlight():
|
|
print('* ON *')
|
|
else:
|
|
print('* OFF *')
|
|
if args.turnon:
|
|
dev.set_power(True)
|
|
if dev.check_power():
|
|
print('== Turned * ON * ==')
|
|
else:
|
|
print('!! Still OFF !!')
|
|
if args.turnoff:
|
|
dev.set_power(False)
|
|
if dev.check_power():
|
|
print('!! Still ON !!')
|
|
else:
|
|
print('== Turned * OFF * ==')
|
|
if args.turnnlon:
|
|
dev.set_nightlight(True)
|
|
if dev.check_nightlight():
|
|
print('== Turned * ON * ==')
|
|
else:
|
|
print('!! Still OFF !!')
|
|
if args.turnnloff:
|
|
dev.set_nightlight(False)
|
|
if dev.check_nightlight():
|
|
print('!! Still ON !!')
|
|
else:
|
|
print('== Turned * OFF * ==')
|
|
if args.switch:
|
|
if dev.check_power():
|
|
dev.set_power(False)
|
|
print('* Switch to OFF *')
|
|
else:
|
|
dev.set_power(True)
|
|
print('* Switch to ON *')
|
|
if args.rflearn:
|
|
if args.frequency:
|
|
frequency = args.frequency
|
|
print("Press the button you want to learn, a short press...")
|
|
else:
|
|
dev.sweep_frequency()
|
|
print("Detecting radiofrequency, press and hold the button to learn...")
|
|
|
|
start = time.time()
|
|
while time.time() - start < TIMEOUT:
|
|
time.sleep(1)
|
|
locked, frequency = dev.check_frequency()
|
|
if locked:
|
|
break
|
|
else:
|
|
print("Radiofrequency not found")
|
|
dev.cancel_sweep_frequency()
|
|
exit(1)
|
|
|
|
print("Radiofrequency detected: {}MHz".format(frequency))
|
|
print("You can now let go of the button")
|
|
|
|
input("Press enter to continue...")
|
|
|
|
print("Press the button again, now a short press.")
|
|
|
|
dev.find_rf_packet(frequency)
|
|
|
|
start = time.time()
|
|
while time.time() - start < TIMEOUT:
|
|
time.sleep(1)
|
|
try:
|
|
data = dev.check_data()
|
|
except (ReadError, StorageError):
|
|
continue
|
|
else:
|
|
break
|
|
else:
|
|
print("No data received...")
|
|
exit(1)
|
|
|
|
print("Packet found!")
|
|
raw_fmt = data.hex()
|
|
base64_fmt = base64.b64encode(data).decode('ascii')
|
|
pulse_fmt = format_pulses(data_to_pulses(data))
|
|
|
|
print("Raw:", raw_fmt)
|
|
print("Base64:", base64_fmt)
|
|
print("Pulses:", pulse_fmt)
|
|
|
|
if args.learnfile:
|
|
print("Saving to {}".format(args.learnfile))
|
|
with open(args.learnfile, "w") as text_file:
|
|
text_file.write(pulse_fmt if args.durations else raw_fmt)
|