Skip to content

Commit

Permalink
Add fan module (#764)
Browse files Browse the repository at this point in the history
Co-authored-by: Steven B <51370195+sdb9696@users.noreply.github.com>
  • Loading branch information
rytilahti and sdb9696 committed Apr 17, 2024
1 parent da441bc commit 700643d
Show file tree
Hide file tree
Showing 8 changed files with 120 additions and 7 deletions.
1 change: 1 addition & 0 deletions kasa/device_type.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ class DeviceType(Enum):
LightStrip = "lightstrip"
Sensor = "sensor"
Hub = "hub"
Fan = "fan"
Unknown = "unknown"

@staticmethod
Expand Down
2 changes: 2 additions & 0 deletions kasa/smart/modules/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@
from .colortemp import ColorTemperatureModule
from .devicemodule import DeviceModule
from .energymodule import EnergyModule
from .fanmodule import FanModule
from .firmware import Firmware
from .humidity import HumiditySensor
from .ledmodule import LedModule
Expand All @@ -30,6 +31,7 @@
"AutoOffModule",
"LedModule",
"Brightness",
"FanModule",
"Firmware",
"CloudModule",
"LightTransitionModule",
Expand Down
66 changes: 66 additions & 0 deletions kasa/smart/modules/fanmodule.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
"""Implementation of fan_control module."""
from typing import TYPE_CHECKING, Dict

from ...feature import Feature, FeatureType
from ..smartmodule import SmartModule

if TYPE_CHECKING:
from ..smartdevice import SmartDevice


class FanModule(SmartModule):
"""Implementation of fan_control module."""

REQUIRED_COMPONENT = "fan_control"

def __init__(self, device: "SmartDevice", module: str):
super().__init__(device, module)

self._add_feature(
Feature(
device,
"Fan speed level",
container=self,
attribute_getter="fan_speed_level",
attribute_setter="set_fan_speed_level",
icon="mdi:fan",
type=FeatureType.Number,
minimum_value=1,
maximum_value=4,
)
)
self._add_feature(
Feature(
device,
"Fan sleep mode",
container=self,
attribute_getter="sleep_mode",
attribute_setter="set_sleep_mode",
icon="mdi:sleep",
type=FeatureType.Switch
)
)

def query(self) -> Dict:
"""Query to execute during the update cycle."""
return {}

@property
def fan_speed_level(self) -> int:
"""Return fan speed level."""
return self.data["fan_speed_level"]

async def set_fan_speed_level(self, level: int):
"""Set fan speed level."""
if level < 1 or level > 4:
raise ValueError("Invalid level, should be in range 1-4.")
return await self.call("set_device_info", {"fan_speed_level": level})

@property
def sleep_mode(self) -> bool:
"""Return sleep mode status."""
return self.data["fan_sleep_mode_on"]

async def set_sleep_mode(self, on: bool):
"""Set sleep mode."""
return await self.call("set_device_info", {"fan_sleep_mode_on": on})
2 changes: 2 additions & 0 deletions kasa/smart/smartchilddevice.py
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,8 @@ def device_type(self) -> DeviceType:
child_device_map = {
"plug.powerstrip.sub-plug": DeviceType.Plug,
"subg.trigger.temp-hmdt-sensor": DeviceType.Sensor,
"kasa.switch.outlet.sub-fan": DeviceType.Fan,
"kasa.switch.outlet.sub-dimmer": DeviceType.Dimmer,
}
dev_type = child_device_map.get(self.sys_info["category"])
if dev_type is None:
Expand Down
7 changes: 3 additions & 4 deletions kasa/smart/smartdevice.py
Original file line number Diff line number Diff line change
Expand Up @@ -314,12 +314,11 @@ def internal_state(self) -> Any:
return self._last_update

def _update_internal_state(self, info):
"""Update internal state.
"""Update the internal info state.
This is used by the parent to push updates to its children
This is used by the parent to push updates to its children.
"""
# TODO: cleanup the _last_update, _info mess.
self._last_update = self._info = info
self._info = info

async def _query_helper(
self, method: str, params: Optional[Dict] = None, child_ids=None
Expand Down
2 changes: 1 addition & 1 deletion kasa/smart/smartmodule.py
Original file line number Diff line number Diff line change
Expand Up @@ -62,7 +62,7 @@ def data(self):
q = self.query()

if not q:
return dev.internal_state["get_device_info"]
return dev.sys_info

q_keys = list(q.keys())
query_key = q_keys[0]
Expand Down
43 changes: 43 additions & 0 deletions kasa/tests/smart/modules/test_fan.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
from pytest_mock import MockerFixture

from kasa import SmartDevice
from kasa.smart.modules import FanModule
from kasa.tests.device_fixtures import parametrize

fan = parametrize(
"has fan", component_filter="fan_control", protocol_filter={"SMART.CHILD"}
)


@fan
async def test_fan_speed(dev: SmartDevice, mocker: MockerFixture):
"""Test fan speed feature."""
fan: FanModule = dev.modules["FanModule"]
level_feature = fan._module_features["fan_speed_level"]
assert level_feature.minimum_value <= level_feature.value <= level_feature.maximum_value

call = mocker.spy(fan, "call")
await fan.set_fan_speed_level(3)
call.assert_called_with("set_device_info", {"fan_sleep_level": 3})

await dev.update()

assert fan.fan_speed_level == 3
assert level_feature.value == 3


@fan
async def test_sleep_mode(dev: SmartDevice, mocker: MockerFixture):
"""Test sleep mode feature."""
fan: FanModule = dev.modules["FanModule"]
sleep_feature = fan._module_features["fan_sleep_mode"]
assert isinstance(sleep_feature.value, bool)

call = mocker.spy(fan, "call")
await fan.set_sleep_mode(True)
call.assert_called_with("set_device_info", {"fan_sleep_mode_on": True})

await dev.update()

assert fan.sleep_mode is True
assert sleep_feature.value is True
4 changes: 2 additions & 2 deletions kasa/tests/test_childdevice.py
Original file line number Diff line number Diff line change
Expand Up @@ -32,8 +32,8 @@ async def test_childdevice_update(dev, dummy_protocol, mocker):

await dev.update()

assert dev._last_update != first._last_update
assert child_list[0] == first._last_update
assert dev._info != first._info
assert child_list[0] == first._info


@strip_smart
Expand Down

0 comments on commit 700643d

Please sign in to comment.