core/homeassistant/components/weather/significant_change.py

172 lines
5.2 KiB
Python

"""Helper to test significant Weather state changes."""
from __future__ import annotations
from typing import Any
from homeassistant.const import UnitOfPressure, UnitOfSpeed, UnitOfTemperature
from homeassistant.core import HomeAssistant, callback
from homeassistant.helpers.significant_change import (
check_absolute_change,
check_valid_float,
)
from .const import (
ATTR_WEATHER_APPARENT_TEMPERATURE,
ATTR_WEATHER_CLOUD_COVERAGE,
ATTR_WEATHER_DEW_POINT,
ATTR_WEATHER_HUMIDITY,
ATTR_WEATHER_OZONE,
ATTR_WEATHER_PRESSURE,
ATTR_WEATHER_PRESSURE_UNIT,
ATTR_WEATHER_TEMPERATURE,
ATTR_WEATHER_TEMPERATURE_UNIT,
ATTR_WEATHER_UV_INDEX,
ATTR_WEATHER_VISIBILITY,
ATTR_WEATHER_WIND_BEARING,
ATTR_WEATHER_WIND_GUST_SPEED,
ATTR_WEATHER_WIND_SPEED,
ATTR_WEATHER_WIND_SPEED_UNIT,
)
SIGNIFICANT_ATTRIBUTES: set[str] = {
ATTR_WEATHER_APPARENT_TEMPERATURE,
ATTR_WEATHER_CLOUD_COVERAGE,
ATTR_WEATHER_DEW_POINT,
ATTR_WEATHER_HUMIDITY,
ATTR_WEATHER_OZONE,
ATTR_WEATHER_PRESSURE,
ATTR_WEATHER_TEMPERATURE,
ATTR_WEATHER_UV_INDEX,
ATTR_WEATHER_VISIBILITY,
ATTR_WEATHER_WIND_BEARING,
ATTR_WEATHER_WIND_GUST_SPEED,
ATTR_WEATHER_WIND_SPEED,
}
VALID_CARDINAL_DIRECTIONS: list[str] = [
"n",
"nne",
"ne",
"ene",
"e",
"ese",
"se",
"sse",
"s",
"ssw",
"sw",
"wsw",
"w",
"wnw",
"nw",
"nnw",
]
def _cardinal_to_degrees(value: str | float | None) -> int | float | None:
"""Translate a cardinal direction into azimuth angle (degrees)."""
if not isinstance(value, str):
return value
try:
return float(360 / 16 * VALID_CARDINAL_DIRECTIONS.index(value.lower()))
except ValueError:
return None
@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."""
# state changes are always significant
if old_state != new_state:
return True
old_attrs_s = set(
{k: v for k, v in old_attrs.items() if k in SIGNIFICANT_ATTRIBUTES}.items()
)
new_attrs_s = set(
{k: v for k, v in new_attrs.items() if k in SIGNIFICANT_ATTRIBUTES}.items()
)
changed_attrs: set[str] = {item[0] for item in old_attrs_s ^ new_attrs_s}
for attr_name in changed_attrs:
old_attr_value = old_attrs.get(attr_name)
new_attr_value = new_attrs.get(attr_name)
absolute_change: float | None = None
if attr_name == ATTR_WEATHER_WIND_BEARING:
old_attr_value = _cardinal_to_degrees(old_attr_value)
new_attr_value = _cardinal_to_degrees(new_attr_value)
if new_attr_value is None or not check_valid_float(new_attr_value):
# New attribute value is invalid, ignore it
continue
if old_attr_value is None or not check_valid_float(old_attr_value):
# Old attribute value was invalid, we should report again
return True
if attr_name in (
ATTR_WEATHER_APPARENT_TEMPERATURE,
ATTR_WEATHER_DEW_POINT,
ATTR_WEATHER_TEMPERATURE,
):
if (
unit := new_attrs.get(ATTR_WEATHER_TEMPERATURE_UNIT)
) is not None and unit == UnitOfTemperature.FAHRENHEIT:
absolute_change = 1.0
else:
absolute_change = 0.5
if attr_name in (
ATTR_WEATHER_WIND_GUST_SPEED,
ATTR_WEATHER_WIND_SPEED,
):
if (
unit := new_attrs.get(ATTR_WEATHER_WIND_SPEED_UNIT)
) is None or unit in (
UnitOfSpeed.KILOMETERS_PER_HOUR,
UnitOfSpeed.MILES_PER_HOUR, # 1km/h = 0.62mi/s
UnitOfSpeed.FEET_PER_SECOND, # 1km/h = 0.91ft/s
):
absolute_change = 1.0
elif unit == UnitOfSpeed.METERS_PER_SECOND: # 1km/h = 0.277m/s
absolute_change = 0.5
if attr_name in (
ATTR_WEATHER_CLOUD_COVERAGE, # range 0-100%
ATTR_WEATHER_HUMIDITY, # range 0-100%
ATTR_WEATHER_OZONE, # range ~20-100ppm
ATTR_WEATHER_VISIBILITY, # range 0-240km (150mi)
ATTR_WEATHER_WIND_BEARING, # range 0-359°
):
absolute_change = 1.0
if attr_name == ATTR_WEATHER_UV_INDEX: # range 1-11
absolute_change = 0.1
if attr_name == ATTR_WEATHER_PRESSURE: # local variation of around 100 hpa
if (unit := new_attrs.get(ATTR_WEATHER_PRESSURE_UNIT)) is None or unit in (
UnitOfPressure.HPA,
UnitOfPressure.MBAR, # 1hPa = 1mbar
UnitOfPressure.MMHG, # 1hPa = 0.75mmHg
):
absolute_change = 1.0
elif unit == UnitOfPressure.INHG: # 1hPa = 0.03inHg
absolute_change = 0.05
# check for significant attribute value change
if absolute_change is not None:
if check_absolute_change(old_attr_value, new_attr_value, absolute_change):
return True
# no significant attribute change detected
return False