mirror of
https://github.com/python-kasa/python-kasa.git
synced 2024-12-22 11:13:34 +00:00
486 lines
16 KiB
Python
Executable File
486 lines
16 KiB
Python
Executable File
"""Module for multi-socket devices (HS300, HS107, KP303, ..)."""
|
|
|
|
from __future__ import annotations
|
|
|
|
import logging
|
|
from collections import defaultdict
|
|
from datetime import datetime, timedelta
|
|
from typing import TYPE_CHECKING, Any
|
|
|
|
from ..device_type import DeviceType
|
|
from ..deviceconfig import DeviceConfig
|
|
from ..emeterstatus import EmeterStatus
|
|
from ..exceptions import KasaException
|
|
from ..feature import Feature
|
|
from ..interfaces import Energy
|
|
from ..module import Module
|
|
from ..protocols import BaseProtocol
|
|
from .iotdevice import (
|
|
IotDevice,
|
|
requires_update,
|
|
)
|
|
from .iotmodule import IotModule
|
|
from .iotplug import IotPlug
|
|
from .modules import Antitheft, Cloud, Countdown, Emeter, Led, Schedule, Time, Usage
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
def merge_sums(dicts: list[dict]) -> dict:
|
|
"""Merge the sum of dicts."""
|
|
total_dict: defaultdict[int, float] = defaultdict(lambda: 0.0)
|
|
for sum_dict in dicts:
|
|
for day, value in sum_dict.items():
|
|
total_dict[day] += value
|
|
return total_dict
|
|
|
|
|
|
class IotStrip(IotDevice):
|
|
r"""Representation of a TP-Link Smart Power Strip.
|
|
|
|
A strip consists of the parent device and its children.
|
|
All methods of the parent act on all children, while the child devices
|
|
share the common API with the :class:`SmartPlug` class.
|
|
|
|
To initialize, you have to await :func:`update()` at least once.
|
|
This will allow accessing the properties using the exposed properties.
|
|
|
|
All changes to the device are done using awaitable methods,
|
|
which will not change the cached values,
|
|
but you must await :func:`update()` separately.
|
|
|
|
Errors reported by the device are raised as :class:`KasaException`\s,
|
|
and should be handled by the user of the library.
|
|
|
|
Examples:
|
|
>>> import asyncio
|
|
>>> strip = IotStrip("127.0.0.1")
|
|
>>> asyncio.run(strip.update())
|
|
>>> strip.alias
|
|
Bedroom Power Strip
|
|
|
|
All methods act on the whole strip:
|
|
|
|
>>> for plug in strip.children:
|
|
>>> print(f"{plug.alias}: {plug.is_on}")
|
|
Plug 1: True
|
|
Plug 2: False
|
|
Plug 3: False
|
|
>>> strip.is_on
|
|
True
|
|
>>> asyncio.run(strip.turn_off())
|
|
>>> asyncio.run(strip.update())
|
|
|
|
Accessing individual plugs can be done using the `children` property:
|
|
|
|
>>> len(strip.children)
|
|
3
|
|
>>> for plug in strip.children:
|
|
>>> print(f"{plug.alias}: {plug.is_on}")
|
|
Plug 1: False
|
|
Plug 2: False
|
|
Plug 3: False
|
|
>>> asyncio.run(strip.children[1].turn_on())
|
|
>>> asyncio.run(strip.update())
|
|
>>> strip.is_on
|
|
True
|
|
|
|
For more examples, see the :class:`Device` class.
|
|
"""
|
|
|
|
def __init__(
|
|
self,
|
|
host: str,
|
|
*,
|
|
config: DeviceConfig | None = None,
|
|
protocol: BaseProtocol | None = None,
|
|
) -> None:
|
|
super().__init__(host=host, config=config, protocol=protocol)
|
|
self.emeter_type = "emeter"
|
|
self._device_type = DeviceType.Strip
|
|
|
|
async def _initialize_modules(self) -> None:
|
|
"""Initialize modules."""
|
|
# Strip has different modules to plug so do not call super
|
|
self.add_module(Module.IotAntitheft, Antitheft(self, "anti_theft"))
|
|
self.add_module(Module.IotSchedule, Schedule(self, "schedule"))
|
|
self.add_module(Module.IotUsage, Usage(self, "schedule"))
|
|
self.add_module(Module.Time, Time(self, "time"))
|
|
self.add_module(Module.IotCountdown, Countdown(self, "countdown"))
|
|
self.add_module(Module.Led, Led(self, "system"))
|
|
self.add_module(Module.IotCloud, Cloud(self, "cnCloud"))
|
|
if self.has_emeter:
|
|
_LOGGER.debug(
|
|
"The device has emeter, querying its information along sysinfo"
|
|
)
|
|
self.add_module(Module.Energy, StripEmeter(self, self.emeter_type))
|
|
|
|
@property # type: ignore
|
|
@requires_update
|
|
def is_on(self) -> bool:
|
|
"""Return if any of the outlets are on."""
|
|
return any(plug.is_on for plug in self.children)
|
|
|
|
async def update(self, update_children: bool = True) -> None:
|
|
"""Update some of the attributes.
|
|
|
|
Needed for methods that are decorated with `requires_update`.
|
|
"""
|
|
# Super initializes modules and features
|
|
await super().update(update_children)
|
|
|
|
initialize_children = not self.children
|
|
# Initialize the child devices during the first update.
|
|
if initialize_children:
|
|
children = self.sys_info["children"]
|
|
_LOGGER.debug("Initializing %s child sockets", len(children))
|
|
self._children = {
|
|
f"{self.mac}_{child['id']}": IotStripPlug(
|
|
self.host, parent=self, child_id=child["id"]
|
|
)
|
|
for child in children
|
|
}
|
|
for child in self._children.values():
|
|
await child._initialize_modules()
|
|
|
|
if update_children:
|
|
for plug in self.children:
|
|
if TYPE_CHECKING:
|
|
assert isinstance(plug, IotStripPlug)
|
|
await plug._update()
|
|
|
|
if not self.features:
|
|
await self._initialize_features()
|
|
|
|
async def _initialize_features(self) -> None:
|
|
"""Initialize common features."""
|
|
# Do not initialize features until children are created
|
|
if not self.children:
|
|
return
|
|
await super()._initialize_features()
|
|
|
|
async def turn_on(self, **kwargs) -> dict:
|
|
"""Turn the strip on."""
|
|
return await self._query_helper("system", "set_relay_state", {"state": 1})
|
|
|
|
async def turn_off(self, **kwargs) -> dict:
|
|
"""Turn the strip off."""
|
|
return await self._query_helper("system", "set_relay_state", {"state": 0})
|
|
|
|
@property # type: ignore
|
|
@requires_update
|
|
def on_since(self) -> datetime | None:
|
|
"""Return the maximum on-time of all outlets."""
|
|
if self.is_off:
|
|
return None
|
|
|
|
return min(plug.on_since for plug in self.children if plug.on_since is not None)
|
|
|
|
|
|
class StripEmeter(IotModule, Energy):
|
|
"""Energy module implementation to aggregate child modules."""
|
|
|
|
_supported = (
|
|
Energy.ModuleFeature.CONSUMPTION_TOTAL
|
|
| Energy.ModuleFeature.PERIODIC_STATS
|
|
| Energy.ModuleFeature.VOLTAGE_CURRENT
|
|
)
|
|
|
|
def supports(self, module_feature: Energy.ModuleFeature) -> bool:
|
|
"""Return True if module supports the feature."""
|
|
return module_feature in self._supported
|
|
|
|
def query(self) -> dict:
|
|
"""Return the base query."""
|
|
return {}
|
|
|
|
@property
|
|
def current_consumption(self) -> float | None:
|
|
"""Get the current power consumption in watts."""
|
|
return sum(
|
|
v if (v := plug.modules[Module.Energy].current_consumption) else 0.0
|
|
for plug in self._device.children
|
|
)
|
|
|
|
async def get_status(self) -> EmeterStatus:
|
|
"""Retrieve current energy readings."""
|
|
emeter_rt = await self._async_get_emeter_sum("get_status", {})
|
|
# Voltage is averaged since each read will result
|
|
# in a slightly different voltage since they are not atomic
|
|
emeter_rt["voltage_mv"] = int(
|
|
emeter_rt["voltage_mv"] / len(self._device.children)
|
|
)
|
|
return EmeterStatus(emeter_rt)
|
|
|
|
async def get_daily_stats(
|
|
self, year: int | None = None, month: int | None = None, kwh: bool = True
|
|
) -> dict:
|
|
"""Retrieve daily statistics for a given month.
|
|
|
|
:param year: year for which to retrieve statistics (default: this year)
|
|
:param month: month for which to retrieve statistics (default: this
|
|
month)
|
|
:param kwh: return usage in kWh (default: True)
|
|
:return: mapping of day of month to value
|
|
"""
|
|
return await self._async_get_emeter_sum(
|
|
"get_daily_stats", {"year": year, "month": month, "kwh": kwh}
|
|
)
|
|
|
|
async def get_monthly_stats(
|
|
self, year: int | None = None, kwh: bool = True
|
|
) -> dict:
|
|
"""Retrieve monthly statistics for a given year.
|
|
|
|
:param year: year for which to retrieve statistics (default: this year)
|
|
:param kwh: return usage in kWh (default: True)
|
|
"""
|
|
return await self._async_get_emeter_sum(
|
|
"get_monthly_stats", {"year": year, "kwh": kwh}
|
|
)
|
|
|
|
async def _async_get_emeter_sum(self, func: str, kwargs: dict[str, Any]) -> dict:
|
|
"""Retrieve emeter stats for a time period from children."""
|
|
return merge_sums(
|
|
[
|
|
await getattr(plug.modules[Module.Energy], func)(**kwargs)
|
|
for plug in self._device.children
|
|
]
|
|
)
|
|
|
|
async def erase_stats(self) -> dict:
|
|
"""Erase energy meter statistics for all plugs."""
|
|
for plug in self._device.children:
|
|
await plug.modules[Module.Energy].erase_stats()
|
|
|
|
return {}
|
|
|
|
@property # type: ignore
|
|
def consumption_this_month(self) -> float | None:
|
|
"""Return this month's energy consumption in kWh."""
|
|
return sum(
|
|
v if (v := plug.modules[Module.Energy].consumption_this_month) else 0.0
|
|
for plug in self._device.children
|
|
)
|
|
|
|
@property # type: ignore
|
|
def consumption_today(self) -> float | None:
|
|
"""Return this month's energy consumption in kWh."""
|
|
return sum(
|
|
v if (v := plug.modules[Module.Energy].consumption_today) else 0.0
|
|
for plug in self._device.children
|
|
)
|
|
|
|
@property # type: ignore
|
|
def consumption_total(self) -> float | None:
|
|
"""Return total energy consumption since reboot in kWh."""
|
|
return sum(
|
|
v if (v := plug.modules[Module.Energy].consumption_total) else 0.0
|
|
for plug in self._device.children
|
|
)
|
|
|
|
@property # type: ignore
|
|
def status(self) -> EmeterStatus:
|
|
"""Return current energy readings."""
|
|
emeter = merge_sums(
|
|
[plug.modules[Module.Energy].status for plug in self._device.children]
|
|
)
|
|
# Voltage is averaged since each read will result
|
|
# in a slightly different voltage since they are not atomic
|
|
emeter["voltage_mv"] = int(emeter["voltage_mv"] / len(self._device.children))
|
|
return EmeterStatus(emeter)
|
|
|
|
@property
|
|
def current(self) -> float | None:
|
|
"""Return the current in A."""
|
|
return self.status.current
|
|
|
|
@property
|
|
def voltage(self) -> float | None:
|
|
"""Get the current voltage in V."""
|
|
return self.status.voltage
|
|
|
|
|
|
class IotStripPlug(IotPlug):
|
|
"""Representation of a single socket in a power strip.
|
|
|
|
This allows you to use the sockets as they were SmartPlug objects.
|
|
Instead of calling an update on any of these, you should call an update
|
|
on the parent device before accessing the properties.
|
|
|
|
The plug inherits (most of) the system information from the parent.
|
|
"""
|
|
|
|
_parent: IotStrip
|
|
|
|
def __init__(self, host: str, parent: IotStrip, child_id: str) -> None:
|
|
super().__init__(host)
|
|
|
|
self._parent = parent
|
|
self.child_id = child_id
|
|
self._last_update = parent._last_update
|
|
self._set_sys_info(parent.sys_info)
|
|
self._device_type = DeviceType.StripSocket
|
|
self.protocol = parent.protocol # Must use the same connection as the parent
|
|
self._on_since: datetime | None = None
|
|
|
|
async def _initialize_modules(self) -> None:
|
|
"""Initialize modules not added in init."""
|
|
if self.has_emeter:
|
|
self.add_module(Module.Energy, Emeter(self, self.emeter_type))
|
|
self.add_module(Module.IotUsage, Usage(self, "schedule"))
|
|
self.add_module(Module.IotAntitheft, Antitheft(self, "anti_theft"))
|
|
self.add_module(Module.IotSchedule, Schedule(self, "schedule"))
|
|
self.add_module(Module.IotCountdown, Countdown(self, "countdown"))
|
|
|
|
async def _initialize_features(self) -> None:
|
|
"""Initialize common features."""
|
|
self._add_feature(
|
|
Feature(
|
|
self,
|
|
id="state",
|
|
name="State",
|
|
attribute_getter="is_on",
|
|
attribute_setter="set_state",
|
|
type=Feature.Type.Switch,
|
|
category=Feature.Category.Primary,
|
|
)
|
|
)
|
|
self._add_feature(
|
|
Feature(
|
|
device=self,
|
|
id="on_since",
|
|
name="On since",
|
|
attribute_getter="on_since",
|
|
icon="mdi:clock",
|
|
category=Feature.Category.Info,
|
|
type=Feature.Type.Sensor,
|
|
)
|
|
)
|
|
|
|
for module in self.modules.values():
|
|
module._initialize_features()
|
|
for module_feat in module._module_features.values():
|
|
self._add_feature(module_feat)
|
|
|
|
async def update(self, update_children: bool = True) -> None:
|
|
"""Query the device to update the data.
|
|
|
|
Needed for properties that are decorated with `requires_update`.
|
|
"""
|
|
await self._update(update_children)
|
|
|
|
async def _update(self, update_children: bool = True) -> None:
|
|
"""Query the device to update the data.
|
|
|
|
Internal implementation to allow patching of public update in the cli
|
|
or test framework.
|
|
"""
|
|
await self._modular_update({})
|
|
for module in self._modules.values():
|
|
await module._post_update_hook()
|
|
|
|
if not self._features:
|
|
await self._initialize_features()
|
|
|
|
def _create_request(
|
|
self,
|
|
target: str,
|
|
cmd: str,
|
|
arg: dict | None = None,
|
|
child_ids: list | None = None,
|
|
) -> dict:
|
|
request: dict[str, Any] = {
|
|
"context": {"child_ids": [self.child_id]},
|
|
target: {cmd: arg},
|
|
}
|
|
return request
|
|
|
|
async def _query_helper(
|
|
self,
|
|
target: str,
|
|
cmd: str,
|
|
arg: dict | None = None,
|
|
child_ids: list | None = None,
|
|
) -> dict:
|
|
"""Override query helper to include the child_ids."""
|
|
return await self._parent._query_helper(
|
|
target, cmd, arg, child_ids=[self.child_id]
|
|
)
|
|
|
|
@property # type: ignore
|
|
@requires_update
|
|
def is_on(self) -> bool:
|
|
"""Return whether device is on."""
|
|
info = self._get_child_info()
|
|
return bool(info["state"])
|
|
|
|
@property # type: ignore
|
|
@requires_update
|
|
def led(self) -> bool:
|
|
"""Return the state of the led.
|
|
|
|
This is always false for subdevices.
|
|
"""
|
|
return False
|
|
|
|
@property # type: ignore
|
|
@requires_update
|
|
def device_id(self) -> str:
|
|
"""Return unique ID for the socket.
|
|
|
|
This is a combination of MAC and child's ID.
|
|
"""
|
|
return f"{self.mac}_{self.child_id}"
|
|
|
|
@property # type: ignore
|
|
@requires_update
|
|
def alias(self) -> str:
|
|
"""Return device name (alias)."""
|
|
info = self._get_child_info()
|
|
return info["alias"]
|
|
|
|
@property # type: ignore
|
|
@requires_update
|
|
def next_action(self) -> dict:
|
|
"""Return next scheduled(?) action."""
|
|
info = self._get_child_info()
|
|
return info["next_action"]
|
|
|
|
@property # type: ignore
|
|
@requires_update
|
|
def on_since(self) -> datetime | None:
|
|
"""Return on-time, if available."""
|
|
if self.is_off:
|
|
self._on_since = None
|
|
return None
|
|
|
|
info = self._get_child_info()
|
|
on_time = info["on_time"]
|
|
|
|
time = self._parent.time
|
|
|
|
on_since = time - timedelta(seconds=on_time)
|
|
if not self._on_since or timedelta(
|
|
seconds=0
|
|
) < on_since - self._on_since > timedelta(seconds=5):
|
|
self._on_since = on_since
|
|
return self._on_since
|
|
|
|
@property # type: ignore
|
|
@requires_update
|
|
def model(self) -> str:
|
|
"""Return device model for a child socket."""
|
|
sys_info = self._parent.sys_info
|
|
return f"Socket for {sys_info['model']}"
|
|
|
|
def _get_child_info(self) -> dict:
|
|
"""Return the subdevice information for this device."""
|
|
for plug in self._parent.sys_info["children"]:
|
|
if plug["id"] == self.child_id:
|
|
return plug
|
|
|
|
raise KasaException(
|
|
f"Unable to find children {self.child_id}"
|
|
) # pragma: no cover
|