Move temperature utility to unit_conversion (#78960)

This commit is contained in:
epenet 2022-09-22 18:31:50 +02:00 committed by GitHub
parent 090d004122
commit ddf56baf7a
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
6 changed files with 166 additions and 70 deletions

View file

@ -30,7 +30,7 @@ from homeassistant.helpers import entity_registry
from homeassistant.helpers.json import JSONEncoder from homeassistant.helpers.json import JSONEncoder
from homeassistant.helpers.storage import STORAGE_DIR from homeassistant.helpers.storage import STORAGE_DIR
from homeassistant.helpers.typing import UNDEFINED, UndefinedType from homeassistant.helpers.typing import UNDEFINED, UndefinedType
from homeassistant.util import dt as dt_util, temperature as temperature_util from homeassistant.util import dt as dt_util
from homeassistant.util.unit_conversion import ( from homeassistant.util.unit_conversion import (
BaseUnitConverter, BaseUnitConverter,
EnergyConverter, EnergyConverter,
@ -157,7 +157,9 @@ def _convert_temperature_from_c(to_unit: str, value: float | None) -> float | No
"""Convert temperature in °C to to_unit.""" """Convert temperature in °C to to_unit."""
if value is None: if value is None:
return None return None
return temperature_util.convert(value, temperature_util.NORMALIZED_UNIT, to_unit) return TemperatureConverter.convert(
value, TemperatureConverter.NORMALIZED_UNIT, to_unit
)
def _convert_volume_from_m3(to_unit: str, value: float | None) -> float | None: def _convert_volume_from_m3(to_unit: str, value: float | None) -> float | None:
@ -176,15 +178,15 @@ STATISTIC_UNIT_TO_UNIT_CLASS: dict[str | None, str] = {
EnergyConverter.NORMALIZED_UNIT: EnergyConverter.UNIT_CLASS, EnergyConverter.NORMALIZED_UNIT: EnergyConverter.UNIT_CLASS,
PowerConverter.NORMALIZED_UNIT: PowerConverter.UNIT_CLASS, PowerConverter.NORMALIZED_UNIT: PowerConverter.UNIT_CLASS,
PressureConverter.NORMALIZED_UNIT: PressureConverter.UNIT_CLASS, PressureConverter.NORMALIZED_UNIT: PressureConverter.UNIT_CLASS,
temperature_util.NORMALIZED_UNIT: "temperature", TemperatureConverter.NORMALIZED_UNIT: TemperatureConverter.UNIT_CLASS,
VolumeConverter.NORMALIZED_UNIT: "volume", VolumeConverter.NORMALIZED_UNIT: VolumeConverter.UNIT_CLASS,
} }
STATISTIC_UNIT_TO_UNIT_CONVERTER: dict[str | None, type[BaseUnitConverter]] = { STATISTIC_UNIT_TO_UNIT_CONVERTER: dict[str | None, type[BaseUnitConverter]] = {
EnergyConverter.NORMALIZED_UNIT: EnergyConverter, EnergyConverter.NORMALIZED_UNIT: EnergyConverter,
PowerConverter.NORMALIZED_UNIT: PowerConverter, PowerConverter.NORMALIZED_UNIT: PowerConverter,
PressureConverter.NORMALIZED_UNIT: PressureConverter, PressureConverter.NORMALIZED_UNIT: PressureConverter,
temperature_util.NORMALIZED_UNIT: TemperatureConverter, TemperatureConverter.NORMALIZED_UNIT: TemperatureConverter,
VolumeConverter.NORMALIZED_UNIT: VolumeConverter, VolumeConverter.NORMALIZED_UNIT: VolumeConverter,
} }
@ -196,7 +198,7 @@ STATISTIC_UNIT_TO_DISPLAY_UNIT_FUNCTIONS: dict[
EnergyConverter.NORMALIZED_UNIT: _convert_energy_from_kwh, EnergyConverter.NORMALIZED_UNIT: _convert_energy_from_kwh,
PowerConverter.NORMALIZED_UNIT: _convert_power_from_w, PowerConverter.NORMALIZED_UNIT: _convert_power_from_w,
PressureConverter.NORMALIZED_UNIT: _convert_pressure_from_pa, PressureConverter.NORMALIZED_UNIT: _convert_pressure_from_pa,
temperature_util.NORMALIZED_UNIT: _convert_temperature_from_c, TemperatureConverter.NORMALIZED_UNIT: _convert_temperature_from_c,
VolumeConverter.NORMALIZED_UNIT: _convert_volume_from_m3, VolumeConverter.NORMALIZED_UNIT: _convert_volume_from_m3,
} }

View file

@ -19,11 +19,12 @@ from homeassistant.const import (
from homeassistant.core import HomeAssistant, callback, valid_entity_id from homeassistant.core import HomeAssistant, callback, valid_entity_id
from homeassistant.helpers import config_validation as cv from homeassistant.helpers import config_validation as cv
from homeassistant.helpers.json import JSON_DUMP from homeassistant.helpers.json import JSON_DUMP
from homeassistant.util import dt as dt_util, temperature as temperature_util from homeassistant.util import dt as dt_util
from homeassistant.util.unit_conversion import ( from homeassistant.util.unit_conversion import (
EnergyConverter, EnergyConverter,
PowerConverter, PowerConverter,
PressureConverter, PressureConverter,
TemperatureConverter,
) )
from .const import MAX_QUEUE_BACKLOG from .const import MAX_QUEUE_BACKLOG
@ -123,7 +124,7 @@ async def ws_handle_get_statistics_during_period(
vol.Optional("energy"): vol.In(EnergyConverter.VALID_UNITS), vol.Optional("energy"): vol.In(EnergyConverter.VALID_UNITS),
vol.Optional("power"): vol.In(PowerConverter.VALID_UNITS), vol.Optional("power"): vol.In(PowerConverter.VALID_UNITS),
vol.Optional("pressure"): vol.In(PressureConverter.VALID_UNITS), vol.Optional("pressure"): vol.In(PressureConverter.VALID_UNITS),
vol.Optional("temperature"): vol.In(temperature_util.VALID_UNITS), vol.Optional("temperature"): vol.In(TemperatureConverter.VALID_UNITS),
vol.Optional("volume"): vol.Any(VOLUME_CUBIC_FEET, VOLUME_CUBIC_METERS), vol.Optional("volume"): vol.Any(VOLUME_CUBIC_FEET, VOLUME_CUBIC_METERS),
} }
), ),

View file

@ -47,7 +47,7 @@ from homeassistant.const import (
from homeassistant.core import HomeAssistant, State from homeassistant.core import HomeAssistant, State
from homeassistant.exceptions import HomeAssistantError from homeassistant.exceptions import HomeAssistantError
from homeassistant.helpers.entity import entity_sources from homeassistant.helpers.entity import entity_sources
from homeassistant.util import dt as dt_util, temperature as temperature_util from homeassistant.util import dt as dt_util
from homeassistant.util.unit_conversion import ( from homeassistant.util.unit_conversion import (
BaseUnitConverter, BaseUnitConverter,
EnergyConverter, EnergyConverter,
@ -111,11 +111,11 @@ UNIT_CONVERSIONS: dict[str, dict[str, Callable]] = {
PRESSURE_PSI: lambda x: x / PressureConverter.UNIT_CONVERSION[PRESSURE_PSI], PRESSURE_PSI: lambda x: x / PressureConverter.UNIT_CONVERSION[PRESSURE_PSI],
}, },
# Convert temperature to °C # Convert temperature to °C
# Note: temperature_util.convert is bypassed to avoid redundant error checking # Note: TemperatureConverter.convert is bypassed to avoid redundant error checking
SensorDeviceClass.TEMPERATURE: { SensorDeviceClass.TEMPERATURE: {
TEMP_CELSIUS: lambda x: x, TEMP_CELSIUS: lambda x: x,
TEMP_FAHRENHEIT: temperature_util.fahrenheit_to_celsius, TEMP_FAHRENHEIT: TemperatureConverter.fahrenheit_to_celsius,
TEMP_KELVIN: temperature_util.kelvin_to_celsius, TEMP_KELVIN: TemperatureConverter.kelvin_to_celsius,
}, },
# Convert volume to cubic meter # Convert volume to cubic meter
SensorDeviceClass.GAS: { SensorDeviceClass.GAS: {

View file

@ -1,5 +1,5 @@
"""Temperature util functions.""" """Temperature util functions."""
from homeassistant.const import ( from homeassistant.const import ( # pylint: disable=unused-import # noqa: F401
TEMP_CELSIUS, TEMP_CELSIUS,
TEMP_FAHRENHEIT, TEMP_FAHRENHEIT,
TEMP_KELVIN, TEMP_KELVIN,
@ -7,69 +7,40 @@ from homeassistant.const import (
UNIT_NOT_RECOGNIZED_TEMPLATE, UNIT_NOT_RECOGNIZED_TEMPLATE,
) )
VALID_UNITS: tuple[str, ...] = ( from .unit_conversion import TemperatureConverter
TEMP_CELSIUS,
TEMP_FAHRENHEIT,
TEMP_KELVIN,
)
NORMALIZED_UNIT = TEMP_CELSIUS VALID_UNITS = TemperatureConverter.VALID_UNITS
def fahrenheit_to_celsius(fahrenheit: float, interval: bool = False) -> float: def fahrenheit_to_celsius(fahrenheit: float, interval: bool = False) -> float:
"""Convert a temperature in Fahrenheit to Celsius.""" """Convert a temperature in Fahrenheit to Celsius."""
if interval: # Need to add warning when core migration finished
return fahrenheit / 1.8 return TemperatureConverter.fahrenheit_to_celsius(fahrenheit, interval)
return (fahrenheit - 32.0) / 1.8
def kelvin_to_celsius(kelvin: float, interval: bool = False) -> float: def kelvin_to_celsius(kelvin: float, interval: bool = False) -> float:
"""Convert a temperature in Kelvin to Celsius.""" """Convert a temperature in Kelvin to Celsius."""
if interval: # Need to add warning when core migration finished
return kelvin return TemperatureConverter.kelvin_to_celsius(kelvin, interval)
return kelvin - 273.15
def celsius_to_fahrenheit(celsius: float, interval: bool = False) -> float: def celsius_to_fahrenheit(celsius: float, interval: bool = False) -> float:
"""Convert a temperature in Celsius to Fahrenheit.""" """Convert a temperature in Celsius to Fahrenheit."""
if interval: # Need to add warning when core migration finished
return celsius * 1.8 return TemperatureConverter.celsius_to_fahrenheit(celsius, interval)
return celsius * 1.8 + 32.0
def celsius_to_kelvin(celsius: float, interval: bool = False) -> float: def celsius_to_kelvin(celsius: float, interval: bool = False) -> float:
"""Convert a temperature in Celsius to Fahrenheit.""" """Convert a temperature in Celsius to Fahrenheit."""
if interval: # Need to add warning when core migration finished
return celsius return TemperatureConverter.celsius_to_kelvin(celsius, interval)
return celsius + 273.15
def convert( def convert(
temperature: float, from_unit: str, to_unit: str, interval: bool = False temperature: float, from_unit: str, to_unit: str, interval: bool = False
) -> float: ) -> float:
"""Convert a temperature from one unit to another.""" """Convert a temperature from one unit to another."""
if from_unit not in VALID_UNITS: # Need to add warning when core migration finished
raise ValueError(UNIT_NOT_RECOGNIZED_TEMPLATE.format(from_unit, TEMPERATURE)) return TemperatureConverter.convert(
if to_unit not in VALID_UNITS: temperature, from_unit, to_unit, interval=interval
raise ValueError(UNIT_NOT_RECOGNIZED_TEMPLATE.format(to_unit, TEMPERATURE)) )
if from_unit == to_unit:
return temperature
if from_unit == TEMP_CELSIUS:
if to_unit == TEMP_FAHRENHEIT:
return celsius_to_fahrenheit(temperature, interval)
# kelvin
return celsius_to_kelvin(temperature, interval)
if from_unit == TEMP_FAHRENHEIT:
if to_unit == TEMP_CELSIUS:
return fahrenheit_to_celsius(temperature, interval)
# kelvin
return celsius_to_kelvin(fahrenheit_to_celsius(temperature, interval), interval)
# from_unit == kelvin
if to_unit == TEMP_CELSIUS:
return kelvin_to_celsius(temperature, interval)
# fahrenheit
return celsius_to_fahrenheit(kelvin_to_celsius(temperature, interval), interval)

View file

@ -1,7 +1,7 @@
"""Typing Helpers for Home Assistant.""" """Typing Helpers for Home Assistant."""
from __future__ import annotations from __future__ import annotations
from collections.abc import Callable from abc import abstractmethod
from numbers import Number from numbers import Number
from homeassistant.const import ( from homeassistant.const import (
@ -20,6 +20,8 @@ from homeassistant.const import (
PRESSURE_PA, PRESSURE_PA,
PRESSURE_PSI, PRESSURE_PSI,
TEMP_CELSIUS, TEMP_CELSIUS,
TEMP_FAHRENHEIT,
TEMP_KELVIN,
UNIT_NOT_RECOGNIZED_TEMPLATE, UNIT_NOT_RECOGNIZED_TEMPLATE,
VOLUME_CUBIC_FEET, VOLUME_CUBIC_FEET,
VOLUME_CUBIC_METERS, VOLUME_CUBIC_METERS,
@ -29,7 +31,6 @@ from homeassistant.const import (
VOLUME_MILLILITERS, VOLUME_MILLILITERS,
) )
from . import temperature as temperature_util
from .distance import FOOT_TO_M, IN_TO_M from .distance import FOOT_TO_M, IN_TO_M
# Volume conversion constants # Volume conversion constants
@ -43,20 +44,13 @@ _CUBIC_FOOT_TO_CUBIC_METER = pow(FOOT_TO_M, 3)
class BaseUnitConverter: class BaseUnitConverter:
"""Define the format of a conversion utility.""" """Define the format of a conversion utility."""
UNIT_CLASS: str
NORMALIZED_UNIT: str NORMALIZED_UNIT: str
VALID_UNITS: tuple[str, ...] VALID_UNITS: tuple[str, ...]
convert: Callable[[float, str, str], float]
class BaseUnitConverterWithUnitConversion(BaseUnitConverter):
"""Define the format of a conversion utility."""
UNIT_CLASS: str
UNIT_CONVERSION: dict[str, float]
@classmethod @classmethod
def convert(cls, value: float, from_unit: str, to_unit: str) -> float: def _check_arguments(cls, value: float, from_unit: str, to_unit: str) -> None:
"""Convert one unit of measurement to another.""" """Check that arguments are all valid."""
if from_unit not in cls.VALID_UNITS: if from_unit not in cls.VALID_UNITS:
raise ValueError( raise ValueError(
UNIT_NOT_RECOGNIZED_TEMPLATE.format(from_unit, cls.UNIT_CLASS) UNIT_NOT_RECOGNIZED_TEMPLATE.format(from_unit, cls.UNIT_CLASS)
@ -69,6 +63,22 @@ class BaseUnitConverterWithUnitConversion(BaseUnitConverter):
if not isinstance(value, Number): if not isinstance(value, Number):
raise TypeError(f"{value} is not of numeric type") raise TypeError(f"{value} is not of numeric type")
@classmethod
@abstractmethod
def convert(cls, value: float, from_unit: str, to_unit: str) -> float:
"""Convert one unit of measurement to another."""
class BaseUnitConverterWithUnitConversion(BaseUnitConverter):
"""Define the format of a conversion utility."""
UNIT_CONVERSION: dict[str, float]
@classmethod
def convert(cls, value: float, from_unit: str, to_unit: str) -> float:
"""Convert one unit of measurement to another."""
cls._check_arguments(value, from_unit, to_unit)
if from_unit == to_unit: if from_unit == to_unit:
return value return value
@ -140,9 +150,73 @@ class PressureConverter(BaseUnitConverterWithUnitConversion):
class TemperatureConverter(BaseUnitConverter): class TemperatureConverter(BaseUnitConverter):
"""Utility to convert temperature values.""" """Utility to convert temperature values."""
UNIT_CLASS = "temperature"
NORMALIZED_UNIT = TEMP_CELSIUS NORMALIZED_UNIT = TEMP_CELSIUS
VALID_UNITS = temperature_util.VALID_UNITS VALID_UNITS: tuple[str, ...] = (
convert = temperature_util.convert TEMP_CELSIUS,
TEMP_FAHRENHEIT,
TEMP_KELVIN,
)
@classmethod
def convert(
cls, value: float, from_unit: str, to_unit: str, *, interval: bool = False
) -> float:
"""Convert a temperature from one unit to another."""
cls._check_arguments(value, from_unit, to_unit)
if from_unit == to_unit:
return value
if from_unit == TEMP_CELSIUS:
if to_unit == TEMP_FAHRENHEIT:
return cls.celsius_to_fahrenheit(value, interval)
# kelvin
return cls.celsius_to_kelvin(value, interval)
if from_unit == TEMP_FAHRENHEIT:
if to_unit == TEMP_CELSIUS:
return cls.fahrenheit_to_celsius(value, interval)
# kelvin
return cls.celsius_to_kelvin(
cls.fahrenheit_to_celsius(value, interval), interval
)
# from_unit == kelvin
if to_unit == TEMP_CELSIUS:
return cls.kelvin_to_celsius(value, interval)
# fahrenheit
return cls.celsius_to_fahrenheit(
cls.kelvin_to_celsius(value, interval), interval
)
@classmethod
def fahrenheit_to_celsius(cls, fahrenheit: float, interval: bool = False) -> float:
"""Convert a temperature in Fahrenheit to Celsius."""
if interval:
return fahrenheit / 1.8
return (fahrenheit - 32.0) / 1.8
@classmethod
def kelvin_to_celsius(cls, kelvin: float, interval: bool = False) -> float:
"""Convert a temperature in Kelvin to Celsius."""
if interval:
return kelvin
return kelvin - 273.15
@classmethod
def celsius_to_fahrenheit(cls, celsius: float, interval: bool = False) -> float:
"""Convert a temperature in Celsius to Fahrenheit."""
if interval:
return celsius * 1.8
return celsius * 1.8 + 32.0
@classmethod
def celsius_to_kelvin(cls, celsius: float, interval: bool = False) -> float:
"""Convert a temperature in Celsius to Kelvin."""
if interval:
return celsius
return celsius + 273.15
class VolumeConverter(BaseUnitConverterWithUnitConversion): class VolumeConverter(BaseUnitConverterWithUnitConversion):

View file

@ -15,6 +15,9 @@ from homeassistant.const import (
PRESSURE_MMHG, PRESSURE_MMHG,
PRESSURE_PA, PRESSURE_PA,
PRESSURE_PSI, PRESSURE_PSI,
TEMP_CELSIUS,
TEMP_FAHRENHEIT,
TEMP_KELVIN,
VOLUME_CUBIC_FEET, VOLUME_CUBIC_FEET,
VOLUME_CUBIC_METERS, VOLUME_CUBIC_METERS,
VOLUME_FLUID_OUNCE, VOLUME_FLUID_OUNCE,
@ -27,6 +30,7 @@ from homeassistant.util.unit_conversion import (
EnergyConverter, EnergyConverter,
PowerConverter, PowerConverter,
PressureConverter, PressureConverter,
TemperatureConverter,
VolumeConverter, VolumeConverter,
) )
@ -49,6 +53,9 @@ INVALID_SYMBOL = "bob"
(PressureConverter, PRESSURE_CBAR), (PressureConverter, PRESSURE_CBAR),
(PressureConverter, PRESSURE_MMHG), (PressureConverter, PRESSURE_MMHG),
(PressureConverter, PRESSURE_PSI), (PressureConverter, PRESSURE_PSI),
(TemperatureConverter, TEMP_CELSIUS),
(TemperatureConverter, TEMP_FAHRENHEIT),
(TemperatureConverter, TEMP_KELVIN),
(VolumeConverter, VOLUME_LITERS), (VolumeConverter, VOLUME_LITERS),
(VolumeConverter, VOLUME_MILLILITERS), (VolumeConverter, VOLUME_MILLILITERS),
(VolumeConverter, VOLUME_GALLONS), (VolumeConverter, VOLUME_GALLONS),
@ -66,6 +73,7 @@ def test_convert_same_unit(converter: type[BaseUnitConverter], valid_unit: str)
(EnergyConverter, ENERGY_KILO_WATT_HOUR), (EnergyConverter, ENERGY_KILO_WATT_HOUR),
(PowerConverter, POWER_WATT), (PowerConverter, POWER_WATT),
(PressureConverter, PRESSURE_PA), (PressureConverter, PRESSURE_PA),
(TemperatureConverter, TEMP_CELSIUS),
(VolumeConverter, VOLUME_LITERS), (VolumeConverter, VOLUME_LITERS),
], ],
) )
@ -86,6 +94,7 @@ def test_convert_invalid_unit(
(EnergyConverter, ENERGY_WATT_HOUR, ENERGY_KILO_WATT_HOUR), (EnergyConverter, ENERGY_WATT_HOUR, ENERGY_KILO_WATT_HOUR),
(PowerConverter, POWER_WATT, POWER_KILO_WATT), (PowerConverter, POWER_WATT, POWER_KILO_WATT),
(PressureConverter, PRESSURE_HPA, PRESSURE_INHG), (PressureConverter, PRESSURE_HPA, PRESSURE_INHG),
(TemperatureConverter, TEMP_CELSIUS, TEMP_FAHRENHEIT),
(VolumeConverter, VOLUME_GALLONS, VOLUME_LITERS), (VolumeConverter, VOLUME_GALLONS, VOLUME_LITERS),
], ],
) )
@ -176,6 +185,45 @@ def test_pressure_convert(
assert PressureConverter.convert(value, from_unit, to_unit) == expected assert PressureConverter.convert(value, from_unit, to_unit) == expected
@pytest.mark.parametrize(
"value,from_unit,expected,to_unit",
[
(100, TEMP_CELSIUS, 212, TEMP_FAHRENHEIT),
(100, TEMP_CELSIUS, 373.15, TEMP_KELVIN),
(100, TEMP_FAHRENHEIT, pytest.approx(37.77777777777778), TEMP_CELSIUS),
(100, TEMP_FAHRENHEIT, pytest.approx(310.92777777777775), TEMP_KELVIN),
(100, TEMP_KELVIN, pytest.approx(-173.15), TEMP_CELSIUS),
(100, TEMP_KELVIN, pytest.approx(-279.66999999999996), TEMP_FAHRENHEIT),
],
)
def test_temperature_convert(
value: float, from_unit: str, expected: float, to_unit: str
) -> None:
"""Test conversion to other units."""
assert TemperatureConverter.convert(value, from_unit, to_unit) == expected
@pytest.mark.parametrize(
"value,from_unit,expected,to_unit",
[
(100, TEMP_CELSIUS, 180, TEMP_FAHRENHEIT),
(100, TEMP_CELSIUS, 100, TEMP_KELVIN),
(100, TEMP_FAHRENHEIT, pytest.approx(55.55555555555556), TEMP_CELSIUS),
(100, TEMP_FAHRENHEIT, pytest.approx(55.55555555555556), TEMP_KELVIN),
(100, TEMP_KELVIN, 100, TEMP_CELSIUS),
(100, TEMP_KELVIN, 180, TEMP_FAHRENHEIT),
],
)
def test_temperature_convert_with_interval(
value: float, from_unit: str, expected: float, to_unit: str
) -> None:
"""Test conversion to other units."""
assert (
TemperatureConverter.convert(value, from_unit, to_unit, interval=True)
== expected
)
@pytest.mark.parametrize( @pytest.mark.parametrize(
"value,from_unit,expected,to_unit", "value,from_unit,expected,to_unit",
[ [