mirror of
https://github.com/python-kasa/python-kasa.git
synced 2025-01-10 06:47:06 +00:00
2d6376b597
* Add support for new-style protocol Newer devices (including my LB130) seem to include the request length in the previously empty message header, and ignore requests that lack it. They also don't send an empty packet as the final part of a response, which can lead to hangs. Add support for this, with luck not breaking existing devices in the process. * Fix tests We now include the request length in the encrypted packet header, so strip the header rather than assuming that it's just zeroes. * Create a SmartDevice parent class Add a generic SmartDevice class that SmartPlug can inherit from, in preparation for adding support for other device types. * Add support for TP-Link smartbulbs These bulbs use the same protocol as the smart plugs, but have additional commands for controlling bulb-specific features. In addition, the bulbs have their emeter under a different target and return responses that include the energy unit in the key names. * Add tests for bulbs Not entirely comprehensive, but has pretty much the same level of testing as plugs
255 lines
11 KiB
Python
255 lines
11 KiB
Python
from pyHS100.protocol import TPLinkSmartHomeProtocol
|
|
from pyHS100 import SmartPlugException
|
|
import logging
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
def get_realtime(obj, x):
|
|
return {"current":0.268587,"voltage":125.836131,"power":33.495623,"total":0.199000}
|
|
|
|
def get_monthstat(obj, x):
|
|
if x["year"] < 2016:
|
|
return {"month_list":[]}
|
|
|
|
return {"month_list": [{"year": 2016, "month": 11, "energy": 1.089000}, {"year": 2016, "month": 12, "energy": 1.582000}]}
|
|
|
|
def get_daystat(obj, x):
|
|
if x["year"] < 2016:
|
|
return {"day_list":[]}
|
|
|
|
return {"day_list": [{"year": 2016, "month": 11, "day": 24, "energy": 0.026000},
|
|
{"year": 2016, "month": 11, "day": 25, "energy": 0.109000}]}
|
|
|
|
emeter_support = {"get_realtime": get_realtime,
|
|
"get_monthstat": get_monthstat,
|
|
"get_daystat": get_daystat,}
|
|
|
|
def get_realtime_units(obj, x):
|
|
return {"power_mw": 10800}
|
|
|
|
def get_monthstat_units(obj, x):
|
|
if x["year"] < 2016:
|
|
return {"month_list":[]}
|
|
|
|
return {"month_list": [{"year": 2016, "month": 11, "energy_wh": 32}, {"year": 2016, "month": 12, "energy_wh": 16}]}
|
|
|
|
def get_daystat_units(obj, x):
|
|
if x["year"] < 2016:
|
|
return {"day_list":[]}
|
|
|
|
return {"day_list": [{"year": 2016, "month": 11, "day": 24, "energy_wh": 20},
|
|
{"year": 2016, "month": 11, "day": 25, "energy_wh": 32}]}
|
|
|
|
emeter_units_support = {"get_realtime": get_realtime_units,
|
|
"get_monthstat": get_monthstat_units,
|
|
"get_daystat": get_daystat_units,}
|
|
|
|
sysinfo_hs110 = {'system': {'get_sysinfo':
|
|
{'active_mode': 'schedule',
|
|
'alias': 'Mobile Plug',
|
|
'dev_name': 'Wi-Fi Smart Plug With Energy Monitoring',
|
|
'deviceId': '800654F32938FCBA8F7327887A386476172B5B53',
|
|
'err_code': 0,
|
|
'feature': 'TIM:ENE',
|
|
'fwId': 'E16EB3E95DB6B47B5B72B3FD86FD1438',
|
|
'hwId': '60FF6B258734EA6880E186F8C96DDC61',
|
|
'hw_ver': '1.0',
|
|
'icon_hash': '',
|
|
'latitude': 12.2,
|
|
'led_off': 0,
|
|
'longitude': -12.2,
|
|
'mac': 'AA:BB:CC:11:22:33',
|
|
'model': 'HS110(US)',
|
|
'oemId': 'FFF22CFF774A0B89F7624BFC6F50D5DE',
|
|
'on_time': 9022,
|
|
'relay_state': 1,
|
|
'rssi': -61,
|
|
'sw_ver': '1.0.8 Build 151113 Rel.24658',
|
|
'type': 'IOT.SMARTPLUGSWITCH',
|
|
'updating': 0}
|
|
},
|
|
"emeter": emeter_support,
|
|
}
|
|
|
|
sysinfo_hs200 = {'system': {'get_sysinfo': {'active_mode': 'schedule',
|
|
'alias': 'Christmas Tree Switch',
|
|
'dev_name': 'Wi-Fi Smart Light Switch',
|
|
'deviceId': '8006E0D62C90698C6A3EF72944F56DDC17D0DB80',
|
|
'err_code': 0,
|
|
'feature': 'TIM',
|
|
'fwId': 'DB4F3246CD85AA59CAE738A63E7B9C34',
|
|
'hwId': 'A0E3CC8F5C1166B27A16D56BE262A6D3',
|
|
'hw_ver': '1.0',
|
|
'icon_hash': '',
|
|
'latitude': 12.2,
|
|
'led_off': 0,
|
|
'longitude': -12.2,
|
|
'mac': 'AA:BB:CC:11:22:33',
|
|
'mic_type': 'IOT.SMARTPLUGSWITCH',
|
|
'model': 'HS200(US)',
|
|
'oemId': '4AFE44A41F868FD2340E6D1308D8551D',
|
|
'on_time': 9586,
|
|
'relay_state': 1,
|
|
'rssi': -53,
|
|
'sw_ver': '1.1.0 Build 160521 Rel.085826',
|
|
'updating': 0}}
|
|
}
|
|
|
|
sysinfo_lb130 = {'system': {'get_sysinfo':
|
|
{'active_mode': 'none',
|
|
'alias': 'Living Room Side Table',
|
|
'ctrl_protocols': {'name': 'Linkie', 'version': '1.0'},
|
|
'description': 'Smart Wi-Fi LED Bulb with Color Changing',
|
|
'dev_state': 'normal',
|
|
'deviceId': '80123C4640E9FC33A9019A0F3FD8BF5C17B7D9A8',
|
|
'disco_ver': '1.0',
|
|
'heapsize': 347000,
|
|
'hwId': '111E35908497A05512E259BB76801E10',
|
|
'hw_ver': '1.0',
|
|
'is_color': 1,
|
|
'is_dimmable': 1,
|
|
'is_factory': False,
|
|
'is_variable_color_temp': 1,
|
|
'light_state': {'brightness': 100,
|
|
'color_temp': 3700,
|
|
'hue': 0,
|
|
'mode': 'normal',
|
|
'on_off': 1,
|
|
'saturation': 0},
|
|
'mic_mac': '50C7BF104865',
|
|
'mic_type': 'IOT.SMARTBULB',
|
|
'model': 'LB130(US)',
|
|
'oemId': '05BF7B3BE1675C5A6867B7A7E4C9F6F7',
|
|
'preferred_state': [{'brightness': 50,
|
|
'color_temp': 2700,
|
|
'hue': 0,
|
|
'index': 0,
|
|
'saturation': 0},
|
|
{'brightness': 100,
|
|
'color_temp': 0,
|
|
'hue': 0,
|
|
'index': 1,
|
|
'saturation': 75},
|
|
{'brightness': 100,
|
|
'color_temp': 0,
|
|
'hue': 120,
|
|
'index': 2,
|
|
'saturation': 75},
|
|
{'brightness': 100,
|
|
'color_temp': 0,
|
|
'hue': 240,
|
|
'index': 3,
|
|
'saturation': 75}],
|
|
'rssi': -55,
|
|
'sw_ver': '1.1.2 Build 160927 Rel.111100'}},
|
|
'smartlife.iot.smartbulb.lightingservice': {'get_light_state':
|
|
{'on_off':1,
|
|
'mode':'normal',
|
|
'hue': 0,
|
|
'saturation': 0,
|
|
'color_temp': 3700,
|
|
'brightness': 100,
|
|
'err_code': 0}},
|
|
'smartlife.iot.common.emeter': emeter_units_support,
|
|
}
|
|
|
|
def error(cls, target, cmd="no-command", msg="default msg"):
|
|
return {target: {cmd: {"err_code": -1323, "msg": msg}}}
|
|
|
|
|
|
def success(target, cmd, res):
|
|
if res:
|
|
res.update({"err_code": 0})
|
|
else:
|
|
res = {"err_code": 0}
|
|
return {target: {cmd: res}}
|
|
|
|
|
|
class FakeTransportProtocol(TPLinkSmartHomeProtocol):
|
|
def __init__(self, sysinfo, invalid=False):
|
|
""" invalid is set only for testing
|
|
to force query() to throw the exception for non-connected """
|
|
proto = FakeTransportProtocol.baseproto
|
|
for target in sysinfo:
|
|
for cmd in sysinfo[target]:
|
|
proto[target][cmd] = sysinfo[target][cmd]
|
|
self.proto = proto
|
|
self.invalid = invalid
|
|
|
|
def set_alias(self, x):
|
|
_LOGGER.debug("Setting alias to %s", x["alias"])
|
|
self.proto["system"]["get_sysinfo"]["alias"] = x["alias"]
|
|
|
|
def set_relay_state(self, x):
|
|
_LOGGER.debug("Setting relay state to %s", x)
|
|
self.proto["system"]["get_sysinfo"]["relay_state"] = x["state"]
|
|
|
|
def set_led_off(self, x):
|
|
_LOGGER.debug("Setting led off to %s", x)
|
|
self.proto["system"]["get_sysinfo"]["led_off"] = x["off"]
|
|
|
|
def set_mac(self, x):
|
|
_LOGGER.debug("Setting mac to %s", x)
|
|
self.proto["system"]["get_sysinfo"][""]
|
|
|
|
def transition_light_state(self, x):
|
|
_LOGGER.debug("Setting light state to %s", x)
|
|
for key in x:
|
|
self.proto["smartlife.iot.smartbulb.lightingservice"]["get_light_state"][key]=x[key]
|
|
|
|
baseproto = {
|
|
"system": { "set_relay_state": set_relay_state,
|
|
"set_dev_alias": set_alias,
|
|
"set_led_off": set_led_off,
|
|
"get_dev_icon": {"icon": None, "hash": None},
|
|
"set_mac_addr": set_mac,
|
|
"get_sysinfo": None,
|
|
},
|
|
"emeter": { "get_realtime": None,
|
|
"get_daystat": None,
|
|
"get_monthstat": None,
|
|
"erase_emeter_state": None
|
|
},
|
|
"smartlife.iot.common.emeter": { "get_realtime": None,
|
|
"get_daystat": None,
|
|
"get_monthstat": None,
|
|
"erase_emeter_state": None
|
|
},
|
|
"smartlife.iot.smartbulb.lightingservice": { "get_light_state": None,
|
|
"transition_light_state": transition_light_state,
|
|
},
|
|
"time": { "get_time": { "year": 2017, "month": 1, "mday": 2, "hour": 3, "min": 4, "sec": 5 },
|
|
"get_timezone": {'zone_str': "test", 'dst_offset': -1, 'index': 12, 'tz_str': "test2" },
|
|
"set_timezone": None,
|
|
|
|
}
|
|
}
|
|
|
|
def query(self, host, request, port=9999):
|
|
if self.invalid:
|
|
raise SmartPlugException("Invalid connection, can't query!")
|
|
|
|
proto = self.proto
|
|
|
|
target = next(iter(request))
|
|
if target not in proto.keys():
|
|
return error(target, msg="target not found")
|
|
|
|
cmd = next(iter(request[target]))
|
|
if cmd not in proto[target].keys():
|
|
return error(target, cmd, msg="command not found")
|
|
|
|
params = request[target][cmd]
|
|
_LOGGER.debug("Going to execute {}.{} (params: {}).. ".format(target, cmd, params))
|
|
|
|
if callable(proto[target][cmd]):
|
|
res = proto[target][cmd](self, params)
|
|
# verify that change didn't break schema, requires refactoring..
|
|
#TestSmartPlug.sysinfo_schema(self.proto["system"]["get_sysinfo"])
|
|
return success(target, cmd, res)
|
|
elif isinstance(proto[target][cmd], dict):
|
|
return success(target, cmd, proto[target][cmd])
|
|
else:
|
|
raise NotImplementedError("target {} cmd {}".format(target, cmd))
|