Add more type hints
This commit is contained in:
parent
ad0d065b03
commit
9eb99f283b
30 changed files with 392 additions and 243 deletions
4
mypy.ini
Normal file
4
mypy.ini
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
[mypy]
|
||||||
|
check_untyped_defs = True
|
||||||
|
disallow_any_generics = True
|
||||||
|
disallow_untyped_defs = True
|
|
@ -6,16 +6,17 @@ import logging
|
||||||
import sys
|
import sys
|
||||||
from getpass import getpass
|
from getpass import getpass
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
from typing import Tuple, Dict, Any
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
sys.path.insert(0, str(Path(__file__).parent.parent))
|
sys.path.insert(0, str(Path(__file__).parent.parent))
|
||||||
|
|
||||||
from pyhon import Hon, HonAPI, helper, diagnose
|
from pyhon import Hon, HonAPI, diagnose, printer
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
def get_arguments():
|
def get_arguments() -> Dict[str, Any]:
|
||||||
"""Get parsed arguments."""
|
"""Get parsed arguments."""
|
||||||
parser = argparse.ArgumentParser(description="pyhOn: Command Line Utility")
|
parser = argparse.ArgumentParser(description="pyhOn: Command Line Utility")
|
||||||
parser.add_argument("-u", "--user", help="user for haier hOn account")
|
parser.add_argument("-u", "--user", help="user for haier hOn account")
|
||||||
|
@ -39,7 +40,7 @@ def get_arguments():
|
||||||
return vars(parser.parse_args())
|
return vars(parser.parse_args())
|
||||||
|
|
||||||
|
|
||||||
async def translate(language, json_output=False):
|
async def translate(language: str, json_output: bool = False) -> None:
|
||||||
async with HonAPI(anonymous=True) as hon:
|
async with HonAPI(anonymous=True) as hon:
|
||||||
keys = await hon.translation_keys(language)
|
keys = await hon.translation_keys(language)
|
||||||
if json_output:
|
if json_output:
|
||||||
|
@ -52,10 +53,10 @@ async def translate(language, json_output=False):
|
||||||
.replace("\\r", "")
|
.replace("\\r", "")
|
||||||
)
|
)
|
||||||
keys = json.loads(clean_keys)
|
keys = json.loads(clean_keys)
|
||||||
print(helper.pretty_print(keys))
|
print(printer.pretty_print(keys))
|
||||||
|
|
||||||
|
|
||||||
def get_login_data(args):
|
def get_login_data(args: Dict[str, str]) -> Tuple[str, str]:
|
||||||
if not (user := args["user"]):
|
if not (user := args["user"]):
|
||||||
user = input("User for hOn account: ")
|
user = input("User for hOn account: ")
|
||||||
if not (password := args["password"]):
|
if not (password := args["password"]):
|
||||||
|
@ -63,44 +64,44 @@ def get_login_data(args):
|
||||||
return user, password
|
return user, password
|
||||||
|
|
||||||
|
|
||||||
async def main():
|
async def main() -> None:
|
||||||
args = get_arguments()
|
args = get_arguments()
|
||||||
if language := args.get("translate"):
|
if language := args.get("translate"):
|
||||||
await translate(language, json_output=args.get("json"))
|
await translate(language, json_output=args.get("json", ""))
|
||||||
return
|
return
|
||||||
async with Hon(*get_login_data(args)) as hon:
|
async with Hon(*get_login_data(args)) as hon:
|
||||||
for device in hon.appliances:
|
for device in hon.appliances:
|
||||||
if args.get("export"):
|
if args.get("export"):
|
||||||
anonymous = args.get("anonymous", False)
|
anonymous = args.get("anonymous", False)
|
||||||
path = Path(args.get("directory"))
|
path = Path(args.get("directory", "."))
|
||||||
if not args.get("zip"):
|
if not args.get("zip"):
|
||||||
for file in await diagnose.appliance_data(device, path, anonymous):
|
for file in await diagnose.appliance_data(device, path, anonymous):
|
||||||
print(f"Created {file}")
|
print(f"Created {file}")
|
||||||
else:
|
else:
|
||||||
file = await diagnose.zip_archive(device, path, anonymous)
|
archive = await diagnose.zip_archive(device, path, anonymous)
|
||||||
print(f"Created {file}")
|
print(f"Created {archive}")
|
||||||
continue
|
continue
|
||||||
print("=" * 10, device.appliance_type, "-", device.nick_name, "=" * 10)
|
print("=" * 10, device.appliance_type, "-", device.nick_name, "=" * 10)
|
||||||
if args.get("keys"):
|
if args.get("keys"):
|
||||||
data = device.data.copy()
|
data = device.data.copy()
|
||||||
attr = "get" if args.get("all") else "pop"
|
attr = "get" if args.get("all") else "pop"
|
||||||
print(
|
print(
|
||||||
helper.key_print(
|
printer.key_print(
|
||||||
data["attributes"].__getattribute__(attr)("parameters")
|
data["attributes"].__getattribute__(attr)("parameters")
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
print(helper.key_print(data.__getattribute__(attr)("appliance")))
|
print(printer.key_print(data.__getattribute__(attr)("appliance")))
|
||||||
print(helper.key_print(data))
|
print(printer.key_print(data))
|
||||||
print(
|
print(
|
||||||
helper.pretty_print(
|
printer.pretty_print(
|
||||||
helper.create_command(device.commands, concat=True)
|
printer.create_command(device.commands, concat=True)
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
else:
|
else:
|
||||||
print(diagnose.yaml_export(device))
|
print(diagnose.yaml_export(device))
|
||||||
|
|
||||||
|
|
||||||
def start():
|
def start() -> None:
|
||||||
try:
|
try:
|
||||||
asyncio.run(main())
|
asyncio.run(main())
|
||||||
except KeyboardInterrupt:
|
except KeyboardInterrupt:
|
||||||
|
|
|
@ -2,14 +2,15 @@ import importlib
|
||||||
import logging
|
import logging
|
||||||
from datetime import datetime, timedelta
|
from datetime import datetime, timedelta
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from typing import Optional, Dict, Any, TYPE_CHECKING
|
from typing import Optional, Dict, Any, TYPE_CHECKING, List
|
||||||
|
|
||||||
from pyhon import diagnose
|
from pyhon import diagnose, exceptions
|
||||||
from pyhon.attributes import HonAttribute
|
from pyhon.attributes import HonAttribute
|
||||||
from pyhon.command_loader import HonCommandLoader
|
from pyhon.command_loader import HonCommandLoader
|
||||||
from pyhon.commands import HonCommand
|
from pyhon.commands import HonCommand
|
||||||
from pyhon.parameter.base import HonParameter
|
from pyhon.parameter.base import HonParameter
|
||||||
from pyhon.parameter.range import HonParameterRange
|
from pyhon.parameter.range import HonParameterRange
|
||||||
|
from pyhon.typedefs import Parameter
|
||||||
|
|
||||||
if TYPE_CHECKING:
|
if TYPE_CHECKING:
|
||||||
from pyhon import HonAPI
|
from pyhon import HonAPI
|
||||||
|
@ -25,16 +26,16 @@ class HonAppliance:
|
||||||
) -> None:
|
) -> None:
|
||||||
if attributes := info.get("attributes"):
|
if attributes := info.get("attributes"):
|
||||||
info["attributes"] = {v["parName"]: v["parValue"] for v in attributes}
|
info["attributes"] = {v["parName"]: v["parValue"] for v in attributes}
|
||||||
self._info: Dict = info
|
self._info: Dict[str, Any] = info
|
||||||
self._api: Optional[HonAPI] = api
|
self._api: Optional[HonAPI] = api
|
||||||
self._appliance_model: Dict = {}
|
self._appliance_model: Dict[str, Any] = {}
|
||||||
|
|
||||||
self._commands: Dict[str, HonCommand] = {}
|
self._commands: Dict[str, HonCommand] = {}
|
||||||
self._statistics: Dict = {}
|
self._statistics: Dict[str, Any] = {}
|
||||||
self._attributes: Dict = {}
|
self._attributes: Dict[str, Any] = {}
|
||||||
self._zone: int = zone
|
self._zone: int = zone
|
||||||
self._additional_data: Dict[str, Any] = {}
|
self._additional_data: Dict[str, Any] = {}
|
||||||
self._last_update = None
|
self._last_update: Optional[datetime] = None
|
||||||
self._default_setting = HonParameter("", {}, "")
|
self._default_setting = HonParameter("", {}, "")
|
||||||
|
|
||||||
try:
|
try:
|
||||||
|
@ -44,7 +45,7 @@ class HonAppliance:
|
||||||
except ModuleNotFoundError:
|
except ModuleNotFoundError:
|
||||||
self._extra = None
|
self._extra = None
|
||||||
|
|
||||||
def __getitem__(self, item):
|
def __getitem__(self, item: str) -> Any:
|
||||||
if self._zone:
|
if self._zone:
|
||||||
item += f"Z{self._zone}"
|
item += f"Z{self._zone}"
|
||||||
if "." in item:
|
if "." in item:
|
||||||
|
@ -61,7 +62,7 @@ class HonAppliance:
|
||||||
return self.attributes["parameters"][item].value
|
return self.attributes["parameters"][item].value
|
||||||
return self.info[item]
|
return self.info[item]
|
||||||
|
|
||||||
def get(self, item, default=None):
|
def get(self, item: str, default: Any = None) -> Any:
|
||||||
try:
|
try:
|
||||||
return self[item]
|
return self[item]
|
||||||
except (KeyError, IndexError):
|
except (KeyError, IndexError):
|
||||||
|
@ -113,7 +114,7 @@ class HonAppliance:
|
||||||
return self._info.get("applianceModelId", 0)
|
return self._info.get("applianceModelId", 0)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def options(self):
|
def options(self) -> Dict[str, Any]:
|
||||||
return self._appliance_model.get("options", {})
|
return self._appliance_model.get("options", {})
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
@ -121,19 +122,19 @@ class HonAppliance:
|
||||||
return self._commands
|
return self._commands
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def attributes(self):
|
def attributes(self) -> Dict[str, Any]:
|
||||||
return self._attributes
|
return self._attributes
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def statistics(self):
|
def statistics(self) -> Dict[str, Any]:
|
||||||
return self._statistics
|
return self._statistics
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def info(self):
|
def info(self) -> Dict[str, Any]:
|
||||||
return self._info
|
return self._info
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def additional_data(self):
|
def additional_data(self) -> Dict[str, Any]:
|
||||||
return self._additional_data
|
return self._additional_data
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
@ -141,17 +142,20 @@ class HonAppliance:
|
||||||
return self._zone
|
return self._zone
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def api(self) -> Optional["HonAPI"]:
|
def api(self) -> "HonAPI":
|
||||||
|
"""api connection object"""
|
||||||
|
if self._api is None:
|
||||||
|
raise exceptions.NoAuthenticationException("Missing hOn login")
|
||||||
return self._api
|
return self._api
|
||||||
|
|
||||||
async def load_commands(self):
|
async def load_commands(self) -> None:
|
||||||
command_loader = HonCommandLoader(self.api, self)
|
command_loader = HonCommandLoader(self.api, self)
|
||||||
await command_loader.load_commands()
|
await command_loader.load_commands()
|
||||||
self._commands = command_loader.commands
|
self._commands = command_loader.commands
|
||||||
self._additional_data = command_loader.additional_data
|
self._additional_data = command_loader.additional_data
|
||||||
self._appliance_model = command_loader.appliance_data
|
self._appliance_model = command_loader.appliance_data
|
||||||
|
|
||||||
async def load_attributes(self):
|
async def load_attributes(self) -> None:
|
||||||
self._attributes = await self.api.load_attributes(self)
|
self._attributes = await self.api.load_attributes(self)
|
||||||
for name, values in self._attributes.pop("shadow").get("parameters").items():
|
for name, values in self._attributes.pop("shadow").get("parameters").items():
|
||||||
if name in self._attributes.get("parameters", {}):
|
if name in self._attributes.get("parameters", {}):
|
||||||
|
@ -163,11 +167,11 @@ class HonAppliance:
|
||||||
if self._extra:
|
if self._extra:
|
||||||
self._attributes = self._extra.attributes(self._attributes)
|
self._attributes = self._extra.attributes(self._attributes)
|
||||||
|
|
||||||
async def load_statistics(self):
|
async def load_statistics(self) -> None:
|
||||||
self._statistics = await self.api.load_statistics(self)
|
self._statistics = await self.api.load_statistics(self)
|
||||||
self._statistics |= await self.api.load_maintenance(self)
|
self._statistics |= await self.api.load_maintenance(self)
|
||||||
|
|
||||||
async def update(self, force=False):
|
async def update(self, force: bool = False) -> None:
|
||||||
now = datetime.now()
|
now = datetime.now()
|
||||||
if (
|
if (
|
||||||
force
|
force
|
||||||
|
@ -179,11 +183,11 @@ class HonAppliance:
|
||||||
await self.load_attributes()
|
await self.load_attributes()
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def command_parameters(self):
|
def command_parameters(self) -> Dict[str, Dict[str, str | float]]:
|
||||||
return {n: c.parameter_value for n, c in self._commands.items()}
|
return {n: c.parameter_value for n, c in self._commands.items()}
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def settings(self):
|
def settings(self) -> Dict[str, Parameter]:
|
||||||
result = {}
|
result = {}
|
||||||
for name, command in self._commands.items():
|
for name, command in self._commands.items():
|
||||||
for key in command.setting_keys:
|
for key in command.setting_keys:
|
||||||
|
@ -194,7 +198,7 @@ class HonAppliance:
|
||||||
return result
|
return result
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def available_settings(self):
|
def available_settings(self) -> List[str]:
|
||||||
result = []
|
result = []
|
||||||
for name, command in self._commands.items():
|
for name, command in self._commands.items():
|
||||||
for key in command.setting_keys:
|
for key in command.setting_keys:
|
||||||
|
@ -202,7 +206,7 @@ class HonAppliance:
|
||||||
return result
|
return result
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def data(self):
|
def data(self) -> Dict[str, Any]:
|
||||||
result = {
|
result = {
|
||||||
"attributes": self.attributes,
|
"attributes": self.attributes,
|
||||||
"appliance": self.info,
|
"appliance": self.info,
|
||||||
|
@ -220,15 +224,16 @@ class HonAppliance:
|
||||||
async def data_archive(self, path: Path) -> str:
|
async def data_archive(self, path: Path) -> str:
|
||||||
return await diagnose.zip_archive(self, path, anonymous=True)
|
return await diagnose.zip_archive(self, path, anonymous=True)
|
||||||
|
|
||||||
def sync_to_params(self, command_name):
|
def sync_to_params(self, command_name: str) -> None:
|
||||||
command: HonCommand = self.commands.get(command_name)
|
if not (command := self.commands.get(command_name)):
|
||||||
|
return
|
||||||
for key, value in self.attributes.get("parameters", {}).items():
|
for key, value in self.attributes.get("parameters", {}).items():
|
||||||
if isinstance(value, str) and (new := command.parameters.get(key)):
|
if isinstance(value, str) and (new := command.parameters.get(key)):
|
||||||
self.attributes["parameters"][key].update(
|
self.attributes["parameters"][key].update(
|
||||||
str(new.intern_value), shield=True
|
str(new.intern_value), shield=True
|
||||||
)
|
)
|
||||||
|
|
||||||
def sync_command(self, main, target=None) -> None:
|
def sync_command(self, main: str, target: Optional[List[str]] = None) -> None:
|
||||||
base: Optional[HonCommand] = self.commands.get(main)
|
base: Optional[HonCommand] = self.commands.get(main)
|
||||||
if not base:
|
if not base:
|
||||||
return
|
return
|
||||||
|
|
|
@ -1,15 +1,25 @@
|
||||||
|
from typing import Dict, Any, TYPE_CHECKING
|
||||||
|
|
||||||
|
from pyhon.parameter.program import HonParameterProgram
|
||||||
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from pyhon.appliance import HonAppliance
|
||||||
|
|
||||||
|
|
||||||
class ApplianceBase:
|
class ApplianceBase:
|
||||||
def __init__(self, appliance):
|
def __init__(self, appliance: "HonAppliance"):
|
||||||
self.parent = appliance
|
self.parent = appliance
|
||||||
|
|
||||||
def attributes(self, data):
|
def attributes(self, data: Dict[str, Any]) -> Dict[str, Any]:
|
||||||
program_name = "No Program"
|
program_name = "No Program"
|
||||||
if program := int(str(data.get("parameters", {}).get("prCode", "0"))):
|
if program := int(str(data.get("parameters", {}).get("prCode", "0"))):
|
||||||
if start_cmd := self.parent.settings.get("startProgram.program"):
|
if start_cmd := self.parent.settings.get("startProgram.program"):
|
||||||
if ids := start_cmd.ids:
|
if isinstance(start_cmd, HonParameterProgram) and (
|
||||||
|
ids := start_cmd.ids
|
||||||
|
):
|
||||||
program_name = ids.get(program, program_name)
|
program_name = ids.get(program, program_name)
|
||||||
data["programName"] = program_name
|
data["programName"] = program_name
|
||||||
return data
|
return data
|
||||||
|
|
||||||
def settings(self, settings):
|
def settings(self, settings: Dict[str, Any]) -> Dict[str, Any]:
|
||||||
return settings
|
return settings
|
||||||
|
|
|
@ -1,8 +1,10 @@
|
||||||
|
from typing import Any, Dict
|
||||||
|
|
||||||
from pyhon.appliances.base import ApplianceBase
|
from pyhon.appliances.base import ApplianceBase
|
||||||
|
|
||||||
|
|
||||||
class Appliance(ApplianceBase):
|
class Appliance(ApplianceBase):
|
||||||
def attributes(self, data):
|
def attributes(self, data: Dict[str, Any]) -> Dict[str, Any]:
|
||||||
data = super().attributes(data)
|
data = super().attributes(data)
|
||||||
if data.get("lastConnEvent", {}).get("category", "") == "DISCONNECTED":
|
if data.get("lastConnEvent", {}).get("category", "") == "DISCONNECTED":
|
||||||
data["parameters"]["machMode"].value = "0"
|
data["parameters"]["machMode"].value = "0"
|
||||||
|
|
|
@ -1,8 +1,11 @@
|
||||||
|
from typing import Any, Dict
|
||||||
|
|
||||||
from pyhon.appliances.base import ApplianceBase
|
from pyhon.appliances.base import ApplianceBase
|
||||||
|
from pyhon.parameter.program import HonParameterProgram
|
||||||
|
|
||||||
|
|
||||||
class Appliance(ApplianceBase):
|
class Appliance(ApplianceBase):
|
||||||
def attributes(self, data):
|
def attributes(self, data: Dict[str, Any]) -> Dict[str, Any]:
|
||||||
data = super().attributes(data)
|
data = super().attributes(data)
|
||||||
if data.get("lastConnEvent", {}).get("category", "") == "DISCONNECTED":
|
if data.get("lastConnEvent", {}).get("category", "") == "DISCONNECTED":
|
||||||
data["parameters"]["temp"].value = "0"
|
data["parameters"]["temp"].value = "0"
|
||||||
|
@ -13,7 +16,9 @@ class Appliance(ApplianceBase):
|
||||||
data["active"] = data["parameters"]["onOffStatus"] == "1"
|
data["active"] = data["parameters"]["onOffStatus"] == "1"
|
||||||
|
|
||||||
if program := int(data["parameters"]["prCode"]):
|
if program := int(data["parameters"]["prCode"]):
|
||||||
ids = self.parent.settings["startProgram.program"].ids
|
if (setting := self.parent.settings["startProgram.program"]) and isinstance(
|
||||||
data["programName"] = ids.get(program, "")
|
setting, HonParameterProgram
|
||||||
|
):
|
||||||
|
data["programName"] = setting.ids.get(program, "")
|
||||||
|
|
||||||
return data
|
return data
|
||||||
|
|
|
@ -1,8 +1,10 @@
|
||||||
|
from typing import Dict, Any
|
||||||
|
|
||||||
from pyhon.appliances.base import ApplianceBase
|
from pyhon.appliances.base import ApplianceBase
|
||||||
|
|
||||||
|
|
||||||
class Appliance(ApplianceBase):
|
class Appliance(ApplianceBase):
|
||||||
def attributes(self, data):
|
def attributes(self, data: Dict[str, Any]) -> Dict[str, Any]:
|
||||||
data = super().attributes(data)
|
data = super().attributes(data)
|
||||||
if data["parameters"]["holidayMode"] == "1":
|
if data["parameters"]["holidayMode"] == "1":
|
||||||
data["modeZ1"] = "holiday"
|
data["modeZ1"] = "holiday"
|
||||||
|
|
|
@ -1,9 +1,11 @@
|
||||||
|
from typing import Any, Dict
|
||||||
|
|
||||||
from pyhon.appliances.base import ApplianceBase
|
from pyhon.appliances.base import ApplianceBase
|
||||||
from pyhon.parameter.fixed import HonParameterFixed
|
from pyhon.parameter.fixed import HonParameterFixed
|
||||||
|
|
||||||
|
|
||||||
class Appliance(ApplianceBase):
|
class Appliance(ApplianceBase):
|
||||||
def attributes(self, data):
|
def attributes(self, data: Dict[str, Any]) -> Dict[str, Any]:
|
||||||
data = super().attributes(data)
|
data = super().attributes(data)
|
||||||
if data.get("lastConnEvent", {}).get("category", "") == "DISCONNECTED":
|
if data.get("lastConnEvent", {}).get("category", "") == "DISCONNECTED":
|
||||||
data["parameters"]["machMode"].value = "0"
|
data["parameters"]["machMode"].value = "0"
|
||||||
|
@ -11,7 +13,7 @@ class Appliance(ApplianceBase):
|
||||||
data["pause"] = data["parameters"]["machMode"] == "3"
|
data["pause"] = data["parameters"]["machMode"] == "3"
|
||||||
return data
|
return data
|
||||||
|
|
||||||
def settings(self, settings):
|
def settings(self, settings: Dict[str, Any]) -> Dict[str, Any]:
|
||||||
dry_level = settings.get("startProgram.dryLevel")
|
dry_level = settings.get("startProgram.dryLevel")
|
||||||
if isinstance(dry_level, HonParameterFixed) and dry_level.value == "11":
|
if isinstance(dry_level, HonParameterFixed) and dry_level.value == "11":
|
||||||
settings.pop("startProgram.dryLevel", None)
|
settings.pop("startProgram.dryLevel", None)
|
||||||
|
|
|
@ -1,8 +1,10 @@
|
||||||
|
from typing import Dict, Any
|
||||||
|
|
||||||
from pyhon.appliances.base import ApplianceBase
|
from pyhon.appliances.base import ApplianceBase
|
||||||
|
|
||||||
|
|
||||||
class Appliance(ApplianceBase):
|
class Appliance(ApplianceBase):
|
||||||
def attributes(self, data):
|
def attributes(self, data: Dict[str, Any]) -> Dict[str, Any]:
|
||||||
data = super().attributes(data)
|
data = super().attributes(data)
|
||||||
if data.get("lastConnEvent", {}).get("category", "") == "DISCONNECTED":
|
if data.get("lastConnEvent", {}).get("category", "") == "DISCONNECTED":
|
||||||
data["parameters"]["machMode"].value = "0"
|
data["parameters"]["machMode"].value = "0"
|
||||||
|
@ -10,5 +12,5 @@ class Appliance(ApplianceBase):
|
||||||
data["pause"] = data["parameters"]["machMode"] == "3"
|
data["pause"] = data["parameters"]["machMode"] == "3"
|
||||||
return data
|
return data
|
||||||
|
|
||||||
def settings(self, settings):
|
def settings(self, settings: Dict[str, Any]) -> Dict[str, Any]:
|
||||||
return settings
|
return settings
|
||||||
|
|
|
@ -1,8 +1,10 @@
|
||||||
|
from typing import Any, Dict
|
||||||
|
|
||||||
from pyhon.appliances.base import ApplianceBase
|
from pyhon.appliances.base import ApplianceBase
|
||||||
|
|
||||||
|
|
||||||
class Appliance(ApplianceBase):
|
class Appliance(ApplianceBase):
|
||||||
def attributes(self, data):
|
def attributes(self, data: Dict[str, Any]) -> Dict[str, Any]:
|
||||||
data = super().attributes(data)
|
data = super().attributes(data)
|
||||||
if data.get("lastConnEvent", {}).get("category", "") == "DISCONNECTED":
|
if data.get("lastConnEvent", {}).get("category", "") == "DISCONNECTED":
|
||||||
data["parameters"]["machMode"].value = "0"
|
data["parameters"]["machMode"].value = "0"
|
||||||
|
@ -10,5 +12,5 @@ class Appliance(ApplianceBase):
|
||||||
data["pause"] = data["parameters"]["machMode"] == "3"
|
data["pause"] = data["parameters"]["machMode"] == "3"
|
||||||
return data
|
return data
|
||||||
|
|
||||||
def settings(self, settings):
|
def settings(self, settings: Dict[str, Any]) -> Dict[str, Any]:
|
||||||
return settings
|
return settings
|
||||||
|
|
|
@ -7,7 +7,7 @@ from pyhon.helper import str_to_float
|
||||||
class HonAttribute:
|
class HonAttribute:
|
||||||
_LOCK_TIMEOUT: Final = 10
|
_LOCK_TIMEOUT: Final = 10
|
||||||
|
|
||||||
def __init__(self, data):
|
def __init__(self, data: Dict[str, str] | str):
|
||||||
self._value: str = ""
|
self._value: str = ""
|
||||||
self._last_update: Optional[datetime] = None
|
self._last_update: Optional[datetime] = None
|
||||||
self._lock_timestamp: Optional[datetime] = None
|
self._lock_timestamp: Optional[datetime] = None
|
||||||
|
@ -22,7 +22,7 @@ class HonAttribute:
|
||||||
return self._value
|
return self._value
|
||||||
|
|
||||||
@value.setter
|
@value.setter
|
||||||
def value(self, value) -> None:
|
def value(self, value: str) -> None:
|
||||||
self._value = value
|
self._value = value
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
import asyncio
|
import asyncio
|
||||||
from contextlib import suppress
|
from contextlib import suppress
|
||||||
from copy import copy
|
from copy import copy
|
||||||
from typing import Dict, Any, Optional, TYPE_CHECKING, List
|
from typing import Dict, Any, Optional, TYPE_CHECKING, List, Collection
|
||||||
|
|
||||||
from pyhon.commands import HonCommand
|
from pyhon.commands import HonCommand
|
||||||
from pyhon.parameter.fixed import HonParameterFixed
|
from pyhon.parameter.fixed import HonParameterFixed
|
||||||
|
@ -15,7 +15,7 @@ if TYPE_CHECKING:
|
||||||
class HonCommandLoader:
|
class HonCommandLoader:
|
||||||
"""Loads and parses hOn command data"""
|
"""Loads and parses hOn command data"""
|
||||||
|
|
||||||
def __init__(self, api, appliance):
|
def __init__(self, api: "HonAPI", appliance: "HonAppliance") -> None:
|
||||||
self._api_commands: Dict[str, Any] = {}
|
self._api_commands: Dict[str, Any] = {}
|
||||||
self._favourites: List[Dict[str, Any]] = []
|
self._favourites: List[Dict[str, Any]] = []
|
||||||
self._command_history: List[Dict[str, Any]] = []
|
self._command_history: List[Dict[str, Any]] = []
|
||||||
|
@ -52,7 +52,7 @@ class HonCommandLoader:
|
||||||
"""Get command additional data"""
|
"""Get command additional data"""
|
||||||
return self._additional_data
|
return self._additional_data
|
||||||
|
|
||||||
async def load_commands(self):
|
async def load_commands(self) -> None:
|
||||||
"""Trigger loading of command data"""
|
"""Trigger loading of command data"""
|
||||||
await self._load_data()
|
await self._load_data()
|
||||||
self._appliance_data = self._api_commands.pop("applianceModel")
|
self._appliance_data = self._api_commands.pop("applianceModel")
|
||||||
|
@ -60,17 +60,17 @@ class HonCommandLoader:
|
||||||
self._add_favourites()
|
self._add_favourites()
|
||||||
self._recover_last_command_states()
|
self._recover_last_command_states()
|
||||||
|
|
||||||
async def _load_commands(self):
|
async def _load_commands(self) -> None:
|
||||||
self._api_commands = await self._api.load_commands(self._appliance)
|
self._api_commands = await self._api.load_commands(self._appliance)
|
||||||
|
|
||||||
async def _load_favourites(self):
|
async def _load_favourites(self) -> None:
|
||||||
self._favourites = await self._api.load_favourites(self._appliance)
|
self._favourites = await self._api.load_favourites(self._appliance)
|
||||||
|
|
||||||
async def _load_command_history(self):
|
async def _load_command_history(self) -> None:
|
||||||
self._command_history = await self._api.load_command_history(self._appliance)
|
self._command_history = await self._api.load_command_history(self._appliance)
|
||||||
|
|
||||||
async def _load_data(self):
|
async def _load_data(self) -> None:
|
||||||
"""Request parallel all relevant data"""
|
"""Callback parallel all relevant data"""
|
||||||
await asyncio.gather(
|
await asyncio.gather(
|
||||||
*[
|
*[
|
||||||
self._load_commands(),
|
self._load_commands(),
|
||||||
|
@ -102,14 +102,24 @@ class HonCommandLoader:
|
||||||
self._commands = {c.name: c for c in commands}
|
self._commands = {c.name: c for c in commands}
|
||||||
|
|
||||||
def _parse_command(
|
def _parse_command(
|
||||||
self, data: Dict[str, Any] | str, command_name: str, **kwargs
|
self,
|
||||||
|
data: Dict[str, Any] | str,
|
||||||
|
command_name: str,
|
||||||
|
categories: Optional[Dict[str, "HonCommand"]] = None,
|
||||||
|
category_name: str = "",
|
||||||
) -> Optional[HonCommand]:
|
) -> Optional[HonCommand]:
|
||||||
"""Try to crate HonCommand object"""
|
"""Try to crate HonCommand object"""
|
||||||
if not isinstance(data, dict):
|
if not isinstance(data, dict):
|
||||||
self._additional_data[command_name] = data
|
self._additional_data[command_name] = data
|
||||||
return None
|
return None
|
||||||
if self._is_command(data):
|
if self._is_command(data):
|
||||||
return HonCommand(command_name, data, self._appliance, **kwargs)
|
return HonCommand(
|
||||||
|
command_name,
|
||||||
|
data,
|
||||||
|
self._appliance,
|
||||||
|
category_name=category_name,
|
||||||
|
categories=categories,
|
||||||
|
)
|
||||||
if category := self._parse_categories(data, command_name):
|
if category := self._parse_categories(data, command_name):
|
||||||
return category
|
return category
|
||||||
return None
|
return None
|
||||||
|
@ -120,8 +130,9 @@ class HonCommandLoader:
|
||||||
"""Parse categories and create reference to other"""
|
"""Parse categories and create reference to other"""
|
||||||
categories: Dict[str, HonCommand] = {}
|
categories: Dict[str, HonCommand] = {}
|
||||||
for category, value in data.items():
|
for category, value in data.items():
|
||||||
kwargs = {"category_name": category, "categories": categories}
|
if command := self._parse_command(
|
||||||
if command := self._parse_command(value, command_name, **kwargs):
|
value, command_name, category_name=category, categories=categories
|
||||||
|
):
|
||||||
categories[self._clean_name(category)] = command
|
categories[self._clean_name(category)] = command
|
||||||
if categories:
|
if categories:
|
||||||
# setParameters should be at first place
|
# setParameters should be at first place
|
||||||
|
|
|
@ -9,6 +9,7 @@ from pyhon.parameter.fixed import HonParameterFixed
|
||||||
from pyhon.parameter.program import HonParameterProgram
|
from pyhon.parameter.program import HonParameterProgram
|
||||||
from pyhon.parameter.range import HonParameterRange
|
from pyhon.parameter.range import HonParameterRange
|
||||||
from pyhon.rules import HonRuleSet
|
from pyhon.rules import HonRuleSet
|
||||||
|
from pyhon.typedefs import Parameter
|
||||||
|
|
||||||
if TYPE_CHECKING:
|
if TYPE_CHECKING:
|
||||||
from pyhon import HonAPI
|
from pyhon import HonAPI
|
||||||
|
@ -43,7 +44,7 @@ class HonCommand:
|
||||||
return f"{self._name} command"
|
return f"{self._name} command"
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def name(self):
|
def name(self) -> str:
|
||||||
return self._name
|
return self._name
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
@ -57,7 +58,7 @@ class HonCommand:
|
||||||
return self._appliance
|
return self._appliance
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def data(self):
|
def data(self) -> Dict[str, Any]:
|
||||||
return self._data
|
return self._data
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
@ -79,14 +80,16 @@ class HonCommand:
|
||||||
def parameter_value(self) -> Dict[str, Union[str, float]]:
|
def parameter_value(self) -> Dict[str, Union[str, float]]:
|
||||||
return {n: p.value for n, p in self._parameters.items()}
|
return {n: p.value for n, p in self._parameters.items()}
|
||||||
|
|
||||||
def _load_parameters(self, attributes):
|
def _load_parameters(self, attributes: Dict[str, Dict[str, Any]]) -> None:
|
||||||
for key, items in attributes.items():
|
for key, items in attributes.items():
|
||||||
for name, data in items.items():
|
for name, data in items.items():
|
||||||
self._create_parameters(data, name, key)
|
self._create_parameters(data, name, key)
|
||||||
for rule in self._rules:
|
for rule in self._rules:
|
||||||
rule.patch()
|
rule.patch()
|
||||||
|
|
||||||
def _create_parameters(self, data: Dict, name: str, parameter: str) -> None:
|
def _create_parameters(
|
||||||
|
self, data: Dict[str, Any], name: str, parameter: str
|
||||||
|
) -> None:
|
||||||
if name == "zoneMap" and self._appliance.zone:
|
if name == "zoneMap" and self._appliance.zone:
|
||||||
data["default"] = self._appliance.zone
|
data["default"] = self._appliance.zone
|
||||||
if data.get("category") == "rule":
|
if data.get("category") == "rule":
|
||||||
|
@ -147,7 +150,7 @@ class HonCommand:
|
||||||
)
|
)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def _more_options(first: HonParameter, second: HonParameter):
|
def _more_options(first: Parameter, second: Parameter) -> Parameter:
|
||||||
if isinstance(first, HonParameterFixed) and not isinstance(
|
if isinstance(first, HonParameterFixed) and not isinstance(
|
||||||
second, HonParameterFixed
|
second, HonParameterFixed
|
||||||
):
|
):
|
||||||
|
@ -157,8 +160,8 @@ class HonCommand:
|
||||||
return first
|
return first
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def available_settings(self) -> Dict[str, HonParameter]:
|
def available_settings(self) -> Dict[str, Parameter]:
|
||||||
result: Dict[str, HonParameter] = {}
|
result: Dict[str, Parameter] = {}
|
||||||
for command in self.categories.values():
|
for command in self.categories.values():
|
||||||
for name, parameter in command.parameters.items():
|
for name, parameter in command.parameters.items():
|
||||||
if name in result:
|
if name in result:
|
||||||
|
|
|
@ -3,7 +3,8 @@ import logging
|
||||||
from datetime import datetime
|
from datetime import datetime
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from pprint import pformat
|
from pprint import pformat
|
||||||
from typing import Dict, Optional, Any, List, no_type_check
|
from types import TracebackType
|
||||||
|
from typing import Dict, Optional, Any, List, no_type_check, Type
|
||||||
|
|
||||||
from aiohttp import ClientSession
|
from aiohttp import ClientSession
|
||||||
from typing_extensions import Self
|
from typing_extensions import Self
|
||||||
|
@ -36,7 +37,12 @@ class HonAPI:
|
||||||
async def __aenter__(self) -> Self:
|
async def __aenter__(self) -> Self:
|
||||||
return await self.create()
|
return await self.create()
|
||||||
|
|
||||||
async def __aexit__(self, exc_type, exc_val, exc_tb) -> None:
|
async def __aexit__(
|
||||||
|
self,
|
||||||
|
exc_type: Optional[Type[BaseException]],
|
||||||
|
exc: Optional[BaseException],
|
||||||
|
traceback: Optional[TracebackType],
|
||||||
|
) -> None:
|
||||||
await self.close()
|
await self.close()
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
@ -46,13 +52,13 @@ class HonAPI:
|
||||||
return self._hon.auth
|
return self._hon.auth
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def _hon(self):
|
def _hon(self) -> HonConnectionHandler:
|
||||||
if self._hon_handler is None:
|
if self._hon_handler is None:
|
||||||
raise exceptions.NoAuthenticationException
|
raise exceptions.NoAuthenticationException
|
||||||
return self._hon_handler
|
return self._hon_handler
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def _hon_anonymous(self):
|
def _hon_anonymous(self) -> HonAnonymousConnectionHandler:
|
||||||
if self._hon_anonymous_handler is None:
|
if self._hon_anonymous_handler is None:
|
||||||
raise exceptions.NoAuthenticationException
|
raise exceptions.NoAuthenticationException
|
||||||
return self._hon_anonymous_handler
|
return self._hon_anonymous_handler
|
||||||
|
@ -74,7 +80,7 @@ class HonAPI:
|
||||||
return []
|
return []
|
||||||
|
|
||||||
async def load_commands(self, appliance: HonAppliance) -> Dict[str, Any]:
|
async def load_commands(self, appliance: HonAppliance) -> Dict[str, Any]:
|
||||||
params: Dict = {
|
params: Dict[str, str | int] = {
|
||||||
"applianceType": appliance.appliance_type,
|
"applianceType": appliance.appliance_type,
|
||||||
"applianceModelId": appliance.appliance_model_id,
|
"applianceModelId": appliance.appliance_model_id,
|
||||||
"macAddress": appliance.mac_address,
|
"macAddress": appliance.mac_address,
|
||||||
|
@ -90,7 +96,7 @@ class HonAPI:
|
||||||
params["series"] = series
|
params["series"] = series
|
||||||
url: str = f"{const.API_URL}/commands/v1/retrieve"
|
url: str = f"{const.API_URL}/commands/v1/retrieve"
|
||||||
async with self._hon.get(url, params=params) as response:
|
async with self._hon.get(url, params=params) as response:
|
||||||
result: Dict = (await response.json()).get("payload", {})
|
result: Dict[str, Any] = (await response.json()).get("payload", {})
|
||||||
if not result or result.pop("resultCode") != "0":
|
if not result or result.pop("resultCode") != "0":
|
||||||
_LOGGER.error(await response.json())
|
_LOGGER.error(await response.json())
|
||||||
return {}
|
return {}
|
||||||
|
@ -103,7 +109,7 @@ class HonAPI:
|
||||||
f"{const.API_URL}/commands/v1/appliance/{appliance.mac_address}/history"
|
f"{const.API_URL}/commands/v1/appliance/{appliance.mac_address}/history"
|
||||||
)
|
)
|
||||||
async with self._hon.get(url) as response:
|
async with self._hon.get(url) as response:
|
||||||
result: Dict = await response.json()
|
result: Dict[str, Any] = await response.json()
|
||||||
if not result or not result.get("payload"):
|
if not result or not result.get("payload"):
|
||||||
return []
|
return []
|
||||||
return result["payload"]["history"]
|
return result["payload"]["history"]
|
||||||
|
@ -113,34 +119,34 @@ class HonAPI:
|
||||||
f"{const.API_URL}/commands/v1/appliance/{appliance.mac_address}/favourite"
|
f"{const.API_URL}/commands/v1/appliance/{appliance.mac_address}/favourite"
|
||||||
)
|
)
|
||||||
async with self._hon.get(url) as response:
|
async with self._hon.get(url) as response:
|
||||||
result: Dict = await response.json()
|
result: Dict[str, Any] = await response.json()
|
||||||
if not result or not result.get("payload"):
|
if not result or not result.get("payload"):
|
||||||
return []
|
return []
|
||||||
return result["payload"]["favourites"]
|
return result["payload"]["favourites"]
|
||||||
|
|
||||||
async def load_last_activity(self, appliance: HonAppliance) -> Dict[str, Any]:
|
async def load_last_activity(self, appliance: HonAppliance) -> Dict[str, Any]:
|
||||||
url: str = f"{const.API_URL}/commands/v1/retrieve-last-activity"
|
url: str = f"{const.API_URL}/commands/v1/retrieve-last-activity"
|
||||||
params: Dict = {"macAddress": appliance.mac_address}
|
params: Dict[str, str] = {"macAddress": appliance.mac_address}
|
||||||
async with self._hon.get(url, params=params) as response:
|
async with self._hon.get(url, params=params) as response:
|
||||||
result: Dict = await response.json()
|
result: Dict[str, Any] = await response.json()
|
||||||
if result and (activity := result.get("attributes")):
|
if result and (activity := result.get("attributes")):
|
||||||
return activity
|
return activity
|
||||||
return {}
|
return {}
|
||||||
|
|
||||||
async def load_appliance_data(self, appliance: HonAppliance) -> Dict[str, Any]:
|
async def load_appliance_data(self, appliance: HonAppliance) -> Dict[str, Any]:
|
||||||
url: str = f"{const.API_URL}/commands/v1/appliance-model"
|
url: str = f"{const.API_URL}/commands/v1/appliance-model"
|
||||||
params: Dict = {
|
params: Dict[str, str] = {
|
||||||
"code": appliance.code,
|
"code": appliance.code,
|
||||||
"macAddress": appliance.mac_address,
|
"macAddress": appliance.mac_address,
|
||||||
}
|
}
|
||||||
async with self._hon.get(url, params=params) as response:
|
async with self._hon.get(url, params=params) as response:
|
||||||
result: Dict = await response.json()
|
result: Dict[str, Any] = await response.json()
|
||||||
if result:
|
if result:
|
||||||
return result.get("payload", {}).get("applianceModel", {})
|
return result.get("payload", {}).get("applianceModel", {})
|
||||||
return {}
|
return {}
|
||||||
|
|
||||||
async def load_attributes(self, appliance: HonAppliance) -> Dict[str, Any]:
|
async def load_attributes(self, appliance: HonAppliance) -> Dict[str, Any]:
|
||||||
params: Dict = {
|
params: Dict[str, str] = {
|
||||||
"macAddress": appliance.mac_address,
|
"macAddress": appliance.mac_address,
|
||||||
"applianceType": appliance.appliance_type,
|
"applianceType": appliance.appliance_type,
|
||||||
"category": "CYCLE",
|
"category": "CYCLE",
|
||||||
|
@ -150,7 +156,7 @@ class HonAPI:
|
||||||
return (await response.json()).get("payload", {})
|
return (await response.json()).get("payload", {})
|
||||||
|
|
||||||
async def load_statistics(self, appliance: HonAppliance) -> Dict[str, Any]:
|
async def load_statistics(self, appliance: HonAppliance) -> Dict[str, Any]:
|
||||||
params: Dict = {
|
params: Dict[str, str] = {
|
||||||
"macAddress": appliance.mac_address,
|
"macAddress": appliance.mac_address,
|
||||||
"applianceType": appliance.appliance_type,
|
"applianceType": appliance.appliance_type,
|
||||||
}
|
}
|
||||||
|
@ -168,11 +174,11 @@ class HonAPI:
|
||||||
self,
|
self,
|
||||||
appliance: HonAppliance,
|
appliance: HonAppliance,
|
||||||
command: str,
|
command: str,
|
||||||
parameters: Dict,
|
parameters: Dict[str, Any],
|
||||||
ancillary_parameters: Dict,
|
ancillary_parameters: Dict[str, Any],
|
||||||
) -> bool:
|
) -> bool:
|
||||||
now: str = datetime.utcnow().isoformat()
|
now: str = datetime.utcnow().isoformat()
|
||||||
data: Dict = {
|
data: Dict[str, Any] = {
|
||||||
"macAddress": appliance.mac_address,
|
"macAddress": appliance.mac_address,
|
||||||
"timestamp": f"{now[:-3]}Z",
|
"timestamp": f"{now[:-3]}Z",
|
||||||
"commandName": command,
|
"commandName": command,
|
||||||
|
@ -190,7 +196,7 @@ class HonAPI:
|
||||||
}
|
}
|
||||||
url: str = f"{const.API_URL}/commands/v1/send"
|
url: str = f"{const.API_URL}/commands/v1/send"
|
||||||
async with self._hon.post(url, json=data) as response:
|
async with self._hon.post(url, json=data) as response:
|
||||||
json_data: Dict = await response.json()
|
json_data: Dict[str, Any] = await response.json()
|
||||||
if json_data.get("payload", {}).get("resultCode") == "0":
|
if json_data.get("payload", {}).get("resultCode") == "0":
|
||||||
return True
|
return True
|
||||||
_LOGGER.error(await response.text())
|
_LOGGER.error(await response.text())
|
||||||
|
@ -200,7 +206,7 @@ class HonAPI:
|
||||||
async def appliance_configuration(self) -> Dict[str, Any]:
|
async def appliance_configuration(self) -> Dict[str, Any]:
|
||||||
url: str = f"{const.API_URL}/config/v1/program-list-rules"
|
url: str = f"{const.API_URL}/config/v1/program-list-rules"
|
||||||
async with self._hon_anonymous.get(url) as response:
|
async with self._hon_anonymous.get(url) as response:
|
||||||
result: Dict = await response.json()
|
result: Dict[str, Any] = await response.json()
|
||||||
if result and (data := result.get("payload")):
|
if result and (data := result.get("payload")):
|
||||||
return data
|
return data
|
||||||
return {}
|
return {}
|
||||||
|
@ -209,7 +215,7 @@ class HonAPI:
|
||||||
self, language: str = "en", beta: bool = True
|
self, language: str = "en", beta: bool = True
|
||||||
) -> Dict[str, Any]:
|
) -> Dict[str, Any]:
|
||||||
url: str = f"{const.API_URL}/app-config"
|
url: str = f"{const.API_URL}/app-config"
|
||||||
payload_data: Dict = {
|
payload_data: Dict[str, str | int] = {
|
||||||
"languageCode": language,
|
"languageCode": language,
|
||||||
"beta": beta,
|
"beta": beta,
|
||||||
"appVersion": const.APP_VERSION,
|
"appVersion": const.APP_VERSION,
|
||||||
|
@ -237,12 +243,12 @@ class HonAPI:
|
||||||
|
|
||||||
|
|
||||||
class TestAPI(HonAPI):
|
class TestAPI(HonAPI):
|
||||||
def __init__(self, path):
|
def __init__(self, path: Path):
|
||||||
super().__init__()
|
super().__init__()
|
||||||
self._anonymous = True
|
self._anonymous = True
|
||||||
self._path: Path = path
|
self._path: Path = path
|
||||||
|
|
||||||
def _load_json(self, appliance: HonAppliance, file) -> Dict[str, Any]:
|
def _load_json(self, appliance: HonAppliance, file: str) -> Dict[str, Any]:
|
||||||
directory = f"{appliance.appliance_type}_{appliance.appliance_model_id}".lower()
|
directory = f"{appliance.appliance_type}_{appliance.appliance_model_id}".lower()
|
||||||
path = f"{self._path}/{directory}/{file}.json"
|
path = f"{self._path}/{directory}/{file}.json"
|
||||||
with open(path, "r", encoding="utf-8") as json_file:
|
with open(path, "r", encoding="utf-8") as json_file:
|
||||||
|
@ -288,7 +294,7 @@ class TestAPI(HonAPI):
|
||||||
self,
|
self,
|
||||||
appliance: HonAppliance,
|
appliance: HonAppliance,
|
||||||
command: str,
|
command: str,
|
||||||
parameters: Dict,
|
parameters: Dict[str, Any],
|
||||||
ancillary_parameters: Dict,
|
ancillary_parameters: Dict[str, Any],
|
||||||
) -> bool:
|
) -> bool:
|
||||||
return True
|
return True
|
||||||
|
|
|
@ -6,14 +6,16 @@ import urllib
|
||||||
from contextlib import suppress
|
from contextlib import suppress
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
from datetime import datetime, timedelta
|
from datetime import datetime, timedelta
|
||||||
from typing import Dict, Optional
|
from typing import Dict, Optional, Any
|
||||||
from urllib import parse
|
from urllib import parse
|
||||||
from urllib.parse import quote
|
from urllib.parse import quote
|
||||||
|
|
||||||
|
import aiohttp
|
||||||
from aiohttp import ClientResponse
|
from aiohttp import ClientResponse
|
||||||
from yarl import URL
|
from yarl import URL
|
||||||
|
|
||||||
from pyhon import const, exceptions
|
from pyhon import const, exceptions
|
||||||
|
from pyhon.connection.device import HonDevice
|
||||||
from pyhon.connection.handler.auth import HonAuthConnectionHandler
|
from pyhon.connection.handler.auth import HonAuthConnectionHandler
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
@ -25,14 +27,20 @@ class HonLoginData:
|
||||||
email: str = ""
|
email: str = ""
|
||||||
password: str = ""
|
password: str = ""
|
||||||
fw_uid: str = ""
|
fw_uid: str = ""
|
||||||
loaded: Optional[Dict] = None
|
loaded: Optional[Dict[str, Any]] = None
|
||||||
|
|
||||||
|
|
||||||
class HonAuth:
|
class HonAuth:
|
||||||
_TOKEN_EXPIRES_AFTER_HOURS = 8
|
_TOKEN_EXPIRES_AFTER_HOURS = 8
|
||||||
_TOKEN_EXPIRE_WARNING_HOURS = 7
|
_TOKEN_EXPIRE_WARNING_HOURS = 7
|
||||||
|
|
||||||
def __init__(self, session, email, password, device) -> None:
|
def __init__(
|
||||||
|
self,
|
||||||
|
session: aiohttp.ClientSession,
|
||||||
|
email: str,
|
||||||
|
password: str,
|
||||||
|
device: HonDevice,
|
||||||
|
) -> None:
|
||||||
self._session = session
|
self._session = session
|
||||||
self._request = HonAuthConnectionHandler(session)
|
self._request = HonAuthConnectionHandler(session)
|
||||||
self._login_data = HonLoginData()
|
self._login_data = HonLoginData()
|
||||||
|
@ -120,7 +128,7 @@ class HonAuth:
|
||||||
await self._error_logger(response)
|
await self._error_logger(response)
|
||||||
return new_location
|
return new_location
|
||||||
|
|
||||||
async def _handle_redirects(self, login_url) -> str:
|
async def _handle_redirects(self, login_url: str) -> str:
|
||||||
redirect1 = await self._manual_redirect(login_url)
|
redirect1 = await self._manual_redirect(login_url)
|
||||||
redirect2 = await self._manual_redirect(redirect1)
|
redirect2 = await self._manual_redirect(redirect1)
|
||||||
return f"{redirect2}&System=IoT_Mobile_App&RegistrationSubChannel=hOn"
|
return f"{redirect2}&System=IoT_Mobile_App&RegistrationSubChannel=hOn"
|
||||||
|
|
|
@ -32,12 +32,14 @@ class HonDevice:
|
||||||
def mobile_id(self) -> str:
|
def mobile_id(self) -> str:
|
||||||
return self._mobile_id
|
return self._mobile_id
|
||||||
|
|
||||||
def get(self, mobile: bool = False) -> Dict:
|
def get(self, mobile: bool = False) -> Dict[str, str | int]:
|
||||||
result = {
|
result: Dict[str, str | int] = {
|
||||||
"appVersion": self.app_version,
|
"appVersion": self.app_version,
|
||||||
"mobileId": self.mobile_id,
|
"mobileId": self.mobile_id,
|
||||||
"os": self.os,
|
"os": self.os,
|
||||||
"osVersion": self.os_version,
|
"osVersion": self.os_version,
|
||||||
"deviceModel": self.device_model,
|
"deviceModel": self.device_model,
|
||||||
}
|
}
|
||||||
return (result | {"mobileOs": result.pop("os")}) if mobile else result
|
if mobile:
|
||||||
|
result |= {"mobileOs": result.pop("os", "")}
|
||||||
|
return result
|
||||||
|
|
|
@ -1,19 +1,24 @@
|
||||||
import logging
|
import logging
|
||||||
from collections.abc import AsyncIterator
|
from collections.abc import AsyncIterator
|
||||||
from contextlib import asynccontextmanager
|
from contextlib import asynccontextmanager
|
||||||
from typing import Callable, Dict
|
from typing import Dict, Any
|
||||||
|
|
||||||
|
import aiohttp
|
||||||
|
|
||||||
from pyhon import const
|
from pyhon import const
|
||||||
from pyhon.connection.handler.base import ConnectionHandler
|
from pyhon.connection.handler.base import ConnectionHandler
|
||||||
|
from pyhon.typedefs import Callback
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class HonAnonymousConnectionHandler(ConnectionHandler):
|
class HonAnonymousConnectionHandler(ConnectionHandler):
|
||||||
_HEADERS: Dict = ConnectionHandler._HEADERS | {"x-api-key": const.API_KEY}
|
_HEADERS: Dict[str, str] = ConnectionHandler._HEADERS | {"x-api-key": const.API_KEY}
|
||||||
|
|
||||||
@asynccontextmanager
|
@asynccontextmanager
|
||||||
async def _intercept(self, method: Callable, *args, **kwargs) -> AsyncIterator:
|
async def _intercept(
|
||||||
|
self, method: Callback, *args: Any, **kwargs: Any
|
||||||
|
) -> AsyncIterator[aiohttp.ClientResponse]:
|
||||||
kwargs["headers"] = kwargs.pop("headers", {}) | self._HEADERS
|
kwargs["headers"] = kwargs.pop("headers", {}) | self._HEADERS
|
||||||
async with method(*args, **kwargs) as response:
|
async with method(*args, **kwargs) as response:
|
||||||
if response.status == 403:
|
if response.status == 403:
|
||||||
|
|
|
@ -1,12 +1,13 @@
|
||||||
import logging
|
import logging
|
||||||
from collections.abc import AsyncIterator
|
from collections.abc import AsyncIterator
|
||||||
from contextlib import asynccontextmanager
|
from contextlib import asynccontextmanager
|
||||||
from typing import Optional, Callable, List, Tuple
|
from typing import Optional, List, Tuple, Any
|
||||||
|
|
||||||
import aiohttp
|
import aiohttp
|
||||||
|
|
||||||
from pyhon import const
|
from pyhon import const
|
||||||
from pyhon.connection.handler.base import ConnectionHandler
|
from pyhon.connection.handler.base import ConnectionHandler
|
||||||
|
from pyhon.typedefs import Callback
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
@ -28,9 +29,9 @@ class HonAuthConnectionHandler(ConnectionHandler):
|
||||||
|
|
||||||
@asynccontextmanager
|
@asynccontextmanager
|
||||||
async def _intercept(
|
async def _intercept(
|
||||||
self, method: Callable, *args, loop: int = 0, **kwargs
|
self, method: Callback, *args: Any, **kwargs: Any
|
||||||
) -> AsyncIterator:
|
) -> AsyncIterator[aiohttp.ClientResponse]:
|
||||||
kwargs["headers"] = kwargs.pop("headers", {}) | self._HEADERS
|
kwargs["headers"] = kwargs.pop("headers", {}) | self._HEADERS
|
||||||
async with method(*args, **kwargs) as response:
|
async with method(*args, **kwargs) as response:
|
||||||
self._called_urls.append((response.status, response.request_info.url))
|
self._called_urls.append((response.status, str(response.request_info.url)))
|
||||||
yield response
|
yield response
|
||||||
|
|
|
@ -1,18 +1,20 @@
|
||||||
import logging
|
import logging
|
||||||
from collections.abc import AsyncIterator
|
from collections.abc import AsyncIterator
|
||||||
from contextlib import asynccontextmanager
|
from contextlib import asynccontextmanager
|
||||||
from typing import Optional, Callable, Dict
|
from types import TracebackType
|
||||||
|
from typing import Optional, Dict, Type, Any, Protocol
|
||||||
|
|
||||||
import aiohttp
|
import aiohttp
|
||||||
from typing_extensions import Self
|
from typing_extensions import Self
|
||||||
|
|
||||||
from pyhon import const, exceptions
|
from pyhon import const, exceptions
|
||||||
|
from pyhon.typedefs import Callback
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
class ConnectionHandler:
|
class ConnectionHandler:
|
||||||
_HEADERS: Dict = {
|
_HEADERS: Dict[str, str] = {
|
||||||
"user-agent": const.USER_AGENT,
|
"user-agent": const.USER_AGENT,
|
||||||
"Content-Type": "application/json",
|
"Content-Type": "application/json",
|
||||||
}
|
}
|
||||||
|
@ -24,32 +26,49 @@ class ConnectionHandler:
|
||||||
async def __aenter__(self) -> Self:
|
async def __aenter__(self) -> Self:
|
||||||
return await self.create()
|
return await self.create()
|
||||||
|
|
||||||
async def __aexit__(self, exc_type, exc_val, exc_tb) -> None:
|
async def __aexit__(
|
||||||
|
self,
|
||||||
|
exc_type: Optional[Type[BaseException]],
|
||||||
|
exc: Optional[BaseException],
|
||||||
|
traceback: Optional[TracebackType],
|
||||||
|
) -> None:
|
||||||
await self.close()
|
await self.close()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def session(self) -> aiohttp.ClientSession:
|
||||||
|
if self._session is None:
|
||||||
|
raise exceptions.NoSessionException
|
||||||
|
return self._session
|
||||||
|
|
||||||
async def create(self) -> Self:
|
async def create(self) -> Self:
|
||||||
if self._create_session:
|
if self._create_session:
|
||||||
self._session = aiohttp.ClientSession()
|
self._session = aiohttp.ClientSession()
|
||||||
return self
|
return self
|
||||||
|
|
||||||
@asynccontextmanager
|
@asynccontextmanager
|
||||||
def _intercept(self, method: Callable, *args, loop: int = 0, **kwargs):
|
def _intercept(
|
||||||
|
self, method: Callback, *args: Any, loop: int = 0, **kwargs: Any
|
||||||
|
) -> AsyncIterator[aiohttp.ClientResponse]:
|
||||||
raise NotImplementedError
|
raise NotImplementedError
|
||||||
|
|
||||||
@asynccontextmanager
|
@asynccontextmanager
|
||||||
async def get(self, *args, **kwargs) -> AsyncIterator[aiohttp.ClientResponse]:
|
async def get(
|
||||||
|
self, *args: Any, **kwargs: Any
|
||||||
|
) -> AsyncIterator[aiohttp.ClientResponse]:
|
||||||
if self._session is None:
|
if self._session is None:
|
||||||
raise exceptions.NoSessionException()
|
raise exceptions.NoSessionException()
|
||||||
response: aiohttp.ClientResponse
|
response: aiohttp.ClientResponse
|
||||||
async with self._intercept(self._session.get, *args, **kwargs) as response:
|
async with self._intercept(self._session.get, *args, **kwargs) as response: # type: ignore[arg-type]
|
||||||
yield response
|
yield response
|
||||||
|
|
||||||
@asynccontextmanager
|
@asynccontextmanager
|
||||||
async def post(self, *args, **kwargs) -> AsyncIterator[aiohttp.ClientResponse]:
|
async def post(
|
||||||
|
self, *args: Any, **kwargs: Any
|
||||||
|
) -> AsyncIterator[aiohttp.ClientResponse]:
|
||||||
if self._session is None:
|
if self._session is None:
|
||||||
raise exceptions.NoSessionException()
|
raise exceptions.NoSessionException()
|
||||||
response: aiohttp.ClientResponse
|
response: aiohttp.ClientResponse
|
||||||
async with self._intercept(self._session.post, *args, **kwargs) as response:
|
async with self._intercept(self._session.post, *args, **kwargs) as response: # type: ignore[arg-type]
|
||||||
yield response
|
yield response
|
||||||
|
|
||||||
async def close(self) -> None:
|
async def close(self) -> None:
|
||||||
|
|
|
@ -2,7 +2,7 @@ import json
|
||||||
import logging
|
import logging
|
||||||
from collections.abc import AsyncIterator
|
from collections.abc import AsyncIterator
|
||||||
from contextlib import asynccontextmanager
|
from contextlib import asynccontextmanager
|
||||||
from typing import Optional, Callable, Dict
|
from typing import Optional, Dict, Any
|
||||||
|
|
||||||
import aiohttp
|
import aiohttp
|
||||||
from typing_extensions import Self
|
from typing_extensions import Self
|
||||||
|
@ -11,6 +11,7 @@ from pyhon.connection.auth import HonAuth
|
||||||
from pyhon.connection.device import HonDevice
|
from pyhon.connection.device import HonDevice
|
||||||
from pyhon.connection.handler.base import ConnectionHandler
|
from pyhon.connection.handler.base import ConnectionHandler
|
||||||
from pyhon.exceptions import HonAuthenticationError, NoAuthenticationException
|
from pyhon.exceptions import HonAuthenticationError, NoAuthenticationException
|
||||||
|
from pyhon.typedefs import Callback
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
@ -41,10 +42,10 @@ class HonConnectionHandler(ConnectionHandler):
|
||||||
|
|
||||||
async def create(self) -> Self:
|
async def create(self) -> Self:
|
||||||
await super().create()
|
await super().create()
|
||||||
self._auth = HonAuth(self._session, self._email, self._password, self._device)
|
self._auth = HonAuth(self.session, self._email, self._password, self._device)
|
||||||
return self
|
return self
|
||||||
|
|
||||||
async def _check_headers(self, headers: Dict) -> Dict:
|
async def _check_headers(self, headers: Dict[str, str]) -> Dict[str, str]:
|
||||||
if not (self.auth.cognito_token and self.auth.id_token):
|
if not (self.auth.cognito_token and self.auth.id_token):
|
||||||
await self.auth.authenticate()
|
await self.auth.authenticate()
|
||||||
headers["cognito-token"] = self.auth.cognito_token
|
headers["cognito-token"] = self.auth.cognito_token
|
||||||
|
@ -53,18 +54,16 @@ class HonConnectionHandler(ConnectionHandler):
|
||||||
|
|
||||||
@asynccontextmanager
|
@asynccontextmanager
|
||||||
async def _intercept(
|
async def _intercept(
|
||||||
self, method: Callable, *args, loop: int = 0, **kwargs
|
self, method: Callback, *args: Any, loop: int = 0, **kwargs: Dict[str, str]
|
||||||
) -> AsyncIterator:
|
) -> AsyncIterator[aiohttp.ClientResponse]:
|
||||||
kwargs["headers"] = await self._check_headers(kwargs.get("headers", {}))
|
kwargs["headers"] = await self._check_headers(kwargs.get("headers", {}))
|
||||||
async with method(*args, **kwargs) as response:
|
async with method(args[0], *args[1:], **kwargs) as response:
|
||||||
if (
|
if (
|
||||||
self.auth.token_expires_soon or response.status in [401, 403]
|
self.auth.token_expires_soon or response.status in [401, 403]
|
||||||
) and loop == 0:
|
) and loop == 0:
|
||||||
_LOGGER.info("Try refreshing token...")
|
_LOGGER.info("Try refreshing token...")
|
||||||
await self.auth.refresh()
|
await self.auth.refresh()
|
||||||
async with self._intercept(
|
async with self._intercept(method, loop=loop + 1, **kwargs) as result:
|
||||||
method, *args, loop=loop + 1, **kwargs
|
|
||||||
) as result:
|
|
||||||
yield result
|
yield result
|
||||||
elif (
|
elif (
|
||||||
self.auth.token_is_expired or response.status in [401, 403]
|
self.auth.token_is_expired or response.status in [401, 403]
|
||||||
|
@ -76,9 +75,7 @@ class HonConnectionHandler(ConnectionHandler):
|
||||||
await response.text(),
|
await response.text(),
|
||||||
)
|
)
|
||||||
await self.create()
|
await self.create()
|
||||||
async with self._intercept(
|
async with self._intercept(method, loop=loop + 1, **kwargs) as result:
|
||||||
method, *args, loop=loop + 1, **kwargs
|
|
||||||
) as result:
|
|
||||||
yield result
|
yield result
|
||||||
elif loop >= 2:
|
elif loop >= 2:
|
||||||
_LOGGER.error(
|
_LOGGER.error(
|
||||||
|
|
|
@ -5,7 +5,7 @@ import shutil
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from typing import TYPE_CHECKING, List, Tuple
|
from typing import TYPE_CHECKING, List, Tuple
|
||||||
|
|
||||||
from pyhon import helper
|
from pyhon import printer
|
||||||
|
|
||||||
if TYPE_CHECKING:
|
if TYPE_CHECKING:
|
||||||
from pyhon.appliance import HonAppliance
|
from pyhon.appliance import HonAppliance
|
||||||
|
@ -38,7 +38,7 @@ async def load_data(appliance: "HonAppliance", topic: str) -> Tuple[str, str]:
|
||||||
return topic, await getattr(appliance.api, f"load_{topic}")(appliance)
|
return topic, await getattr(appliance.api, f"load_{topic}")(appliance)
|
||||||
|
|
||||||
|
|
||||||
def write_to_json(data: str, topic: str, path: Path, anonymous: bool = False):
|
def write_to_json(data: str, topic: str, path: Path, anonymous: bool = False) -> Path:
|
||||||
json_data = json.dumps(data, indent=4)
|
json_data = json.dumps(data, indent=4)
|
||||||
if anonymous:
|
if anonymous:
|
||||||
json_data = anonymize_data(json_data)
|
json_data = anonymize_data(json_data)
|
||||||
|
@ -65,7 +65,9 @@ async def appliance_data(
|
||||||
return [write_to_json(data, topic, path, anonymous) for topic, data in api_data]
|
return [write_to_json(data, topic, path, anonymous) for topic, data in api_data]
|
||||||
|
|
||||||
|
|
||||||
async def zip_archive(appliance: "HonAppliance", path: Path, anonymous: bool = False):
|
async def zip_archive(
|
||||||
|
appliance: "HonAppliance", path: Path, anonymous: bool = False
|
||||||
|
) -> str:
|
||||||
data = await appliance_data(appliance, path, anonymous)
|
data = await appliance_data(appliance, path, anonymous)
|
||||||
archive = data[0].parent
|
archive = data[0].parent
|
||||||
shutil.make_archive(str(archive.parent), "zip", archive)
|
shutil.make_archive(str(archive.parent), "zip", archive)
|
||||||
|
@ -73,7 +75,7 @@ async def zip_archive(appliance: "HonAppliance", path: Path, anonymous: bool = F
|
||||||
return f"{archive.stem}.zip"
|
return f"{archive.stem}.zip"
|
||||||
|
|
||||||
|
|
||||||
def yaml_export(appliance: "HonAppliance", anonymous=False) -> str:
|
def yaml_export(appliance: "HonAppliance", anonymous: bool = False) -> str:
|
||||||
data = {
|
data = {
|
||||||
"attributes": appliance.attributes.copy(),
|
"attributes": appliance.attributes.copy(),
|
||||||
"appliance": appliance.info,
|
"appliance": appliance.info,
|
||||||
|
@ -89,10 +91,10 @@ def yaml_export(appliance: "HonAppliance", anonymous=False) -> str:
|
||||||
data.get("appliance", {}).pop(sensible, None)
|
data.get("appliance", {}).pop(sensible, None)
|
||||||
data = {
|
data = {
|
||||||
"data": data,
|
"data": data,
|
||||||
"commands": helper.create_command(appliance.commands),
|
"commands": printer.create_command(appliance.commands),
|
||||||
"rules": helper.create_rules(appliance.commands),
|
"rules": printer.create_rules(appliance.commands),
|
||||||
}
|
}
|
||||||
result = helper.pretty_print(data)
|
result = printer.pretty_print(data)
|
||||||
if anonymous:
|
if anonymous:
|
||||||
result = anonymize_data(result)
|
result = anonymize_data(result)
|
||||||
return result
|
return result
|
||||||
|
|
|
@ -1,80 +1,3 @@
|
||||||
def key_print(data, key="", start=True):
|
|
||||||
result = ""
|
|
||||||
if isinstance(data, list):
|
|
||||||
for i, value in enumerate(data):
|
|
||||||
result += key_print(value, key=f"{key}.{i}", start=False)
|
|
||||||
elif isinstance(data, dict):
|
|
||||||
for k, value in sorted(data.items()):
|
|
||||||
result += key_print(value, key=k if start else f"{key}.{k}", start=False)
|
|
||||||
else:
|
|
||||||
result += f"{key}: {data}\n"
|
|
||||||
return result
|
|
||||||
|
|
||||||
|
|
||||||
# yaml.dump() would be done the same, but needs an additional dependency...
|
|
||||||
def pretty_print(data, key="", intend=0, is_list=False, whitespace=" "):
|
|
||||||
result = ""
|
|
||||||
if isinstance(data, list):
|
|
||||||
if key:
|
|
||||||
result += f"{whitespace * intend}{'- ' if is_list else ''}{key}:\n"
|
|
||||||
intend += 1
|
|
||||||
for i, value in enumerate(data):
|
|
||||||
result += pretty_print(
|
|
||||||
value, intend=intend, is_list=True, whitespace=whitespace
|
|
||||||
)
|
|
||||||
elif isinstance(data, dict):
|
|
||||||
if key:
|
|
||||||
result += f"{whitespace * intend}{'- ' if is_list else ''}{key}:\n"
|
|
||||||
intend += 1
|
|
||||||
for i, (key, value) in enumerate(sorted(data.items())):
|
|
||||||
if is_list and not i:
|
|
||||||
result += pretty_print(
|
|
||||||
value, key=key, intend=intend, is_list=True, whitespace=whitespace
|
|
||||||
)
|
|
||||||
elif is_list:
|
|
||||||
result += pretty_print(
|
|
||||||
value, key=key, intend=intend + 1, whitespace=whitespace
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
result += pretty_print(
|
|
||||||
value, key=key, intend=intend, whitespace=whitespace
|
|
||||||
)
|
|
||||||
else:
|
|
||||||
result += f"{whitespace * intend}{'- ' if is_list else ''}{key}{': ' if key else ''}{data}\n"
|
|
||||||
return result
|
|
||||||
|
|
||||||
|
|
||||||
def create_command(commands, concat=False):
|
|
||||||
result = {}
|
|
||||||
for name, command in commands.items():
|
|
||||||
for parameter, data in command.available_settings.items():
|
|
||||||
if data.typology == "enum":
|
|
||||||
value = data.values
|
|
||||||
elif data.typology == "range":
|
|
||||||
value = {"min": data.min, "max": data.max, "step": data.step}
|
|
||||||
else:
|
|
||||||
continue
|
|
||||||
if not concat:
|
|
||||||
result.setdefault(name, {})[parameter] = value
|
|
||||||
else:
|
|
||||||
result[f"{name}.{parameter}"] = value
|
|
||||||
return result
|
|
||||||
|
|
||||||
|
|
||||||
def create_rules(commands, concat=False):
|
|
||||||
result = {}
|
|
||||||
for name, command in commands.items():
|
|
||||||
for parameter, data in command.available_settings.items():
|
|
||||||
value = data.triggers
|
|
||||||
if not value:
|
|
||||||
continue
|
|
||||||
if not concat:
|
|
||||||
result.setdefault(name, {})[parameter] = value
|
|
||||||
else:
|
|
||||||
result[f"{name}.{parameter}"] = value
|
|
||||||
return result
|
|
||||||
|
|
||||||
|
|
||||||
def str_to_float(string: str | float) -> float:
|
def str_to_float(string: str | float) -> float:
|
||||||
try:
|
try:
|
||||||
return int(string)
|
return int(string)
|
||||||
|
|
|
@ -70,11 +70,11 @@ class Hon:
|
||||||
return self._appliances
|
return self._appliances
|
||||||
|
|
||||||
@appliances.setter
|
@appliances.setter
|
||||||
def appliances(self, appliances) -> None:
|
def appliances(self, appliances: List[HonAppliance]) -> None:
|
||||||
self._appliances = appliances
|
self._appliances = appliances
|
||||||
|
|
||||||
async def _create_appliance(
|
async def _create_appliance(
|
||||||
self, appliance_data: Dict[str, Any], api: HonAPI, zone=0
|
self, appliance_data: Dict[str, Any], api: HonAPI, zone: int = 0
|
||||||
) -> None:
|
) -> None:
|
||||||
appliance = HonAppliance(api, appliance_data, zone=zone)
|
appliance = HonAppliance(api, appliance_data, zone=zone)
|
||||||
if appliance.mac_address == "":
|
if appliance.mac_address == "":
|
||||||
|
|
|
@ -12,7 +12,9 @@ class HonParameter:
|
||||||
self._mandatory: int = attributes.get("mandatory", 0)
|
self._mandatory: int = attributes.get("mandatory", 0)
|
||||||
self._value: str | float = ""
|
self._value: str | float = ""
|
||||||
self._group: str = group
|
self._group: str = group
|
||||||
self._triggers: Dict[str, List[Tuple[Callable, "HonRule"]]] = {}
|
self._triggers: Dict[
|
||||||
|
str, List[Tuple[Callable[["HonRule"], None], "HonRule"]]
|
||||||
|
] = {}
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def key(self) -> str:
|
def key(self) -> str:
|
||||||
|
@ -51,20 +53,22 @@ class HonParameter:
|
||||||
def group(self) -> str:
|
def group(self) -> str:
|
||||||
return self._group
|
return self._group
|
||||||
|
|
||||||
def add_trigger(self, value, func, data):
|
def add_trigger(
|
||||||
|
self, value: str, func: Callable[["HonRule"], None], data: "HonRule"
|
||||||
|
) -> None:
|
||||||
if self._value == value:
|
if self._value == value:
|
||||||
func(data)
|
func(data)
|
||||||
self._triggers.setdefault(value, []).append((func, data))
|
self._triggers.setdefault(value, []).append((func, data))
|
||||||
|
|
||||||
def check_trigger(self, value) -> None:
|
def check_trigger(self, value: str | float) -> None:
|
||||||
if str(value) in self._triggers:
|
if str(value) in self._triggers:
|
||||||
for trigger in self._triggers[str(value)]:
|
for trigger in self._triggers[str(value)]:
|
||||||
func, args = trigger
|
func, args = trigger
|
||||||
func(args)
|
func(args)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def triggers(self):
|
def triggers(self) -> Dict[str, Any]:
|
||||||
result = {}
|
result: Dict[str, Any] = {}
|
||||||
for value, rules in self._triggers.items():
|
for value, rules in self._triggers.items():
|
||||||
for _, rule in rules:
|
for _, rule in rules:
|
||||||
if rule.extras:
|
if rule.extras:
|
||||||
|
|
|
@ -3,7 +3,7 @@ from typing import Dict, Any, List
|
||||||
from pyhon.parameter.base import HonParameter
|
from pyhon.parameter.base import HonParameter
|
||||||
|
|
||||||
|
|
||||||
def clean_value(value):
|
def clean_value(value: str | float) -> str:
|
||||||
return str(value).strip("[]").replace("|", "_").lower()
|
return str(value).strip("[]").replace("|", "_").lower()
|
||||||
|
|
||||||
|
|
||||||
|
@ -24,7 +24,7 @@ class HonParameterEnum(HonParameter):
|
||||||
return [clean_value(value) for value in self._values]
|
return [clean_value(value) for value in self._values]
|
||||||
|
|
||||||
@values.setter
|
@values.setter
|
||||||
def values(self, values) -> None:
|
def values(self, values: List[str]) -> None:
|
||||||
self._values = values
|
self._values = values
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
|
|
@ -36,7 +36,7 @@ class HonParameterProgram(HonParameterEnum):
|
||||||
return sorted(values)
|
return sorted(values)
|
||||||
|
|
||||||
@values.setter
|
@values.setter
|
||||||
def values(self, values) -> None:
|
def values(self, values: List[str]) -> None:
|
||||||
return
|
return
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
@ -50,5 +50,5 @@ class HonParameterProgram(HonParameterEnum):
|
||||||
}
|
}
|
||||||
return dict(sorted(values.items()))
|
return dict(sorted(values.items()))
|
||||||
|
|
||||||
def set_value(self, value: str):
|
def set_value(self, value: str) -> None:
|
||||||
self._value = value
|
self._value = value
|
||||||
|
|
|
@ -13,7 +13,7 @@ class HonParameterRange(HonParameter):
|
||||||
self._default: float = str_to_float(attributes.get("defaultValue", self.min))
|
self._default: float = str_to_float(attributes.get("defaultValue", self.min))
|
||||||
self._value: float = self._default
|
self._value: float = self._default
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self) -> str:
|
||||||
return f"{self.__class__} (<{self.key}> [{self.min} - {self.max}])"
|
return f"{self.__class__} (<{self.key}> [{self.min} - {self.max}])"
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
|
94
pyhon/printer.py
Normal file
94
pyhon/printer.py
Normal file
|
@ -0,0 +1,94 @@
|
||||||
|
from typing import Dict, Any, TYPE_CHECKING, List
|
||||||
|
|
||||||
|
from pyhon.parameter.enum import HonParameterEnum
|
||||||
|
from pyhon.parameter.range import HonParameterRange
|
||||||
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from pyhon.commands import HonCommand
|
||||||
|
|
||||||
|
|
||||||
|
def key_print(data: Any, key: str = "", start: bool = True) -> str:
|
||||||
|
result = ""
|
||||||
|
if isinstance(data, list):
|
||||||
|
for i, value in enumerate(data):
|
||||||
|
result += key_print(value, key=f"{key}.{i}", start=False)
|
||||||
|
elif isinstance(data, dict):
|
||||||
|
for k, value in sorted(data.items()):
|
||||||
|
result += key_print(value, key=k if start else f"{key}.{k}", start=False)
|
||||||
|
else:
|
||||||
|
result += f"{key}: {data}\n"
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
|
# yaml.dump() would be done the same, but needs an additional dependency...
|
||||||
|
def pretty_print(
|
||||||
|
data: Any,
|
||||||
|
key: str = "",
|
||||||
|
intend: int = 0,
|
||||||
|
is_list: bool = False,
|
||||||
|
whitespace: str = " ",
|
||||||
|
) -> str:
|
||||||
|
result = ""
|
||||||
|
if isinstance(data, list):
|
||||||
|
if key:
|
||||||
|
result += f"{whitespace * intend}{'- ' if is_list else ''}{key}:\n"
|
||||||
|
intend += 1
|
||||||
|
for i, value in enumerate(data):
|
||||||
|
result += pretty_print(
|
||||||
|
value, intend=intend, is_list=True, whitespace=whitespace
|
||||||
|
)
|
||||||
|
elif isinstance(data, dict):
|
||||||
|
if key:
|
||||||
|
result += f"{whitespace * intend}{'- ' if is_list else ''}{key}:\n"
|
||||||
|
intend += 1
|
||||||
|
for i, (key, value) in enumerate(sorted(data.items())):
|
||||||
|
if is_list and not i:
|
||||||
|
result += pretty_print(
|
||||||
|
value, key=key, intend=intend, is_list=True, whitespace=whitespace
|
||||||
|
)
|
||||||
|
elif is_list:
|
||||||
|
result += pretty_print(
|
||||||
|
value, key=key, intend=intend + 1, whitespace=whitespace
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
result += pretty_print(
|
||||||
|
value, key=key, intend=intend, whitespace=whitespace
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
result += f"{whitespace * intend}{'- ' if is_list else ''}{key}{': ' if key else ''}{data}\n"
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
|
def create_command(
|
||||||
|
commands: Dict[str, "HonCommand"], concat: bool = False
|
||||||
|
) -> Dict[str, Any]:
|
||||||
|
result: Dict[str, Any] = {}
|
||||||
|
for name, command in commands.items():
|
||||||
|
for parameter, data in command.available_settings.items():
|
||||||
|
if isinstance(data, HonParameterEnum):
|
||||||
|
value: List[str] | Dict[str, str | float] = data.values
|
||||||
|
elif isinstance(data, HonParameterRange):
|
||||||
|
value = {"min": data.min, "max": data.max, "step": data.step}
|
||||||
|
else:
|
||||||
|
continue
|
||||||
|
if not concat:
|
||||||
|
result.setdefault(name, {})[parameter] = value
|
||||||
|
else:
|
||||||
|
result[f"{name}.{parameter}"] = value
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
|
def create_rules(
|
||||||
|
commands: Dict[str, "HonCommand"], concat: bool = False
|
||||||
|
) -> Dict[str, Any]:
|
||||||
|
result: Dict[str, Any] = {}
|
||||||
|
for name, command in commands.items():
|
||||||
|
for parameter, data in command.available_settings.items():
|
||||||
|
value = data.triggers
|
||||||
|
if not value:
|
||||||
|
continue
|
||||||
|
if not concat:
|
||||||
|
result.setdefault(name, {})[parameter] = value
|
||||||
|
else:
|
||||||
|
result[f"{name}.{parameter}"] = value
|
||||||
|
return result
|
|
@ -6,6 +6,7 @@ from pyhon.parameter.range import HonParameterRange
|
||||||
|
|
||||||
if TYPE_CHECKING:
|
if TYPE_CHECKING:
|
||||||
from pyhon.commands import HonCommand
|
from pyhon.commands import HonCommand
|
||||||
|
from pyhon.parameter.base import HonParameter
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
|
@ -18,18 +19,24 @@ class HonRule:
|
||||||
|
|
||||||
|
|
||||||
class HonRuleSet:
|
class HonRuleSet:
|
||||||
def __init__(self, command: "HonCommand", rule):
|
def __init__(self, command: "HonCommand", rule: Dict[str, Any]):
|
||||||
self._command: "HonCommand" = command
|
self._command: "HonCommand" = command
|
||||||
self._rules: Dict[str, List[HonRule]] = {}
|
self._rules: Dict[str, List[HonRule]] = {}
|
||||||
self._parse_rule(rule)
|
self._parse_rule(rule)
|
||||||
|
|
||||||
def _parse_rule(self, rule):
|
def _parse_rule(self, rule: Dict[str, Any]) -> None:
|
||||||
for param_key, params in rule.items():
|
for param_key, params in rule.items():
|
||||||
param_key = self._command.appliance.options.get(param_key, param_key)
|
param_key = self._command.appliance.options.get(param_key, param_key)
|
||||||
for trigger_key, trigger_data in params.items():
|
for trigger_key, trigger_data in params.items():
|
||||||
self._parse_conditions(param_key, trigger_key, trigger_data)
|
self._parse_conditions(param_key, trigger_key, trigger_data)
|
||||||
|
|
||||||
def _parse_conditions(self, param_key, trigger_key, trigger_data, extra=None):
|
def _parse_conditions(
|
||||||
|
self,
|
||||||
|
param_key: str,
|
||||||
|
trigger_key: str,
|
||||||
|
trigger_data: Dict[str, Any],
|
||||||
|
extra: Optional[Dict[str, str]] = None,
|
||||||
|
) -> None:
|
||||||
trigger_key = trigger_key.replace("@", "")
|
trigger_key = trigger_key.replace("@", "")
|
||||||
trigger_key = self._command.appliance.options.get(trigger_key, trigger_key)
|
trigger_key = self._command.appliance.options.get(trigger_key, trigger_key)
|
||||||
for multi_trigger_value, param_data in trigger_data.items():
|
for multi_trigger_value, param_data in trigger_data.items():
|
||||||
|
@ -46,16 +53,21 @@ class HonRuleSet:
|
||||||
self._parse_conditions(param_key, extra_key, extra_data, extra)
|
self._parse_conditions(param_key, extra_key, extra_data, extra)
|
||||||
|
|
||||||
def _create_rule(
|
def _create_rule(
|
||||||
self, param_key, trigger_key, trigger_value, param_data, extras=None
|
self,
|
||||||
):
|
param_key: str,
|
||||||
|
trigger_key: str,
|
||||||
|
trigger_value: str,
|
||||||
|
param_data: Dict[str, Any],
|
||||||
|
extras: Optional[Dict[str, str]] = None,
|
||||||
|
) -> None:
|
||||||
if param_data.get("fixedValue") == f"@{param_key}":
|
if param_data.get("fixedValue") == f"@{param_key}":
|
||||||
return
|
return
|
||||||
self._rules.setdefault(trigger_key, []).append(
|
self._rules.setdefault(trigger_key, []).append(
|
||||||
HonRule(trigger_key, trigger_value, param_key, param_data, extras)
|
HonRule(trigger_key, trigger_value, param_key, param_data, extras)
|
||||||
)
|
)
|
||||||
|
|
||||||
def _duplicate_for_extra_conditions(self):
|
def _duplicate_for_extra_conditions(self) -> None:
|
||||||
new = {}
|
new: Dict[str, List[HonRule]] = {}
|
||||||
for rules in self._rules.values():
|
for rules in self._rules.values():
|
||||||
for rule in rules:
|
for rule in rules:
|
||||||
if rule.extras is None:
|
if rule.extras is None:
|
||||||
|
@ -71,8 +83,8 @@ class HonRuleSet:
|
||||||
for rule in rules:
|
for rule in rules:
|
||||||
self._rules.setdefault(key, []).append(rule)
|
self._rules.setdefault(key, []).append(rule)
|
||||||
|
|
||||||
def _add_trigger(self, parameter, data):
|
def _add_trigger(self, parameter: "HonParameter", data: HonRule) -> None:
|
||||||
def apply(rule: HonRule):
|
def apply(rule: HonRule) -> None:
|
||||||
if rule.extras is not None:
|
if rule.extras is not None:
|
||||||
for key, value in rule.extras.items():
|
for key, value in rule.extras.items():
|
||||||
if str(self._command.parameters.get(key)) != str(value):
|
if str(self._command.parameters.get(key)) != str(value):
|
||||||
|
@ -96,10 +108,10 @@ class HonRuleSet:
|
||||||
|
|
||||||
parameter.add_trigger(data.trigger_value, apply, data)
|
parameter.add_trigger(data.trigger_value, apply, data)
|
||||||
|
|
||||||
def patch(self):
|
def patch(self) -> None:
|
||||||
self._duplicate_for_extra_conditions()
|
self._duplicate_for_extra_conditions()
|
||||||
for name, parameter in self._command.parameters.items():
|
for name, parameter in self._command.parameters.items():
|
||||||
if name not in self._rules:
|
if name not in self._rules:
|
||||||
continue
|
continue
|
||||||
for data in self._rules.get(name):
|
for data in self._rules.get(name, []):
|
||||||
self._add_trigger(parameter, data)
|
self._add_trigger(parameter, data)
|
||||||
|
|
27
pyhon/typedefs.py
Normal file
27
pyhon/typedefs.py
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
from typing import Union, Any, TYPE_CHECKING, Protocol
|
||||||
|
|
||||||
|
import aiohttp
|
||||||
|
from yarl import URL
|
||||||
|
|
||||||
|
if TYPE_CHECKING:
|
||||||
|
from pyhon.parameter.base import HonParameter
|
||||||
|
from pyhon.parameter.enum import HonParameterEnum
|
||||||
|
from pyhon.parameter.fixed import HonParameterFixed
|
||||||
|
from pyhon.parameter.program import HonParameterProgram
|
||||||
|
from pyhon.parameter.range import HonParameterRange
|
||||||
|
|
||||||
|
|
||||||
|
class Callback(Protocol):
|
||||||
|
def __call__(
|
||||||
|
self, url: str | URL, *args: Any, **kwargs: Any
|
||||||
|
) -> aiohttp.client._RequestContextManager:
|
||||||
|
...
|
||||||
|
|
||||||
|
|
||||||
|
Parameter = Union[
|
||||||
|
"HonParameter",
|
||||||
|
"HonParameterRange",
|
||||||
|
"HonParameterEnum",
|
||||||
|
"HonParameterFixed",
|
||||||
|
"HonParameterProgram",
|
||||||
|
]
|
Loading…
Reference in a new issue