Compare commits

..

14 Commits

6 changed files with 45 additions and 14 deletions

View File

@ -55,11 +55,13 @@ class HonConnection:
appliances = (await resp.json())["payload"]["appliances"] appliances = (await resp.json())["payload"]["appliances"]
for appliance in appliances: for appliance in appliances:
device = HonDevice(self, appliance) device = HonDevice(self, appliance)
if device.mac_address is None:
continue
await asyncio.gather(*[ await asyncio.gather(*[
device.load_attributes(), device.load_attributes(),
device.load_commands(), device.load_commands(),
device.load_statistics()]) device.load_statistics()])
self._devices.append(device) self._devices.append(device)
except json.JSONDecodeError: except json.JSONDecodeError:
_LOGGER.error("No JSON Data after GET: %s", await resp.text()) _LOGGER.error("No JSON Data after GET: %s", await resp.text())
return False return False
@ -70,7 +72,7 @@ class HonConnection:
"applianceType": device.appliance_type, "applianceType": device.appliance_type,
"code": device.appliance["code"], "code": device.appliance["code"],
"applianceModelId": device.appliance_model_id, "applianceModelId": device.appliance_model_id,
"firmwareId": "41", "firmwareId": device.appliance["eepromId"],
"macAddress": device.mac_address, "macAddress": device.mac_address,
"fwVersion": device.appliance["fwVersion"], "fwVersion": device.appliance["fwVersion"],
"os": const.OS, "os": const.OS,
@ -84,6 +86,14 @@ class HonConnection:
return {} return {}
return result return result
async def command_history(self, device: HonDevice):
url = f"{const.API_URL}/commands/v1/appliance/{device.mac_address}/history"
async with self._session.get(url, headers=await self._headers) as response:
result = await response.json()
if not result or not result.get("payload"):
return {}
return result["payload"]["history"]
async def load_attributes(self, device: HonDevice, loop=False): async def load_attributes(self, device: HonDevice, loop=False):
params = { params = {
"macAddress": device.mac_address, "macAddress": device.mac_address,

View File

@ -2,12 +2,12 @@ from pyhon.parameter import HonParameterFixed, HonParameterEnum, HonParameterRan
class HonCommand: class HonCommand:
def __init__(self, name, attributes, connector, device, multi=None, category=""): def __init__(self, name, attributes, connector, device, multi=None, program=""):
self._connector = connector self._connector = connector
self._device = device self._device = device
self._name = name self._name = name
self._multi = multi or {} self._multi = multi or {}
self._category = category self._program = program
self._description = attributes.get("description", "") self._description = attributes.get("description", "")
self._parameters = self._create_parameters(attributes.get("parameters", {})) self._parameters = self._create_parameters(attributes.get("parameters", {}))
self._ancillary_parameters = self._create_parameters(attributes.get("ancillaryParameters", {})) self._ancillary_parameters = self._create_parameters(attributes.get("ancillaryParameters", {}))
@ -67,5 +67,5 @@ class HonCommand:
@property @property
def settings(self): def settings(self):
"""Parameters with typology enum and range""" """Parameters with typology enum and range"""
return {s: self._parameters[s] for s in self.setting_keys} return {s: self._parameters.get(s) for s in self.setting_keys if self._parameters.get(s) is not None}

View File

@ -3,7 +3,7 @@ API_URL = "https://api-iot.he.services"
APP = "hon" APP = "hon"
# All seen id's (different accounts, different devices) are the same, so I guess this hash is static # All seen id's (different accounts, different devices) are the same, so I guess this hash is static
CLIENT_ID = "3MVG9QDx8IX8nP5T2Ha8ofvlmjLZl5L_gvfbT9.HJvpHGKoAS_dcMN8LYpTSYeVFCraUnV.2Ag1Ki7m4znVO6" CLIENT_ID = "3MVG9QDx8IX8nP5T2Ha8ofvlmjLZl5L_gvfbT9.HJvpHGKoAS_dcMN8LYpTSYeVFCraUnV.2Ag1Ki7m4znVO6"
APP_VERSION = "1.51.9" APP_VERSION = "1.53.7"
OS_VERSION = 31 OS_VERSION = 31
OS = "android" OS = "android"
DEVICE_MODEL = "exynos9820" DEVICE_MODEL = "exynos9820"

View File

@ -1,11 +1,14 @@
import importlib import importlib
from contextlib import suppress
from pyhon.commands import HonCommand from pyhon.commands import HonCommand
from pyhon.parameter import HonParameterFixed
class HonDevice: class HonDevice:
def __init__(self, connector, appliance): def __init__(self, connector, appliance):
appliance["attributes"] = {v["parName"]: v["parValue"] for v in appliance["attributes"]} if attributes := appliance.get("attributes"):
appliance["attributes"] = {v["parName"]: v["parValue"] for v in attributes}
self._appliance = appliance self._appliance = appliance
self._connector = connector self._connector = connector
self._appliance_model = {} self._appliance_model = {}
@ -81,6 +84,21 @@ class HonDevice:
def appliance(self): def appliance(self):
return self._appliance return self._appliance
async def _recover_last_command_states(self, commands):
command_history = await self._connector.command_history(self)
for name, command in commands.items():
last = next((index for (index, d) in enumerate(command_history) if d.get("command", {}).get("commandName") == name), None)
if last is None:
continue
parameters = command_history[last].get("command", {}).get("parameters", {})
if command._multi and parameters.get("program"):
command.set_program(parameters.pop("program").split(".")[-1].lower())
command = self.commands[name]
for key, data in command.settings.items():
if not isinstance(data, HonParameterFixed) and parameters.get(key) is not None:
with suppress(ValueError):
data.value = parameters.get(key)
async def load_commands(self): async def load_commands(self):
raw = await self._connector.load_commands(self) raw = await self._connector.load_commands(self)
self._appliance_model = raw.pop("applianceModel") self._appliance_model = raw.pop("applianceModel")
@ -92,11 +110,13 @@ class HonDevice:
commands[command] = HonCommand(command, attr, self._connector, self) commands[command] = HonCommand(command, attr, self._connector, self)
elif "parameters" in attr[list(attr)[0]]: elif "parameters" in attr[list(attr)[0]]:
multi = {} multi = {}
for category, attr2 in attr.items(): for program, attr2 in attr.items():
cmd = HonCommand(command, attr2, self._connector, self, multi=multi, category=category) program = program.split(".")[-1].lower()
multi[category] = cmd cmd = HonCommand(command, attr2, self._connector, self, multi=multi, program=program)
multi[program] = cmd
commands[command] = cmd commands[command] = cmd
self._commands = commands self._commands = commands
await self._recover_last_command_states(commands)
@property @property
def settings(self): def settings(self):

View File

@ -30,7 +30,7 @@ class HonParameter:
class HonParameterFixed(HonParameter): class HonParameterFixed(HonParameter):
def __init__(self, key, attributes): def __init__(self, key, attributes):
super().__init__(key, attributes) super().__init__(key, attributes)
self._value = attributes["fixedValue"] self._value = attributes.get("fixedValue", None)
def __repr__(self): def __repr__(self):
return f"{self.__class__} (<{self.key}> fixed)" return f"{self.__class__} (<{self.key}> fixed)"
@ -75,6 +75,7 @@ class HonParameterRange(HonParameter):
@value.setter @value.setter
def value(self, value): def value(self, value):
value = int(value)
if self._min <= value <= self._max and not value % self._step: if self._min <= value <= self._max and not value % self._step:
self._value = value self._value = value
else: else:
@ -93,7 +94,7 @@ class HonParameterEnum(HonParameter):
@property @property
def values(self): def values(self):
return [str(value) for value in self._values] return sorted([str(value) for value in self._values])
@property @property
def value(self): def value(self):
@ -111,7 +112,7 @@ class HonParameterProgram(HonParameterEnum):
def __init__(self, key, command): def __init__(self, key, command):
super().__init__(key, {}) super().__init__(key, {})
self._command = command self._command = command
self._value = command._category self._value = command._program
self._values = command._multi self._values = command._multi
self._typology = "enum" self._typology = "enum"

View File

@ -7,7 +7,7 @@ with open("README.md", "r") as f:
setup( setup(
name="pyhOn", name="pyhOn",
version="0.3.4", version="0.3.8",
author="Andre Basche", author="Andre Basche",
description="Control hOn devices with python", description="Control hOn devices with python",
long_description=long_description, long_description=long_description,