diff --git a/README.rst b/README.rst
index 9f3bc0ccd..75c153b4f 100644
--- a/README.rst
+++ b/README.rst
@@ -27,7 +27,7 @@ Supported devices
- Xiaomi Philips LED Ball Lamp (:class:`miio.philips_bulb`)
- Xiaomi Philips Zhirui Smart LED Bulb E14 Candle Lamp (:class:`miio.philips_bulb`)
- Xiaomi Universal IR Remote Controller (Chuangmi IR) (:class:`miio.chuangmi_ir`)
-- Xiaomi Mi Smart Fan (:class:`miio.fan`)
+- Xiaomi Mi Smart Pedestal Fan (:class:`miio.fan`)
- Xiaomi Mi Air Humidifier (:class:`miio.airhumidifier`)
- Xiaomi Mi Water Purifier (Basic support: Turn on & off) (:class:`miio.waterpurifier`)
- Xiaomi PM2.5 Air Quality Monitor (:class:`miio.airqualitymonitor`)
@@ -55,7 +55,8 @@ Home Assistant support
- `Xiaomi Universal IR Remote Controller `__
- `Xiaomi Mi Air Quality Monitor (PM2.5) `__
- `Xiaomi Mi Home Air Conditioner Companion `__
-- `Xiaomi Mi WiFi Repeater 2 `__
+- `Xiaomi Mi WiFi Repeater 2 `__
+- `Xiaomi Mi Smart Pedestal Fan `__
.. |PyPI version| image:: https://badge.fury.io/py/python-miio.svg
diff --git a/miio/discovery.py b/miio/discovery.py
index 40b4aa0ce..b5cf7d793 100644
--- a/miio/discovery.py
+++ b/miio/discovery.py
@@ -12,6 +12,7 @@
WaterPurifier, WifiSpeaker, WifiRepeater, Yeelight, Fan, )
from .chuangmi_plug import (MODEL_CHUANGMI_PLUG_V1, MODEL_CHUANGMI_PLUG_V3,
MODEL_CHUANGMI_PLUG_M1, )
+from .fan import (MODEL_FAN_V2, MODEL_FAN_V3, )
from .powerstrip import (MODEL_POWER_STRIP_V1, MODEL_POWER_STRIP_V2, )
_LOGGER = logging.getLogger(__name__)
@@ -52,8 +53,8 @@
"xiaomi-repeater-v1": WifiRepeater, # name needs to be checked
"xiaomi-repeater-v3": WifiRepeater, # name needs to be checked
"yeelink-light-": Yeelight,
- "zhimi-fan-v2": Fan,
- "zhimi-fan-v3": Fan,
+ "zhimi-fan-v2": partial(Fan, model=MODEL_FAN_V2),
+ "zhimi-fan-v3": partial(Fan, model=MODEL_FAN_V3),
"lumi-gateway-": lambda x: other_package_info(
x, "https://github.com/Danielhiversen/PyXiaomiGateway")
} # type: Dict[str, Union[Callable, Device]]
diff --git a/miio/fan.py b/miio/fan.py
index bc7287742..be8720a1d 100644
--- a/miio/fan.py
+++ b/miio/fan.py
@@ -5,10 +5,42 @@
import click
from .click_common import command, format_output, EnumType
-from .device import Device
+from .device import Device, DeviceException
_LOGGER = logging.getLogger(__name__)
+MODEL_FAN_V2 = 'zimi.fan.v2'
+MODEL_FAN_V3 = 'zimi.fan.v3'
+
+AVAILABLE_PROPERTIES_COMMON = [
+ 'temp_dec',
+ 'humidity',
+ 'angle',
+ 'speed',
+ 'poweroff_time',
+ 'power',
+ 'ac_power',
+ 'battery',
+ 'angle_enable',
+ 'speed_level',
+ 'natural_level',
+ 'child_lock',
+ 'buzzer',
+ 'led_b',
+ 'use_time',
+ 'bat_charge',
+ 'button_pressed',
+]
+
+AVAILABLE_PROPERTIES = {
+ MODEL_FAN_V2: ['led', 'bat_state'] + AVAILABLE_PROPERTIES_COMMON,
+ MODEL_FAN_V3: AVAILABLE_PROPERTIES_COMMON,
+}
+
+
+class FanException(DeviceException):
+ pass
+
class LedBrightness(enum.Enum):
Bright = 0
@@ -22,15 +54,19 @@ class MoveDirection(enum.Enum):
class FanStatus:
- """Container for status reports from the Xiaomi Smart Fan."""
+ """Container for status reports from the Xiaomi Mi Smart Pedestal Fan."""
def __init__(self, data: Dict[str, Any]) -> None:
- # ['temp_dec', 'humidity', 'angle', 'speed', 'poweroff_time', 'power',
- # 'ac_power', 'battery', 'angle_enable', 'speed_level',
- # 'natural_level', 'child_lock', 'buzzer', 'led_b', 'led']
- #
- # [232, 46, 30, 298, 0, 'on', 'off', 98, 'off', 1, 0, 'off', 'on',
- # 1, 'on']
+ """
+ Response of a Fan (zhimi.fan.v3):
+
+ {'temp_dec': 232, 'humidity': 46, 'angle': 118, 'speed': 298,
+ 'poweroff_time': 0, 'power': 'on', 'ac_power': 'off', 'battery': 98,
+ 'angle_enable': 'off', 'speed_level': 1, 'natural_level': 0,
+ 'child_lock': 'off', 'buzzer': 'on', 'led_b': 1, 'led': None,
+ 'natural_enable': None, 'use_time': 0, 'bat_charge': 'complete',
+ 'bat_state': None, 'button_pressed':'speed'}
+ """
self.data = data
@property
@@ -49,16 +85,16 @@ def humidity(self) -> int:
return self.data["humidity"]
@property
- def temperature(self) -> Optional[float]:
+ def temperature(self) -> float:
"""Current temperature, if available."""
- if self.data["temp_dec"] is not None:
- return self.data["temp_dec"] / 10.0
- return None
+ return self.data["temp_dec"] / 10.0
@property
- def led(self) -> bool:
- """True if LED is turned on."""
- return self.data["led"] == "on"
+ def led(self) -> Optional[bool]:
+ """True if LED is turned on, if available."""
+ if "led" in self.data and self.data["led"] is not None:
+ return self.data["led"] == "on"
+ return None
@property
def led_brightness(self) -> Optional[LedBrightness]:
@@ -78,13 +114,13 @@ def child_lock(self) -> bool:
return self.data["child_lock"] == "on"
@property
- def natural_level(self) -> int:
- """Fan speed in natural mode."""
+ def natural_speed(self) -> int:
+ """Speed level in natural mode."""
return self.data["natural_level"]
@property
- def speed_level(self) -> int:
- """Fan speed in direct mode."""
+ def direct_speed(self) -> int:
+ """Speed level in direct mode."""
return self.data["speed_level"]
@property
@@ -97,20 +133,33 @@ def battery(self) -> int:
"""Current battery level."""
return self.data["battery"]
+ @property
+ def battery_charge(self) -> Optional[str]:
+ """State of the battery charger, if available."""
+ if self.data["bat_charge"] is not None:
+ return self.data["bat_charge"]
+ return None
+
+ @property
+ def battery_state(self) -> Optional[str]:
+ """State of the battery, if available."""
+ if "bat_state" in self.data and self.data["bat_state"] is not None:
+ return self.data["bat_state"]
+ return None
+
@property
def ac_power(self) -> bool:
"""True if powered by AC."""
return self.data["ac_power"] == "on"
@property
- def poweroff_time(self) -> int:
- """Time until turning off. FIXME verify"""
+ def delay_off_countdown(self) -> int:
+ """Countdown until turning off in seconds."""
return self.data["poweroff_time"]
@property
def speed(self) -> int:
- """FIXME What is the meaning of this value?
- (cp. speed_level vs. natural_level)"""
+ """Speed of the motor."""
return self.data["speed"]
@property
@@ -118,7 +167,19 @@ def angle(self) -> int:
"""Current angle."""
return self.data["angle"]
- def __str__(self) -> str:
+ @property
+ def use_time(self) -> int:
+ """How long the device has been active in seconds."""
+ return self.data["use_time"]
+
+ @property
+ def button_pressed(self) -> Optional[str]:
+ """Last pressed button."""
+ if self.data["button_pressed"] is not None:
+ return self.data["button_pressed"]
+ return None
+
+ def __repr__(self) -> str:
s = " str:
"led_brightness=%s, " \
"buzzer=%s, " \
"child_lock=%s, " \
- "natural_level=%s, " \
- "speed_level=%s, " \
+ "natural_speed=%s, " \
+ "direct_speed=%s, " \
+ "speed=%s, " \
"oscillate=%s, " \
- "battery=%s, " \
+ "angle=%s, " \
"ac_power=%s, " \
- "poweroff_time=%s, " \
- "speed=%s, " \
- "angle=%s" % \
+ "battery=%s, " \
+ "battery_charge=%s, " \
+ "battery_state=%s, " \
+ "use_time=%s, " \
+ "delay_off_countdown=%s, " \
+ "button_pressed=%s>" % \
(self.power,
self.temperature,
self.humidity,
@@ -141,14 +206,18 @@ def __str__(self) -> str:
self.led_brightness,
self.buzzer,
self.child_lock,
- self.natural_level,
- self.speed_level,
+ self.natural_speed,
+ self.direct_speed,
+ self.speed,
self.oscillate,
- self.battery,
+ self.angle,
self.ac_power,
- self.poweroff_time,
- self.speed_level,
- self.angle)
+ self.battery,
+ self.battery_charge,
+ self.battery_state,
+ self.use_time,
+ self.delay_off_countdown,
+ self.button_pressed)
return s
def __json__(self):
@@ -156,7 +225,17 @@ def __json__(self):
class Fan(Device):
- """Main class representing the Xiaomi Smart Fan."""
+ """Main class representing the Xiaomi Mi Smart Pedestal Fan."""
+
+ def __init__(self, ip: str = None, token: str = None, start_id: int = 0,
+ debug: int = 0, lazy_discover: bool = True,
+ model: str = MODEL_FAN_V3) -> None:
+ super().__init__(ip, token, start_id, debug, lazy_discover)
+
+ if model in AVAILABLE_PROPERTIES:
+ self.model = model
+ else:
+ self.model = MODEL_FAN_V3
@command(
default_output=format_output(
@@ -180,11 +259,7 @@ class Fan(Device):
)
def status(self) -> FanStatus:
"""Retrieve properties."""
- properties = ['temp_dec', 'humidity', 'angle', 'speed',
- 'poweroff_time', 'power', 'ac_power', 'battery',
- 'angle_enable', 'speed_level', 'natural_level',
- 'child_lock', 'buzzer', 'led_b', 'led']
-
+ properties = AVAILABLE_PROPERTIES[self.model]
values = self.send(
"get_prop",
properties
@@ -216,52 +291,61 @@ def off(self):
@command(
click.argument("speed", type=int),
- default_output=format_output("Setting natural level to {level}")
+ default_output=format_output(
+ "Setting speed of the natural mode to {speed}")
)
- def set_natural_level(self, level: int):
+ def set_natural_speed(self, speed: int):
"""Set natural level."""
- level = max(0, min(level, 100))
- return self.send("set_natural_level", [level]) # 0...100
+ if speed < 0 or speed > 100:
+ raise FanException("Invalid speed: %s" % speed)
+
+ return self.send("set_natural_level", [speed])
@command(
click.argument("speed", type=int),
- default_output=format_output("Setting speed level to {level}")
+ default_output=format_output(
+ "Setting speed of the direct mode to {speed}")
)
- def set_speed_level(self, level: int):
- """Set speed level."""
- level = max(0, min(level, 100))
- return self.send("set_speed_level", [level]) # 0...100
+ def set_direct_speed(self, speed: int):
+ """Set speed of the direct mode."""
+ if speed < 0 or speed > 100:
+ raise FanException("Invalid speed: %s" % speed)
+
+ return self.send("set_speed_level", [speed])
@command(
click.argument("direction", type=EnumType(MoveDirection, False)),
default_output=format_output(
- "Setting move direction to {direction}")
+ "Rotating the fan to the {direction}")
)
- def set_direction(self, direction: MoveDirection):
- """Set move direction."""
+ def set_rotate(self, direction: MoveDirection):
+ """Rotate the fan by -5/+5 degrees left/right."""
return self.send("set_move", [direction.value])
@command(
click.argument("angle", type=int),
default_output=format_output("Setting angle to {angle}")
)
- def fan_set_angle(self, angle: int):
- """Set angle."""
- return self.send("set_angle", [angle])
+ def set_angle(self, angle: int):
+ """Set the oscillation angle."""
+ if angle < 0 or angle > 120:
+ raise FanException("Invalid angle: %s" % angle)
- @command(
- default_output=format_output("Turning on oscillate"),
- )
- def oscillate_on(self):
- """Enable oscillate."""
- return self.send("set_angle_enable", ["on"])
+ return self.send("set_angle", [angle])
@command(
- default_output=format_output("Turning off oscillate"),
+ click.argument("oscillate", type=bool),
+ default_output=format_output(
+ lambda lock: "Turning on oscillate"
+ if lock else "Turning off oscillate"
+ )
)
- def oscillate_off(self):
- """Disable oscillate."""
- return self.send("set_angle_enable", ["off"])
+ def set_oscillate(self, oscillate: bool):
+ """Set oscillate on/off."""
+ if oscillate:
+ return self.send("set_angle_enable", ["on"])
+ else:
+ return self.send("set_angle_enable", ["off"])
@command(
click.argument("brightness", type=EnumType(LedBrightness, False)),
@@ -273,29 +357,57 @@ def set_led_brightness(self, brightness: LedBrightness):
return self.send("set_led_b", [brightness.value])
@command(
- default_output=format_output("Turning on LED"),
+ click.argument("led", type=bool),
+ default_output=format_output(
+ lambda led: "Turning on LED"
+ if led else "Turning off LED"
+ )
)
- def led_on(self):
- """Turn led on."""
- return self.send("set_led", ["on"])
+ def set_led(self, led: bool):
+ """Turn led on/off."""
+ if led:
+ return self.send("set_led", ['on'])
+ else:
+ return self.send("set_led", ['off'])
@command(
- default_output=format_output("Turning off LED"),
+ click.argument("buzzer", type=bool),
+ default_output=format_output(
+ lambda buzzer: "Turning on buzzer"
+ if buzzer else "Turning off buzzer"
+ )
)
- def led_off(self):
- """Turn led off."""
- return self.send("set_led", ["off"])
+ def set_buzzer(self, buzzer: bool):
+ """Set buzzer on/off."""
+ if buzzer:
+ return self.send("set_buzzer", ["on"])
+ else:
+ return self.send("set_buzzer", ["off"])
@command(
- default_output=format_output("Turning on buzzer"),
+ click.argument("lock", type=bool),
+ default_output=format_output(
+ lambda lock: "Turning on child lock"
+ if lock else "Turning off child lock"
+ )
)
- def buzzer_on(self):
- """Enable buzzer."""
- return self.send("set_buzzer", ["on"])
+ def set_child_lock(self, lock: bool):
+ """Set child lock on/off."""
+ if lock:
+ return self.send("set_child_lock", ["on"])
+ else:
+ return self.send("set_child_lock", ["off"])
@command(
- default_output=format_output("Turning off buzzer"),
+ click.argument("seconds", type=int),
+ default_output=format_output(
+ "Setting delayed turn off to {seconds} seconds")
)
- def buzzer_off(self):
- """Disable buzzer."""
- return self.send("set_buzzer", ["off"])
+ def delay_off(self, seconds: int):
+ """Set delay off seconds."""
+
+ if seconds < 1:
+ raise FanException(
+ "Invalid value for a delayed turn off: %s" % seconds)
+
+ return self.send("set_poweroff_time", [seconds])
diff --git a/miio/tests/test_fan.py b/miio/tests/test_fan.py
new file mode 100644
index 000000000..ccb1ab5c3
--- /dev/null
+++ b/miio/tests/test_fan.py
@@ -0,0 +1,475 @@
+from unittest import TestCase
+
+import pytest
+
+from miio import Fan
+from miio.fan import (MoveDirection, LedBrightness, FanStatus, FanException,
+ MODEL_FAN_V2, MODEL_FAN_V3, )
+from .dummies import DummyDevice
+
+
+class DummyFanV2(DummyDevice, Fan):
+ def __init__(self, *args, **kwargs):
+ self.model = MODEL_FAN_V2
+ # This example response is just a guess. Please update!
+ self.state = {
+ 'temp_dec': 232,
+ 'humidity': 46,
+ 'angle': 118,
+ 'speed': 298,
+ 'poweroff_time': 0,
+ 'power': 'on',
+ 'ac_power': 'off',
+ 'battery': 98,
+ 'angle_enable': 'off',
+ 'speed_level': 1,
+ 'natural_level': 0,
+ 'child_lock': 'off',
+ 'buzzer': 'on',
+ 'led_b': 1,
+ 'led': 'on',
+ 'natural_enable': None,
+ 'use_time': 0,
+ 'bat_charge': 'complete',
+ 'bat_state': None,
+ 'button_pressed': 'speed'
+ }
+ self.return_values = {
+ 'get_prop': self._get_state,
+ 'set_power': lambda x: self._set_state("power", x),
+ 'set_speed_level': lambda x: self._set_state("speed_level", x),
+ 'set_natural_level': lambda x: self._set_state("natural_level", x),
+ 'set_move': lambda x: True,
+ 'set_angle': lambda x: self._set_state("angle", x),
+ 'set_angle_enable': lambda x: self._set_state("angle_enable", x),
+ 'set_led_b': lambda x: self._set_state("led_b", x),
+ 'set_led': lambda x: self._set_state("led", x),
+ 'set_buzzer': lambda x: self._set_state("buzzer", x),
+ 'set_child_lock': lambda x: self._set_state("child_lock", x),
+ 'set_poweroff_time': lambda x: self._set_state("poweroff_time", x),
+ }
+ super().__init__(args, kwargs)
+
+
+@pytest.fixture(scope="class")
+def fanv2(request):
+ request.cls.device = DummyFanV2()
+ # TODO add ability to test on a real device
+
+
+@pytest.mark.usefixtures("fanv2")
+class TestFanV2(TestCase):
+ def is_on(self):
+ return self.device.status().is_on
+
+ def state(self):
+ return self.device.status()
+
+ def test_on(self):
+ self.device.off() # ensure off
+ assert self.is_on() is False
+
+ self.device.on()
+ assert self.is_on() is True
+
+ def test_off(self):
+ self.device.on() # ensure on
+ assert self.is_on() is True
+
+ self.device.off()
+ assert self.is_on() is False
+
+ def test_status(self):
+ self.device._reset_state()
+
+ assert repr(self.state()) == repr(FanStatus(self.device.start_state))
+
+ assert self.is_on() is True
+ assert self.state().temperature == self.device.start_state["temp_dec"] / 10.0
+ assert self.state().humidity == self.device.start_state["humidity"]
+ assert self.state().angle == self.device.start_state["angle"]
+ assert self.state().speed == self.device.start_state["speed"]
+ assert self.state().delay_off_countdown == self.device.start_state["poweroff_time"]
+ assert self.state().ac_power is (self.device.start_state["ac_power"] == 'on')
+ assert self.state().battery == self.device.start_state["battery"]
+ assert self.state().oscillate is (self.device.start_state["angle_enable"] == 'on')
+ assert self.state().direct_speed == self.device.start_state["speed_level"]
+ assert self.state().natural_speed == self.device.start_state["natural_level"]
+ assert self.state().child_lock is (self.device.start_state["child_lock"] == 'on')
+ assert self.state().buzzer is (self.device.start_state["buzzer"] == 'on')
+ assert self.state().led_brightness == LedBrightness(self.device.start_state["led_b"])
+ assert self.state().led is (self.device.start_state["led"] == "on")
+ assert self.state().use_time == self.device.start_state["use_time"]
+ assert self.state().battery_charge == self.device.start_state["bat_charge"]
+ assert self.state().battery_state == self.device.start_state["bat_state"]
+ assert self.state().button_pressed == self.device.start_state["button_pressed"]
+
+ def test_status_without_led_brightness(self):
+ self.device._reset_state()
+
+ self.device.state["led_b"] = None
+ assert self.state().led_brightness is None
+
+ def test_status_without_battery_charge(self):
+ self.device._reset_state()
+
+ self.device.state["bat_charge"] = None
+ assert self.state().battery_charge is None
+
+ def test_status_without_battery_state(self):
+ self.device._reset_state()
+
+ self.device.state["bat_state"] = None
+ assert self.state().battery_state is None
+
+ def test_status_without_button_pressed(self):
+ self.device._reset_state()
+
+ self.device.state["button_pressed"] = None
+ assert self.state().button_pressed is None
+
+ def test_set_led(self):
+ def led():
+ return self.device.status().led
+
+ self.device.set_led(True)
+ assert led() is True
+
+ self.device.set_led(False)
+ assert led() is False
+
+ def test_set_direct_speed(self):
+ def direct_speed():
+ return self.device.status().direct_speed
+
+ self.device.set_direct_speed(0)
+ assert direct_speed() == 0
+ self.device.set_direct_speed(1)
+ assert direct_speed() == 1
+ self.device.set_direct_speed(100)
+ assert direct_speed() == 100
+
+ with pytest.raises(FanException):
+ self.device.set_direct_speed(-1)
+
+ with pytest.raises(FanException):
+ self.device.set_direct_speed(101)
+
+ def test_set_rotate(self):
+ """The method is open-loop. The new state cannot be retrieved."""
+ self.device.set_rotate(MoveDirection.Left)
+ self.device.set_rotate(MoveDirection.Right)
+
+ def test_set_angle(self):
+ """This test doesn't implement the real behaviour of the device may be.
+
+ The property "angle" doesn't provide the current setting.
+ It's a measurement of the current position probably.
+ """
+ def angle():
+ return self.device.status().angle
+
+ self.device.set_angle(0) # TODO: Is this value allowed?
+ assert angle() == 0
+ self.device.set_angle(1) # TODO: Is this value allowed?
+ assert angle() == 1
+ self.device.set_angle(30)
+ assert angle() == 30
+ self.device.set_angle(60)
+ assert angle() == 60
+ self.device.set_angle(90)
+ assert angle() == 90
+ self.device.set_angle(120)
+ assert angle() == 120
+
+ with pytest.raises(FanException):
+ self.device.set_angle(-1)
+
+ with pytest.raises(FanException):
+ self.device.set_angle(121)
+
+ def test_set_oscillate(self):
+ def oscillate():
+ return self.device.status().oscillate
+
+ self.device.set_oscillate(True)
+ assert oscillate() is True
+
+ self.device.set_oscillate(False)
+ assert oscillate() is False
+
+ def test_set_led_brightness(self):
+ def led_brightness():
+ return self.device.status().led_brightness
+
+ self.device.set_led_brightness(LedBrightness.Bright)
+ assert led_brightness() == LedBrightness.Bright
+
+ self.device.set_led_brightness(LedBrightness.Dim)
+ assert led_brightness() == LedBrightness.Dim
+
+ self.device.set_led_brightness(LedBrightness.Off)
+ assert led_brightness() == LedBrightness.Off
+
+ def test_set_buzzer(self):
+ def buzzer():
+ return self.device.status().buzzer
+
+ self.device.set_buzzer(True)
+ assert buzzer() is True
+
+ self.device.set_buzzer(False)
+ assert buzzer() is False
+
+ def test_set_child_lock(self):
+ def child_lock():
+ return self.device.status().child_lock
+
+ self.device.set_child_lock(True)
+ assert child_lock() is True
+
+ self.device.set_child_lock(False)
+ assert child_lock() is False
+
+ def test_delay_off(self):
+ def delay_off_countdown():
+ return self.device.status().delay_off_countdown
+
+ self.device.delay_off(100)
+ assert delay_off_countdown() == 100
+ self.device.delay_off(200)
+ assert delay_off_countdown() == 200
+
+ with pytest.raises(FanException):
+ self.device.delay_off(-1)
+
+ with pytest.raises(FanException):
+ self.device.delay_off(0)
+
+
+class DummyFanV3(DummyDevice, Fan):
+ def __init__(self, *args, **kwargs):
+ self.model = MODEL_FAN_V3
+ self.state = {
+ 'temp_dec': 232,
+ 'humidity': 46,
+ 'angle': 118,
+ 'speed': 298,
+ 'poweroff_time': 0,
+ 'power': 'on',
+ 'ac_power': 'off',
+ 'battery': 98,
+ 'angle_enable': 'off',
+ 'speed_level': 1,
+ 'natural_level': 0,
+ 'child_lock': 'off',
+ 'buzzer': 'on',
+ 'led_b': 1,
+ 'led': None,
+ 'natural_enable': None,
+ 'use_time': 0,
+ 'bat_charge': 'complete',
+ 'bat_state': None,
+ 'button_pressed': 'speed'
+ }
+ self.return_values = {
+ 'get_prop': self._get_state,
+ 'set_power': lambda x: self._set_state("power", x),
+ 'set_speed_level': lambda x: self._set_state("speed_level", x),
+ 'set_natural_level': lambda x: self._set_state("natural_level", x),
+ 'set_move': lambda x: True,
+ 'set_angle': lambda x: self._set_state("angle", x),
+ 'set_angle_enable': lambda x: self._set_state("angle_enable", x),
+ 'set_led_b': lambda x: self._set_state("led_b", x),
+ 'set_led': lambda x: self._set_state("led", x),
+ 'set_buzzer': lambda x: self._set_state("buzzer", x),
+ 'set_child_lock': lambda x: self._set_state("child_lock", x),
+ 'set_poweroff_time': lambda x: self._set_state("poweroff_time", x),
+ }
+ super().__init__(args, kwargs)
+
+
+@pytest.fixture(scope="class")
+def fanv3(request):
+ request.cls.device = DummyFanV3()
+ # TODO add ability to test on a real device
+
+
+@pytest.mark.usefixtures("fanv3")
+class TestFanV3(TestCase):
+ def is_on(self):
+ return self.device.status().is_on
+
+ def state(self):
+ return self.device.status()
+
+ def test_on(self):
+ self.device.off() # ensure off
+ assert self.is_on() is False
+
+ self.device.on()
+ assert self.is_on() is True
+
+ def test_off(self):
+ self.device.on() # ensure on
+ assert self.is_on() is True
+
+ self.device.off()
+ assert self.is_on() is False
+
+ def test_status(self):
+ self.device._reset_state()
+
+ assert repr(self.state()) == repr(FanStatus(self.device.start_state))
+
+ assert self.is_on() is True
+ assert self.state().temperature == self.device.start_state["temp_dec"] / 10.0
+ assert self.state().humidity == self.device.start_state["humidity"]
+ assert self.state().angle == self.device.start_state["angle"]
+ assert self.state().speed == self.device.start_state["speed"]
+ assert self.state().delay_off_countdown == self.device.start_state["poweroff_time"]
+ assert self.state().ac_power is (self.device.start_state["ac_power"] == 'on')
+ assert self.state().battery == self.device.start_state["battery"]
+ assert self.state().oscillate is (self.device.start_state["angle_enable"] == 'on')
+ assert self.state().direct_speed == self.device.start_state["speed_level"]
+ assert self.state().natural_speed == self.device.start_state["natural_level"]
+ assert self.state().child_lock is (self.device.start_state["child_lock"] == 'on')
+ assert self.state().buzzer is (self.device.start_state["buzzer"] == 'on')
+ assert self.state().led_brightness == LedBrightness(self.device.start_state["led_b"])
+ assert self.state().led is None
+ assert self.state().use_time == self.device.start_state["use_time"]
+ assert self.state().battery_charge == self.device.start_state["bat_charge"]
+ assert self.state().battery_state == self.device.start_state["bat_state"]
+ assert self.state().button_pressed == self.device.start_state["button_pressed"]
+
+ def test_status_without_led_brightness(self):
+ self.device._reset_state()
+
+ self.device.state["led_b"] = None
+ assert self.state().led_brightness is None
+
+ def test_status_without_battery_charge(self):
+ self.device._reset_state()
+
+ self.device.state["bat_charge"] = None
+ assert self.state().battery_charge is None
+
+ def test_status_without_battery_state(self):
+ self.device._reset_state()
+
+ self.device.state["bat_state"] = None
+ assert self.state().battery_state is None
+
+ def test_status_without_button_pressed(self):
+ self.device._reset_state()
+
+ self.device.state["button_pressed"] = None
+ assert self.state().button_pressed is None
+
+ def test_set_direct_speed(self):
+ def direct_speed():
+ return self.device.status().direct_speed
+
+ self.device.set_direct_speed(0)
+ assert direct_speed() == 0
+ self.device.set_direct_speed(1)
+ assert direct_speed() == 1
+ self.device.set_direct_speed(100)
+ assert direct_speed() == 100
+
+ with pytest.raises(FanException):
+ self.device.set_direct_speed(-1)
+
+ with pytest.raises(FanException):
+ self.device.set_direct_speed(101)
+
+ def test_set_rotate(self):
+ """The method is open-loop. The new state cannot be retrieved."""
+ self.device.set_rotate(MoveDirection.Left)
+ self.device.set_rotate(MoveDirection.Right)
+
+ def test_set_angle(self):
+ """This test doesn't implement the real behaviour of the device may be.
+
+ The property "angle" doesn't provide the current setting.
+ It's a measurement of the current position probably.
+ """
+ def angle():
+ return self.device.status().angle
+
+ self.device.set_angle(0) # TODO: Is this value allowed?
+ assert angle() == 0
+ self.device.set_angle(1) # TODO: Is this value allowed?
+ assert angle() == 1
+ self.device.set_angle(30)
+ assert angle() == 30
+ self.device.set_angle(60)
+ assert angle() == 60
+ self.device.set_angle(90)
+ assert angle() == 90
+ self.device.set_angle(120)
+ assert angle() == 120
+
+ with pytest.raises(FanException):
+ self.device.set_angle(-1)
+
+ with pytest.raises(FanException):
+ self.device.set_angle(121)
+
+ def test_set_oscillate(self):
+ def oscillate():
+ return self.device.status().oscillate
+
+ self.device.set_oscillate(True)
+ assert oscillate() is True
+
+ self.device.set_oscillate(False)
+ assert oscillate() is False
+
+ def test_set_led_brightness(self):
+ def led_brightness():
+ return self.device.status().led_brightness
+
+ self.device.set_led_brightness(LedBrightness.Bright)
+ assert led_brightness() == LedBrightness.Bright
+
+ self.device.set_led_brightness(LedBrightness.Dim)
+ assert led_brightness() == LedBrightness.Dim
+
+ self.device.set_led_brightness(LedBrightness.Off)
+ assert led_brightness() == LedBrightness.Off
+
+ def test_set_buzzer(self):
+ def buzzer():
+ return self.device.status().buzzer
+
+ self.device.set_buzzer(True)
+ assert buzzer() is True
+
+ self.device.set_buzzer(False)
+ assert buzzer() is False
+
+ def test_set_child_lock(self):
+ def child_lock():
+ return self.device.status().child_lock
+
+ self.device.set_child_lock(True)
+ assert child_lock() is True
+
+ self.device.set_child_lock(False)
+ assert child_lock() is False
+
+ def test_delay_off(self):
+ def delay_off_countdown():
+ return self.device.status().delay_off_countdown
+
+ self.device.delay_off(100)
+ assert delay_off_countdown() == 100
+ self.device.delay_off(200)
+ assert delay_off_countdown() == 200
+
+ with pytest.raises(FanException):
+ self.device.delay_off(-1)
+
+ with pytest.raises(FanException):
+ self.device.delay_off(0)