Add more sensors to Tuya weather station (#150442)

Co-authored-by: Franck Nijhof <frenck@frenck.nl>
Co-authored-by: epenet <6771947+epenet@users.noreply.github.com>
This commit is contained in:
Nippey
2025-08-12 11:33:51 +02:00
committed by GitHub
parent fb4a452872
commit 596e4883b1
4 changed files with 287 additions and 0 deletions

View File

@@ -24,6 +24,7 @@ from homeassistant.const import (
UnitOfPressure, UnitOfPressure,
UnitOfTemperature, UnitOfTemperature,
UnitOfVolume, UnitOfVolume,
UnitOfVolumetricFlux,
) )
DOMAIN = "tuya" DOMAIN = "tuya"
@@ -296,6 +297,8 @@ class DPCode(StrEnum):
PUMP_RESET = "pump_reset" # Water pump reset PUMP_RESET = "pump_reset" # Water pump reset
PUMP_TIME = "pump_time" # Water pump duration PUMP_TIME = "pump_time" # Water pump duration
OXYGEN = "oxygen" # Oxygen bar OXYGEN = "oxygen" # Oxygen bar
RAIN_24H = "rain_24h" # Total daily rainfall in mm
RAIN_RATE = "rain_rate" # Rain intensity in mm/h
RECORD_MODE = "record_mode" RECORD_MODE = "record_mode"
RECORD_SWITCH = "record_switch" # Recording switch RECORD_SWITCH = "record_switch" # Recording switch
RELAY_STATUS = "relay_status" RELAY_STATUS = "relay_status"
@@ -415,6 +418,7 @@ class DPCode(StrEnum):
UPPER_TEMP = "upper_temp" UPPER_TEMP = "upper_temp"
UPPER_TEMP_F = "upper_temp_f" UPPER_TEMP_F = "upper_temp_f"
UV = "uv" # UV sterilization UV = "uv" # UV sterilization
UV_INDEX = "uv_index"
UV_RUNTIME = "uv_runtime" # UV runtime UV_RUNTIME = "uv_runtime" # UV runtime
VA_BATTERY = "va_battery" VA_BATTERY = "va_battery"
VA_HUMIDITY = "va_humidity" VA_HUMIDITY = "va_humidity"
@@ -439,6 +443,7 @@ class DPCode(StrEnum):
WINDOW_STATE = "window_state" WINDOW_STATE = "window_state"
WINDSPEED = "windspeed" WINDSPEED = "windspeed"
WINDSPEED_AVG = "windspeed_avg" WINDSPEED_AVG = "windspeed_avg"
WIND_DIRECT = "wind_direct"
WIRELESS_BATTERYLOCK = "wireless_batterylock" WIRELESS_BATTERYLOCK = "wireless_batterylock"
WIRELESS_ELECTRICITY = "wireless_electricity" WIRELESS_ELECTRICITY = "wireless_electricity"
WORK_MODE = "work_mode" # Working mode WORK_MODE = "work_mode" # Working mode
@@ -524,6 +529,11 @@ UNITS = (
aliases={"m3"}, aliases={"m3"},
device_classes={SensorDeviceClass.GAS}, device_classes={SensorDeviceClass.GAS},
), ),
UnitOfMeasurement(
unit=UnitOfVolumetricFlux.MILLIMETERS_PER_HOUR,
aliases={"mm"},
device_classes={SensorDeviceClass.PRECIPITATION_INTENSITY},
),
UnitOfMeasurement( UnitOfMeasurement(
unit=LIGHT_LUX, unit=LIGHT_LUX,
aliases={"lux"}, aliases={"lux"},

View File

@@ -2,7 +2,9 @@
from __future__ import annotations from __future__ import annotations
from collections.abc import Callable
from dataclasses import dataclass from dataclasses import dataclass
from typing import Any
from tuya_sharing import CustomerDevice, Manager from tuya_sharing import CustomerDevice, Manager
from tuya_sharing.device import DeviceStatusRange from tuya_sharing.device import DeviceStatusRange
@@ -42,6 +44,25 @@ from .const import (
from .entity import TuyaEntity from .entity import TuyaEntity
from .models import ComplexTypeData, ElectricityTypeData, EnumTypeData, IntegerTypeData from .models import ComplexTypeData, ElectricityTypeData, EnumTypeData, IntegerTypeData
_WIND_DIRECTIONS = {
"north": 0.0,
"north_north_east": 22.5,
"north_east": 45.0,
"east_north_east": 67.5,
"east": 90.0,
"east_south_east": 112.5,
"south_east": 135.0,
"south_south_east": 157.5,
"south": 180.0,
"south_south_west": 202.5,
"south_west": 225.0,
"west_south_west": 247.5,
"west": 270.0,
"west_north_west": 292.5,
"north_west": 315.0,
"north_north_west": 337.5,
}
@dataclass(frozen=True) @dataclass(frozen=True)
class TuyaSensorEntityDescription(SensorEntityDescription): class TuyaSensorEntityDescription(SensorEntityDescription):
@@ -49,6 +70,7 @@ class TuyaSensorEntityDescription(SensorEntityDescription):
complex_type: type[ComplexTypeData] | None = None complex_type: type[ComplexTypeData] | None = None
subkey: str | None = None subkey: str | None = None
state_conversion: Callable[[Any], StateType] | None = None
# Commonly used battery sensors, that are reused in the sensors down below. # Commonly used battery sensors, that are reused in the sensors down below.
@@ -931,6 +953,30 @@ SENSORS: dict[str, tuple[TuyaSensorEntityDescription, ...]] = {
device_class=SensorDeviceClass.WIND_SPEED, device_class=SensorDeviceClass.WIND_SPEED,
state_class=SensorStateClass.MEASUREMENT, state_class=SensorStateClass.MEASUREMENT,
), ),
TuyaSensorEntityDescription(
key=DPCode.RAIN_24H,
translation_key="precipitation_today",
device_class=SensorDeviceClass.PRECIPITATION,
state_class=SensorStateClass.TOTAL_INCREASING,
),
TuyaSensorEntityDescription(
key=DPCode.RAIN_RATE,
translation_key="precipitation_intensity",
device_class=SensorDeviceClass.PRECIPITATION_INTENSITY,
state_class=SensorStateClass.MEASUREMENT,
),
TuyaSensorEntityDescription(
key=DPCode.UV_INDEX,
translation_key="uv_index",
state_class=SensorStateClass.MEASUREMENT,
),
TuyaSensorEntityDescription(
key=DPCode.WIND_DIRECT,
translation_key="wind_direction",
device_class=SensorDeviceClass.WIND_DIRECTION,
state_class=SensorStateClass.MEASUREMENT,
state_conversion=lambda state: _WIND_DIRECTIONS.get(str(state)),
),
*BATTERY_SENSORS, *BATTERY_SENSORS,
), ),
# Gas Detector # Gas Detector
@@ -1625,6 +1671,10 @@ class TuyaSensorEntity(TuyaEntity, SensorEntity):
if value is None: if value is None:
return None return None
# Convert value, if required
if (convert := self.entity_description.state_conversion) is not None:
return convert(value)
# Scale integer/float value # Scale integer/float value
if isinstance(self._type_data, IntegerTypeData): if isinstance(self._type_data, IntegerTypeData):
return self._type_data.scale_value(value) return self._type_data.scale_value(value)

View File

@@ -535,6 +535,18 @@
"air_pressure": { "air_pressure": {
"name": "Air pressure" "name": "Air pressure"
}, },
"precipitation_today": {
"name": "Total precipitation today"
},
"precipitation_intensity": {
"name": "[%key:component::sensor::entity_component::precipitation_intensity::name%]"
},
"uv_index": {
"name": "UV index"
},
"wind_direction": {
"name": "[%key:component::sensor::entity_component::wind_direction::name%]"
},
"pm25": { "pm25": {
"name": "[%key:component::sensor::entity_component::pm25::name%]" "name": "[%key:component::sensor::entity_component::pm25::name%]"
}, },

View File

@@ -1867,6 +1867,62 @@
'state': '0.0', 'state': '0.0',
}) })
# --- # ---
# name: test_platform_setup_and_discovery[sensor.br_7_in_1_wlan_wetterstation_anthrazit_precipitation_intensity-entry]
EntityRegistryEntrySnapshot({
'aliases': set({
}),
'area_id': None,
'capabilities': dict({
'state_class': <SensorStateClass.MEASUREMENT: 'measurement'>,
}),
'config_entry_id': <ANY>,
'config_subentry_id': <ANY>,
'device_class': None,
'device_id': <ANY>,
'disabled_by': None,
'domain': 'sensor',
'entity_category': None,
'entity_id': 'sensor.br_7_in_1_wlan_wetterstation_anthrazit_precipitation_intensity',
'has_entity_name': True,
'hidden_by': None,
'icon': None,
'id': <ANY>,
'labels': set({
}),
'name': None,
'options': dict({
'sensor': dict({
'suggested_display_precision': 0,
}),
}),
'original_device_class': <SensorDeviceClass.PRECIPITATION_INTENSITY: 'precipitation_intensity'>,
'original_icon': None,
'original_name': 'Precipitation intensity',
'platform': 'tuya',
'previous_unique_id': None,
'suggested_object_id': None,
'supported_features': 0,
'translation_key': 'precipitation_intensity',
'unique_id': 'tuya.6tbtkuv3tal1aesfjxqrain_rate',
'unit_of_measurement': <UnitOfVolumetricFlux.MILLIMETERS_PER_HOUR: 'mm/h'>,
})
# ---
# name: test_platform_setup_and_discovery[sensor.br_7_in_1_wlan_wetterstation_anthrazit_precipitation_intensity-state]
StateSnapshot({
'attributes': ReadOnlyDict({
'device_class': 'precipitation_intensity',
'friendly_name': 'BR 7-in-1 WLAN Wetterstation Anthrazit Precipitation intensity',
'state_class': <SensorStateClass.MEASUREMENT: 'measurement'>,
'unit_of_measurement': <UnitOfVolumetricFlux.MILLIMETERS_PER_HOUR: 'mm/h'>,
}),
'context': <ANY>,
'entity_id': 'sensor.br_7_in_1_wlan_wetterstation_anthrazit_precipitation_intensity',
'last_changed': <ANY>,
'last_reported': <ANY>,
'last_updated': <ANY>,
'state': '0.0',
})
# ---
# name: test_platform_setup_and_discovery[sensor.br_7_in_1_wlan_wetterstation_anthrazit_probe_temperature-entry] # name: test_platform_setup_and_discovery[sensor.br_7_in_1_wlan_wetterstation_anthrazit_probe_temperature-entry]
EntityRegistryEntrySnapshot({ EntityRegistryEntrySnapshot({
'aliases': set({ 'aliases': set({
@@ -2147,6 +2203,165 @@
'state': '24.0', 'state': '24.0',
}) })
# --- # ---
# name: test_platform_setup_and_discovery[sensor.br_7_in_1_wlan_wetterstation_anthrazit_total_precipitation_today-entry]
EntityRegistryEntrySnapshot({
'aliases': set({
}),
'area_id': None,
'capabilities': dict({
'state_class': <SensorStateClass.TOTAL_INCREASING: 'total_increasing'>,
}),
'config_entry_id': <ANY>,
'config_subentry_id': <ANY>,
'device_class': None,
'device_id': <ANY>,
'disabled_by': None,
'domain': 'sensor',
'entity_category': None,
'entity_id': 'sensor.br_7_in_1_wlan_wetterstation_anthrazit_total_precipitation_today',
'has_entity_name': True,
'hidden_by': None,
'icon': None,
'id': <ANY>,
'labels': set({
}),
'name': None,
'options': dict({
'sensor': dict({
'suggested_display_precision': 0,
}),
}),
'original_device_class': <SensorDeviceClass.PRECIPITATION: 'precipitation'>,
'original_icon': None,
'original_name': 'Total precipitation today',
'platform': 'tuya',
'previous_unique_id': None,
'suggested_object_id': None,
'supported_features': 0,
'translation_key': 'precipitation_today',
'unique_id': 'tuya.6tbtkuv3tal1aesfjxqrain_24h',
'unit_of_measurement': 'mm',
})
# ---
# name: test_platform_setup_and_discovery[sensor.br_7_in_1_wlan_wetterstation_anthrazit_total_precipitation_today-state]
StateSnapshot({
'attributes': ReadOnlyDict({
'device_class': 'precipitation',
'friendly_name': 'BR 7-in-1 WLAN Wetterstation Anthrazit Total precipitation today',
'state_class': <SensorStateClass.TOTAL_INCREASING: 'total_increasing'>,
'unit_of_measurement': 'mm',
}),
'context': <ANY>,
'entity_id': 'sensor.br_7_in_1_wlan_wetterstation_anthrazit_total_precipitation_today',
'last_changed': <ANY>,
'last_reported': <ANY>,
'last_updated': <ANY>,
'state': '0.0',
})
# ---
# name: test_platform_setup_and_discovery[sensor.br_7_in_1_wlan_wetterstation_anthrazit_uv_index-entry]
EntityRegistryEntrySnapshot({
'aliases': set({
}),
'area_id': None,
'capabilities': dict({
'state_class': <SensorStateClass.MEASUREMENT: 'measurement'>,
}),
'config_entry_id': <ANY>,
'config_subentry_id': <ANY>,
'device_class': None,
'device_id': <ANY>,
'disabled_by': None,
'domain': 'sensor',
'entity_category': None,
'entity_id': 'sensor.br_7_in_1_wlan_wetterstation_anthrazit_uv_index',
'has_entity_name': True,
'hidden_by': None,
'icon': None,
'id': <ANY>,
'labels': set({
}),
'name': None,
'options': dict({
}),
'original_device_class': None,
'original_icon': None,
'original_name': 'UV index',
'platform': 'tuya',
'previous_unique_id': None,
'suggested_object_id': None,
'supported_features': 0,
'translation_key': 'uv_index',
'unique_id': 'tuya.6tbtkuv3tal1aesfjxquv_index',
'unit_of_measurement': '',
})
# ---
# name: test_platform_setup_and_discovery[sensor.br_7_in_1_wlan_wetterstation_anthrazit_uv_index-state]
StateSnapshot({
'attributes': ReadOnlyDict({
'friendly_name': 'BR 7-in-1 WLAN Wetterstation Anthrazit UV index',
'state_class': <SensorStateClass.MEASUREMENT: 'measurement'>,
'unit_of_measurement': '',
}),
'context': <ANY>,
'entity_id': 'sensor.br_7_in_1_wlan_wetterstation_anthrazit_uv_index',
'last_changed': <ANY>,
'last_reported': <ANY>,
'last_updated': <ANY>,
'state': '0.0',
})
# ---
# name: test_platform_setup_and_discovery[sensor.br_7_in_1_wlan_wetterstation_anthrazit_wind_direction-entry]
EntityRegistryEntrySnapshot({
'aliases': set({
}),
'area_id': None,
'capabilities': dict({
'state_class': <SensorStateClass.MEASUREMENT: 'measurement'>,
}),
'config_entry_id': <ANY>,
'config_subentry_id': <ANY>,
'device_class': None,
'device_id': <ANY>,
'disabled_by': None,
'domain': 'sensor',
'entity_category': None,
'entity_id': 'sensor.br_7_in_1_wlan_wetterstation_anthrazit_wind_direction',
'has_entity_name': True,
'hidden_by': None,
'icon': None,
'id': <ANY>,
'labels': set({
}),
'name': None,
'options': dict({
}),
'original_device_class': None,
'original_icon': None,
'original_name': 'Wind direction',
'platform': 'tuya',
'previous_unique_id': None,
'suggested_object_id': None,
'supported_features': 0,
'translation_key': 'wind_direction',
'unique_id': 'tuya.6tbtkuv3tal1aesfjxqwind_direct',
'unit_of_measurement': None,
})
# ---
# name: test_platform_setup_and_discovery[sensor.br_7_in_1_wlan_wetterstation_anthrazit_wind_direction-state]
StateSnapshot({
'attributes': ReadOnlyDict({
'friendly_name': 'BR 7-in-1 WLAN Wetterstation Anthrazit Wind direction',
'state_class': <SensorStateClass.MEASUREMENT: 'measurement'>,
}),
'context': <ANY>,
'entity_id': 'sensor.br_7_in_1_wlan_wetterstation_anthrazit_wind_direction',
'last_changed': <ANY>,
'last_reported': <ANY>,
'last_updated': <ANY>,
'state': 'unknown',
})
# ---
# name: test_platform_setup_and_discovery[sensor.br_7_in_1_wlan_wetterstation_anthrazit_wind_speed-entry] # name: test_platform_setup_and_discovery[sensor.br_7_in_1_wlan_wetterstation_anthrazit_wind_speed-entry]
EntityRegistryEntrySnapshot({ EntityRegistryEntrySnapshot({
'aliases': set({ 'aliases': set({