Add significant Change support for weather (#104840)

pull/104577/head
Michael 2023-12-05 19:17:52 +01:00 committed by GitHub
parent 1edfaed7be
commit 3310f4c130
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 522 additions and 0 deletions

View File

@ -0,0 +1,175 @@
"""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
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 _check_valid_float(value: str | int | float) -> bool:
"""Check if given value is a valid float."""
try:
float(value)
except ValueError:
return False
return True
def _cardinal_to_degrees(value: str | int | 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(old_attrs.items())
new_attrs_s = set(new_attrs.items())
changed_attrs: set[str] = {item[0] for item in old_attrs_s ^ new_attrs_s}
for attr_name in changed_attrs:
if attr_name not in SIGNIFICANT_ATTRIBUTES:
continue
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

View File

@ -0,0 +1,347 @@
"""Test the Weather significant change platform."""
import pytest
from homeassistant.components.weather.const import (
ATTR_WEATHER_APPARENT_TEMPERATURE,
ATTR_WEATHER_CLOUD_COVERAGE,
ATTR_WEATHER_DEW_POINT,
ATTR_WEATHER_HUMIDITY,
ATTR_WEATHER_OZONE,
ATTR_WEATHER_PRECIPITATION_UNIT,
ATTR_WEATHER_PRESSURE,
ATTR_WEATHER_PRESSURE_UNIT,
ATTR_WEATHER_TEMPERATURE,
ATTR_WEATHER_TEMPERATURE_UNIT,
ATTR_WEATHER_UV_INDEX,
ATTR_WEATHER_VISIBILITY_UNIT,
ATTR_WEATHER_WIND_BEARING,
ATTR_WEATHER_WIND_GUST_SPEED,
ATTR_WEATHER_WIND_SPEED,
ATTR_WEATHER_WIND_SPEED_UNIT,
)
from homeassistant.components.weather.significant_change import (
async_check_significant_change,
)
from homeassistant.const import UnitOfPressure, UnitOfSpeed, UnitOfTemperature
async def test_significant_state_change() -> None:
"""Detect Weather significant state changes."""
assert not async_check_significant_change(
None, "clear-night", {}, "clear-night", {}
)
assert async_check_significant_change(None, "clear-night", {}, "cloudy", {})
@pytest.mark.parametrize(
("old_attrs", "new_attrs", "expected_result"),
[
# insignificant attributes
(
{ATTR_WEATHER_PRECIPITATION_UNIT: "a"},
{ATTR_WEATHER_PRECIPITATION_UNIT: "b"},
False,
),
({ATTR_WEATHER_PRESSURE_UNIT: "a"}, {ATTR_WEATHER_PRESSURE_UNIT: "b"}, False),
(
{ATTR_WEATHER_TEMPERATURE_UNIT: "a"},
{ATTR_WEATHER_TEMPERATURE_UNIT: "b"},
False,
),
(
{ATTR_WEATHER_VISIBILITY_UNIT: "a"},
{ATTR_WEATHER_VISIBILITY_UNIT: "b"},
False,
),
(
{ATTR_WEATHER_WIND_SPEED_UNIT: "a"},
{ATTR_WEATHER_WIND_SPEED_UNIT: "b"},
False,
),
(
{ATTR_WEATHER_PRECIPITATION_UNIT: "a", ATTR_WEATHER_WIND_SPEED_UNIT: "a"},
{ATTR_WEATHER_PRECIPITATION_UNIT: "b", ATTR_WEATHER_WIND_SPEED_UNIT: "a"},
False,
),
# significant attributes, close to but not significant change
(
{ATTR_WEATHER_APPARENT_TEMPERATURE: 20},
{ATTR_WEATHER_APPARENT_TEMPERATURE: 20.4},
False,
),
(
{ATTR_WEATHER_APPARENT_TEMPERATURE: 68},
{
ATTR_WEATHER_APPARENT_TEMPERATURE: 68.9,
ATTR_WEATHER_TEMPERATURE_UNIT: UnitOfTemperature.FAHRENHEIT,
},
False,
),
(
{ATTR_WEATHER_DEW_POINT: 20},
{ATTR_WEATHER_DEW_POINT: 20.4},
False,
),
(
{ATTR_WEATHER_TEMPERATURE: 20},
{ATTR_WEATHER_TEMPERATURE: 20.4},
False,
),
(
{ATTR_WEATHER_CLOUD_COVERAGE: 80},
{ATTR_WEATHER_CLOUD_COVERAGE: 80.9},
False,
),
(
{ATTR_WEATHER_HUMIDITY: 90},
{ATTR_WEATHER_HUMIDITY: 89.1},
False,
),
(
{ATTR_WEATHER_WIND_BEARING: "W"}, # W = 270°
{ATTR_WEATHER_WIND_BEARING: 269.1},
False,
),
(
{ATTR_WEATHER_WIND_BEARING: "W"},
{ATTR_WEATHER_WIND_BEARING: "W"},
False,
),
(
{ATTR_WEATHER_WIND_BEARING: 270},
{ATTR_WEATHER_WIND_BEARING: 269.1},
False,
),
(
{ATTR_WEATHER_WIND_GUST_SPEED: 5},
{
ATTR_WEATHER_WIND_GUST_SPEED: 5.9,
ATTR_WEATHER_WIND_SPEED_UNIT: UnitOfSpeed.KILOMETERS_PER_HOUR,
},
False,
),
(
{ATTR_WEATHER_WIND_GUST_SPEED: 5},
{
ATTR_WEATHER_WIND_GUST_SPEED: 5.4,
ATTR_WEATHER_WIND_SPEED_UNIT: UnitOfSpeed.METERS_PER_SECOND,
},
False,
),
(
{ATTR_WEATHER_WIND_SPEED: 5},
{
ATTR_WEATHER_WIND_SPEED: 5.9,
ATTR_WEATHER_WIND_SPEED_UNIT: UnitOfSpeed.KILOMETERS_PER_HOUR,
},
False,
),
(
{ATTR_WEATHER_WIND_SPEED: 5},
{
ATTR_WEATHER_WIND_SPEED: 5.4,
ATTR_WEATHER_WIND_SPEED_UNIT: UnitOfSpeed.METERS_PER_SECOND,
},
False,
),
(
{ATTR_WEATHER_UV_INDEX: 1},
{ATTR_WEATHER_UV_INDEX: 1.09},
False,
),
(
{ATTR_WEATHER_OZONE: 20},
{ATTR_WEATHER_OZONE: 20.9},
False,
),
(
{ATTR_WEATHER_PRESSURE: 1000},
{ATTR_WEATHER_PRESSURE: 1000.9},
False,
),
(
{ATTR_WEATHER_PRESSURE: 750.06},
{
ATTR_WEATHER_PRESSURE: 750.74,
ATTR_WEATHER_PRESSURE_UNIT: UnitOfPressure.MMHG,
},
False,
),
(
{ATTR_WEATHER_PRESSURE: 29.5},
{
ATTR_WEATHER_PRESSURE: 29.54,
ATTR_WEATHER_PRESSURE_UNIT: UnitOfPressure.INHG,
},
False,
),
# significant attributes with significant change
(
{ATTR_WEATHER_APPARENT_TEMPERATURE: 20},
{ATTR_WEATHER_APPARENT_TEMPERATURE: 20.5},
True,
),
(
{ATTR_WEATHER_APPARENT_TEMPERATURE: 68},
{
ATTR_WEATHER_APPARENT_TEMPERATURE: 69,
ATTR_WEATHER_TEMPERATURE_UNIT: UnitOfTemperature.FAHRENHEIT,
},
True,
),
(
{ATTR_WEATHER_DEW_POINT: 20},
{ATTR_WEATHER_DEW_POINT: 20.5},
True,
),
(
{ATTR_WEATHER_TEMPERATURE: 20},
{ATTR_WEATHER_TEMPERATURE: 20.5},
True,
),
(
{ATTR_WEATHER_CLOUD_COVERAGE: 80},
{ATTR_WEATHER_CLOUD_COVERAGE: 81},
True,
),
(
{ATTR_WEATHER_HUMIDITY: 90},
{ATTR_WEATHER_HUMIDITY: 89},
True,
),
(
{ATTR_WEATHER_WIND_BEARING: "W"}, # W = 270°
{ATTR_WEATHER_WIND_BEARING: 269},
True,
),
(
{ATTR_WEATHER_WIND_BEARING: "W"},
{ATTR_WEATHER_WIND_BEARING: "NW"}, # NW = 315°
True,
),
(
{ATTR_WEATHER_WIND_BEARING: 270},
{ATTR_WEATHER_WIND_BEARING: 269},
True,
),
(
{ATTR_WEATHER_WIND_GUST_SPEED: 5},
{
ATTR_WEATHER_WIND_GUST_SPEED: 6,
ATTR_WEATHER_WIND_SPEED_UNIT: UnitOfSpeed.KILOMETERS_PER_HOUR,
},
True,
),
(
{ATTR_WEATHER_WIND_GUST_SPEED: 5},
{
ATTR_WEATHER_WIND_GUST_SPEED: 5.5,
ATTR_WEATHER_WIND_SPEED_UNIT: UnitOfSpeed.METERS_PER_SECOND,
},
True,
),
(
{ATTR_WEATHER_WIND_SPEED: 5},
{
ATTR_WEATHER_WIND_SPEED: 6,
ATTR_WEATHER_WIND_SPEED_UNIT: UnitOfSpeed.KILOMETERS_PER_HOUR,
},
True,
),
(
{ATTR_WEATHER_WIND_SPEED: 5},
{
ATTR_WEATHER_WIND_SPEED: 5.5,
ATTR_WEATHER_WIND_SPEED_UNIT: UnitOfSpeed.METERS_PER_SECOND,
},
True,
),
(
{ATTR_WEATHER_UV_INDEX: 1},
{ATTR_WEATHER_UV_INDEX: 1.1},
True,
),
(
{ATTR_WEATHER_OZONE: 20},
{ATTR_WEATHER_OZONE: 21},
True,
),
(
{ATTR_WEATHER_PRESSURE: 1000},
{ATTR_WEATHER_PRESSURE: 1001},
True,
),
(
{ATTR_WEATHER_PRESSURE: 750},
{
ATTR_WEATHER_PRESSURE: 749,
ATTR_WEATHER_PRESSURE_UNIT: UnitOfPressure.MMHG,
},
True,
),
(
{ATTR_WEATHER_PRESSURE: 29.5},
{
ATTR_WEATHER_PRESSURE: 29.55,
ATTR_WEATHER_PRESSURE_UNIT: UnitOfPressure.INHG,
},
True,
),
],
)
async def test_significant_atributes_change(
old_attrs: dict, new_attrs: dict, expected_result: bool
) -> None:
"""Detect Weather significant attribute changes."""
assert (
async_check_significant_change(None, "state", old_attrs, "state", new_attrs)
== expected_result
)
@pytest.mark.parametrize(
("old_attrs", "new_attrs", "expected_result"),
[
# invalid new values
(
{ATTR_WEATHER_APPARENT_TEMPERATURE: 30},
{ATTR_WEATHER_APPARENT_TEMPERATURE: "invalid"},
False,
),
(
{ATTR_WEATHER_APPARENT_TEMPERATURE: 30},
{ATTR_WEATHER_APPARENT_TEMPERATURE: None},
False,
),
(
{ATTR_WEATHER_WIND_BEARING: "NNW"},
{ATTR_WEATHER_WIND_BEARING: "invalid"},
False,
),
# invalid old values
(
{ATTR_WEATHER_APPARENT_TEMPERATURE: "invalid"},
{ATTR_WEATHER_APPARENT_TEMPERATURE: 30},
True,
),
(
{ATTR_WEATHER_APPARENT_TEMPERATURE: None},
{ATTR_WEATHER_APPARENT_TEMPERATURE: 30},
True,
),
(
{ATTR_WEATHER_WIND_BEARING: "invalid"},
{ATTR_WEATHER_WIND_BEARING: "NNW"},
True,
),
],
)
async def test_invalid_atributes_change(
old_attrs: dict, new_attrs: dict, expected_result: bool
) -> None:
"""Detect Weather invalid attribute changes."""
assert (
async_check_significant_change(None, "state", old_attrs, "state", new_attrs)
== expected_result
)