mirror of
https://github.com/python-kasa/python-kasa.git
synced 2025-01-08 22:07:06 +00:00
Get child emeters with CLI (#623)
* Get child emeters with CLI * Avoid extra IO when not que querying the child emeter
This commit is contained in:
parent
2d8a8d9511
commit
cfe694e5de
18
kasa/cli.py
18
kasa/cli.py
@ -602,14 +602,27 @@ async def raw_command(dev: SmartDevice, module, command, parameters):
|
|||||||
|
|
||||||
@cli.command()
|
@cli.command()
|
||||||
@pass_dev
|
@pass_dev
|
||||||
|
@click.option("--index", type=int, required=False)
|
||||||
|
@click.option("--name", type=str, required=False)
|
||||||
@click.option("--year", type=click.DateTime(["%Y"]), default=None, required=False)
|
@click.option("--year", type=click.DateTime(["%Y"]), default=None, required=False)
|
||||||
@click.option("--month", type=click.DateTime(["%Y-%m"]), default=None, required=False)
|
@click.option("--month", type=click.DateTime(["%Y-%m"]), default=None, required=False)
|
||||||
@click.option("--erase", is_flag=True)
|
@click.option("--erase", is_flag=True)
|
||||||
async def emeter(dev: SmartDevice, year, month, erase):
|
async def emeter(dev: SmartDevice, index: int, name: str, year, month, erase):
|
||||||
"""Query emeter for historical consumption.
|
"""Query emeter for historical consumption.
|
||||||
|
|
||||||
Daily and monthly data provided in CSV format.
|
Daily and monthly data provided in CSV format.
|
||||||
"""
|
"""
|
||||||
|
if index is not None or name is not None:
|
||||||
|
if not dev.is_strip:
|
||||||
|
echo("Index and name are only for power strips!")
|
||||||
|
return
|
||||||
|
|
||||||
|
dev = cast(SmartStrip, dev)
|
||||||
|
if index is not None:
|
||||||
|
dev = dev.get_plug_by_index(index)
|
||||||
|
elif name:
|
||||||
|
dev = dev.get_plug_by_name(name)
|
||||||
|
|
||||||
echo("[bold]== Emeter ==[/bold]")
|
echo("[bold]== Emeter ==[/bold]")
|
||||||
if not dev.has_emeter:
|
if not dev.has_emeter:
|
||||||
echo("Device has no emeter")
|
echo("Device has no emeter")
|
||||||
@ -629,6 +642,9 @@ async def emeter(dev: SmartDevice, year, month, erase):
|
|||||||
usage_data = await dev.get_emeter_daily(year=month.year, month=month.month)
|
usage_data = await dev.get_emeter_daily(year=month.year, month=month.month)
|
||||||
else:
|
else:
|
||||||
# Call with no argument outputs summary data and returns
|
# Call with no argument outputs summary data and returns
|
||||||
|
if index is not None or name is not None:
|
||||||
|
emeter_status = await dev.get_emeter_realtime()
|
||||||
|
else:
|
||||||
emeter_status = dev.emeter_realtime
|
emeter_status = dev.emeter_realtime
|
||||||
|
|
||||||
echo("Current: %s A" % emeter_status["current"])
|
echo("Current: %s A" % emeter_status["current"])
|
||||||
|
@ -128,7 +128,7 @@ def get_device_class_from_sys_info(info: Dict[str, Any]) -> Type[SmartDevice]:
|
|||||||
|
|
||||||
def get_device_class_from_family(device_type: str) -> Optional[Type[SmartDevice]]:
|
def get_device_class_from_family(device_type: str) -> Optional[Type[SmartDevice]]:
|
||||||
"""Return the device class from the type name."""
|
"""Return the device class from the type name."""
|
||||||
supported_device_types: dict[str, Type[SmartDevice]] = {
|
supported_device_types: Dict[str, Type[SmartDevice]] = {
|
||||||
"SMART.TAPOPLUG": TapoPlug,
|
"SMART.TAPOPLUG": TapoPlug,
|
||||||
"SMART.TAPOBULB": TapoBulb,
|
"SMART.TAPOBULB": TapoBulb,
|
||||||
"SMART.KASAPLUG": TapoPlug,
|
"SMART.KASAPLUG": TapoPlug,
|
||||||
@ -147,7 +147,7 @@ def get_protocol(
|
|||||||
protocol_transport_key = (
|
protocol_transport_key = (
|
||||||
protocol_name + "." + config.connection_type.encryption_type.value
|
protocol_name + "." + config.connection_type.encryption_type.value
|
||||||
)
|
)
|
||||||
supported_device_protocols: dict[
|
supported_device_protocols: Dict[
|
||||||
str, Tuple[Type[TPLinkProtocol], Type[BaseTransport]]
|
str, Tuple[Type[TPLinkProtocol], Type[BaseTransport]]
|
||||||
] = {
|
] = {
|
||||||
"IOT.XOR": (TPLinkSmartHomeProtocol, _XorTransport),
|
"IOT.XOR": (TPLinkSmartHomeProtocol, _XorTransport),
|
||||||
|
@ -7,6 +7,7 @@ from asyncclick.testing import CliRunner
|
|||||||
from kasa import (
|
from kasa import (
|
||||||
AuthenticationException,
|
AuthenticationException,
|
||||||
Credentials,
|
Credentials,
|
||||||
|
EmeterStatus,
|
||||||
SmartDevice,
|
SmartDevice,
|
||||||
TPLinkSmartHomeProtocol,
|
TPLinkSmartHomeProtocol,
|
||||||
UnsupportedDeviceException,
|
UnsupportedDeviceException,
|
||||||
@ -104,6 +105,25 @@ async def test_emeter(dev: SmartDevice, mocker):
|
|||||||
|
|
||||||
assert "== Emeter ==" in res.output
|
assert "== Emeter ==" in res.output
|
||||||
|
|
||||||
|
if not dev.is_strip:
|
||||||
|
res = await runner.invoke(emeter, ["--index", "0"], obj=dev)
|
||||||
|
assert "Index and name are only for power strips!" in res.output
|
||||||
|
res = await runner.invoke(emeter, ["--name", "mock"], obj=dev)
|
||||||
|
assert "Index and name are only for power strips!" in res.output
|
||||||
|
|
||||||
|
if dev.is_strip and len(dev.children) > 0:
|
||||||
|
realtime_emeter = mocker.patch.object(dev.children[0], "get_emeter_realtime")
|
||||||
|
realtime_emeter.return_value = EmeterStatus({"voltage_mv": 122066})
|
||||||
|
|
||||||
|
res = await runner.invoke(emeter, ["--index", "0"], obj=dev)
|
||||||
|
assert "Voltage: 122.066 V" in res.output
|
||||||
|
realtime_emeter.assert_called()
|
||||||
|
assert realtime_emeter.call_count == 1
|
||||||
|
|
||||||
|
res = await runner.invoke(emeter, ["--name", dev.children[0].alias], obj=dev)
|
||||||
|
assert "Voltage: 122.066 V" in res.output
|
||||||
|
assert realtime_emeter.call_count == 2
|
||||||
|
|
||||||
monthly = mocker.patch.object(dev, "get_emeter_monthly")
|
monthly = mocker.patch.object(dev, "get_emeter_monthly")
|
||||||
monthly.return_value = {1: 1234}
|
monthly.return_value = {1: 1234}
|
||||||
res = await runner.invoke(emeter, ["--year", "1900"], obj=dev)
|
res = await runner.invoke(emeter, ["--year", "1900"], obj=dev)
|
||||||
|
Loading…
Reference in New Issue
Block a user