mirror of
https://github.com/home-assistant/core.git
synced 2025-07-26 06:37:52 +00:00
Expose Dyson PureCool filter life remaining percentage (#42765)
This commit is contained in:
parent
ebc26c70b9
commit
6b21df9053
@ -13,6 +13,8 @@ SENSOR_UNITS = {
|
|||||||
"air_quality": None,
|
"air_quality": None,
|
||||||
"dust": None,
|
"dust": None,
|
||||||
"filter_life": TIME_HOURS,
|
"filter_life": TIME_HOURS,
|
||||||
|
"carbon_filter_state": PERCENTAGE,
|
||||||
|
"hepa_filter_state": PERCENTAGE,
|
||||||
"humidity": PERCENTAGE,
|
"humidity": PERCENTAGE,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -20,6 +22,8 @@ SENSOR_ICONS = {
|
|||||||
"air_quality": "mdi:fan",
|
"air_quality": "mdi:fan",
|
||||||
"dust": "mdi:cloud",
|
"dust": "mdi:cloud",
|
||||||
"filter_life": "mdi:filter-outline",
|
"filter_life": "mdi:filter-outline",
|
||||||
|
"carbon_filter_state": "mdi:filter-outline",
|
||||||
|
"hepa_filter_state": "mdi:filter-outline",
|
||||||
"humidity": "mdi:water-percent",
|
"humidity": "mdi:water-percent",
|
||||||
"temperature": "mdi:thermometer",
|
"temperature": "mdi:thermometer",
|
||||||
}
|
}
|
||||||
@ -48,6 +52,17 @@ def setup_platform(hass, config, add_entities, discovery_info=None):
|
|||||||
new_entities.append(DysonTemperatureSensor(device, unit))
|
new_entities.append(DysonTemperatureSensor(device, unit))
|
||||||
if f"{device.serial}-humidity" not in device_ids:
|
if f"{device.serial}-humidity" not in device_ids:
|
||||||
new_entities.append(DysonHumiditySensor(device))
|
new_entities.append(DysonHumiditySensor(device))
|
||||||
|
|
||||||
|
# For PureCool+Humidify devices, a single filter exists, called "Combi Filter".
|
||||||
|
# It's reported with the HEPA state, while the Carbon state is set to INValid.
|
||||||
|
if device.state and device.state.carbon_filter_state == "INV":
|
||||||
|
if f"{device.serial}-hepa_filter_state" not in device_ids:
|
||||||
|
new_entities.append(DysonHepaFilterLifeSensor(device, "Combi"))
|
||||||
|
else:
|
||||||
|
if f"{device.serial}-hepa_filter_state" not in device_ids:
|
||||||
|
new_entities.append(DysonHepaFilterLifeSensor(device))
|
||||||
|
if f"{device.serial}-carbon_filter_state" not in device_ids:
|
||||||
|
new_entities.append(DysonCarbonFilterLifeSensor(device))
|
||||||
elif isinstance(device, DysonPureCoolLink):
|
elif isinstance(device, DysonPureCoolLink):
|
||||||
new_entities.append(DysonFilterLifeSensor(device))
|
new_entities.append(DysonFilterLifeSensor(device))
|
||||||
new_entities.append(DysonDustSensor(device))
|
new_entities.append(DysonDustSensor(device))
|
||||||
@ -126,6 +141,38 @@ class DysonFilterLifeSensor(DysonSensor):
|
|||||||
return None
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
class DysonCarbonFilterLifeSensor(DysonSensor):
|
||||||
|
"""Representation of Dyson Carbon Filter Life sensor (in percent)."""
|
||||||
|
|
||||||
|
def __init__(self, device):
|
||||||
|
"""Create a new Dyson Carbon Filter Life sensor."""
|
||||||
|
super().__init__(device, "carbon_filter_state")
|
||||||
|
self._name = f"{self._device.name} Carbon Filter Remaining Life"
|
||||||
|
|
||||||
|
@property
|
||||||
|
def state(self):
|
||||||
|
"""Return filter life remaining in percent."""
|
||||||
|
if self._device.state:
|
||||||
|
return int(self._device.state.carbon_filter_state)
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
class DysonHepaFilterLifeSensor(DysonSensor):
|
||||||
|
"""Representation of Dyson HEPA (or Combi) Filter Life sensor (in percent)."""
|
||||||
|
|
||||||
|
def __init__(self, device, filter_type="HEPA"):
|
||||||
|
"""Create a new Dyson Filter Life sensor."""
|
||||||
|
super().__init__(device, "hepa_filter_state")
|
||||||
|
self._name = f"{self._device.name} {filter_type} Filter Remaining Life"
|
||||||
|
|
||||||
|
@property
|
||||||
|
def state(self):
|
||||||
|
"""Return filter life remaining in percent."""
|
||||||
|
if self._device.state:
|
||||||
|
return int(self._device.state.hepa_filter_state)
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
class DysonDustSensor(DysonSensor):
|
class DysonDustSensor(DysonSensor):
|
||||||
"""Representation of Dyson Dust sensor (lower is better)."""
|
"""Representation of Dyson Dust sensor (lower is better)."""
|
||||||
|
|
||||||
|
@ -8,6 +8,7 @@ from libpurecool.dyson_pure_cool_link import DysonPureCoolLink
|
|||||||
from homeassistant.components import dyson as dyson_parent
|
from homeassistant.components import dyson as dyson_parent
|
||||||
from homeassistant.components.dyson import sensor as dyson
|
from homeassistant.components.dyson import sensor as dyson
|
||||||
from homeassistant.const import (
|
from homeassistant.const import (
|
||||||
|
ATTR_UNIT_OF_MEASUREMENT,
|
||||||
PERCENTAGE,
|
PERCENTAGE,
|
||||||
STATE_OFF,
|
STATE_OFF,
|
||||||
TEMP_CELSIUS,
|
TEMP_CELSIUS,
|
||||||
@ -67,6 +68,36 @@ def _get_with_state():
|
|||||||
return device
|
return device
|
||||||
|
|
||||||
|
|
||||||
|
def _get_purecool_device():
|
||||||
|
"""Return a valid device with filters life state values."""
|
||||||
|
device = mock.Mock(spec=DysonPureCool)
|
||||||
|
load_mock_device(device)
|
||||||
|
device.name = "PureCool"
|
||||||
|
device.state.carbon_filter_state = "0096"
|
||||||
|
device.state.hepa_filter_state = "0056"
|
||||||
|
device.environmental_state.dust = 5
|
||||||
|
device.environmental_state.humidity = 45
|
||||||
|
device.environmental_state.temperature = 295
|
||||||
|
device.environmental_state.volatil_organic_compounds = 2
|
||||||
|
|
||||||
|
return device
|
||||||
|
|
||||||
|
|
||||||
|
def _get_purecool_humidify_device():
|
||||||
|
"""Return a valid device with filters life state values."""
|
||||||
|
device = mock.Mock(spec=DysonPureCool)
|
||||||
|
load_mock_device(device)
|
||||||
|
device.name = "PureCool_Humidify"
|
||||||
|
device.state.carbon_filter_state = "INV"
|
||||||
|
device.state.hepa_filter_state = "0075"
|
||||||
|
device.environmental_state.dust = 5
|
||||||
|
device.environmental_state.humidity = 45
|
||||||
|
device.environmental_state.temperature = 295
|
||||||
|
device.environmental_state.volatil_organic_compounds = 2
|
||||||
|
|
||||||
|
return device
|
||||||
|
|
||||||
|
|
||||||
def _get_with_standby_monitoring():
|
def _get_with_standby_monitoring():
|
||||||
"""Return a valid device with state but with standby monitoring disable."""
|
"""Return a valid device with state but with standby monitoring disable."""
|
||||||
device = mock.Mock()
|
device = mock.Mock()
|
||||||
@ -110,7 +141,10 @@ class DysonTest(unittest.TestCase):
|
|||||||
|
|
||||||
device_fan = _get_device_without_state()
|
device_fan = _get_device_without_state()
|
||||||
device_non_fan = _get_with_state()
|
device_non_fan = _get_with_state()
|
||||||
self.hass.data[dyson.DYSON_DEVICES] = [device_fan, device_non_fan]
|
self.hass.data[dyson.DYSON_DEVICES] = [
|
||||||
|
device_fan,
|
||||||
|
device_non_fan,
|
||||||
|
]
|
||||||
dyson.setup_platform(self.hass, None, _add_device, mock.MagicMock())
|
dyson.setup_platform(self.hass, None, _add_device, mock.MagicMock())
|
||||||
|
|
||||||
def test_dyson_filter_life_sensor(self):
|
def test_dyson_filter_life_sensor(self):
|
||||||
@ -272,4 +306,46 @@ async def test_purecool_component_setup_only_once(devices, login, hass):
|
|||||||
discovery.load_platform(hass, "sensor", dyson_parent.DOMAIN, {}, config)
|
discovery.load_platform(hass, "sensor", dyson_parent.DOMAIN, {}, config)
|
||||||
await hass.async_block_till_done()
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
assert len(hass.data[dyson.DYSON_SENSOR_DEVICES]) == 2
|
assert len(hass.data[dyson.DYSON_SENSOR_DEVICES]) == 4
|
||||||
|
|
||||||
|
|
||||||
|
@patch("libpurecool.dyson.DysonAccount.login", return_value=True)
|
||||||
|
@patch(
|
||||||
|
"libpurecool.dyson.DysonAccount.devices",
|
||||||
|
return_value=[_get_purecool_device()],
|
||||||
|
)
|
||||||
|
async def test_dyson_purecool_filter_state_sensor(devices, login, hass):
|
||||||
|
"""Test filter sensor with values."""
|
||||||
|
config = _get_config()
|
||||||
|
await async_setup_component(hass, dyson_parent.DOMAIN, config)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
state = hass.states.get("sensor.purecool_hepa_filter_remaining_life")
|
||||||
|
assert state is not None
|
||||||
|
assert state.state == "56"
|
||||||
|
assert state.attributes[ATTR_UNIT_OF_MEASUREMENT] == PERCENTAGE
|
||||||
|
assert state.name == "PureCool HEPA Filter Remaining Life"
|
||||||
|
|
||||||
|
state = hass.states.get("sensor.purecool_carbon_filter_remaining_life")
|
||||||
|
assert state is not None
|
||||||
|
assert state.state == "96"
|
||||||
|
assert state.attributes[ATTR_UNIT_OF_MEASUREMENT] == PERCENTAGE
|
||||||
|
assert state.name == "PureCool Carbon Filter Remaining Life"
|
||||||
|
|
||||||
|
|
||||||
|
@patch("libpurecool.dyson.DysonAccount.login", return_value=True)
|
||||||
|
@patch(
|
||||||
|
"libpurecool.dyson.DysonAccount.devices",
|
||||||
|
return_value=[_get_purecool_humidify_device()],
|
||||||
|
)
|
||||||
|
async def test_dyson_purecool_humidify_filter_state_sensor(devices, login, hass):
|
||||||
|
"""Test filter sensor with values."""
|
||||||
|
config = _get_config()
|
||||||
|
await async_setup_component(hass, dyson_parent.DOMAIN, config)
|
||||||
|
await hass.async_block_till_done()
|
||||||
|
|
||||||
|
state = hass.states.get("sensor.purecool_humidify_combi_filter_remaining_life")
|
||||||
|
assert state is not None
|
||||||
|
assert state.state == "75"
|
||||||
|
assert state.attributes[ATTR_UNIT_OF_MEASUREMENT] == PERCENTAGE
|
||||||
|
assert state.name == "PureCool_Humidify Combi Filter Remaining Life"
|
||||||
|
Loading…
x
Reference in New Issue
Block a user