mirror of https://github.com/home-assistant/core
85 lines
2.4 KiB
Python
85 lines
2.4 KiB
Python
"""Helper to test significant sensor state changes."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from typing import Any
|
|
|
|
from homeassistant.const import (
|
|
ATTR_DEVICE_CLASS,
|
|
ATTR_UNIT_OF_MEASUREMENT,
|
|
UnitOfTemperature,
|
|
)
|
|
from homeassistant.core import HomeAssistant, callback
|
|
from homeassistant.helpers.significant_change import (
|
|
check_absolute_change,
|
|
check_percentage_change,
|
|
check_valid_float,
|
|
)
|
|
|
|
from . import SensorDeviceClass
|
|
|
|
|
|
def _absolute_and_relative_change(
|
|
old_state: float | None,
|
|
new_state: float | None,
|
|
absolute_change: float,
|
|
percentage_change: float,
|
|
) -> bool:
|
|
return check_absolute_change(
|
|
old_state, new_state, absolute_change
|
|
) and check_percentage_change(old_state, new_state, percentage_change)
|
|
|
|
|
|
@callback
|
|
def async_check_significant_change(
|
|
hass: HomeAssistant,
|
|
old_state: str,
|
|
old_attrs: dict,
|
|
new_state: str,
|
|
new_attrs: dict,
|
|
**kwargs: Any,
|
|
) -> bool | None:
|
|
"""Test if state significantly changed."""
|
|
if (device_class := new_attrs.get(ATTR_DEVICE_CLASS)) is None:
|
|
return None
|
|
|
|
absolute_change: float | None = None
|
|
percentage_change: float | None = None
|
|
if device_class == SensorDeviceClass.TEMPERATURE:
|
|
if new_attrs.get(ATTR_UNIT_OF_MEASUREMENT) == UnitOfTemperature.FAHRENHEIT:
|
|
absolute_change = 1.0
|
|
else:
|
|
absolute_change = 0.5
|
|
|
|
if device_class in (SensorDeviceClass.BATTERY, SensorDeviceClass.HUMIDITY):
|
|
absolute_change = 1.0
|
|
|
|
if device_class in (
|
|
SensorDeviceClass.AQI,
|
|
SensorDeviceClass.CO,
|
|
SensorDeviceClass.CO2,
|
|
SensorDeviceClass.PM25,
|
|
SensorDeviceClass.PM10,
|
|
SensorDeviceClass.VOLATILE_ORGANIC_COMPOUNDS,
|
|
):
|
|
absolute_change = 1.0
|
|
percentage_change = 2.0
|
|
|
|
if not check_valid_float(new_state):
|
|
# New state is invalid, don't report it
|
|
return False
|
|
|
|
if not check_valid_float(old_state):
|
|
# Old state was invalid, we should report again
|
|
return True
|
|
|
|
if absolute_change is not None and percentage_change is not None:
|
|
return _absolute_and_relative_change(
|
|
float(old_state), float(new_state), absolute_change, percentage_change
|
|
)
|
|
if absolute_change is not None:
|
|
return check_absolute_change(
|
|
float(old_state), float(new_state), absolute_change
|
|
)
|
|
return None
|