mirror of
https://github.com/home-assistant/core.git
synced 2025-07-09 14:27:07 +00:00
Add sensor to gardena (#96691)
This commit is contained in:
parent
13ac8d00f9
commit
088d04fe0f
@ -410,6 +410,7 @@ omit =
|
|||||||
homeassistant/components/gardena_bluetooth/const.py
|
homeassistant/components/gardena_bluetooth/const.py
|
||||||
homeassistant/components/gardena_bluetooth/coordinator.py
|
homeassistant/components/gardena_bluetooth/coordinator.py
|
||||||
homeassistant/components/gardena_bluetooth/number.py
|
homeassistant/components/gardena_bluetooth/number.py
|
||||||
|
homeassistant/components/gardena_bluetooth/sensor.py
|
||||||
homeassistant/components/gardena_bluetooth/switch.py
|
homeassistant/components/gardena_bluetooth/switch.py
|
||||||
homeassistant/components/gc100/*
|
homeassistant/components/gc100/*
|
||||||
homeassistant/components/geniushub/*
|
homeassistant/components/geniushub/*
|
||||||
|
@ -20,7 +20,7 @@ import homeassistant.util.dt as dt_util
|
|||||||
from .const import DOMAIN
|
from .const import DOMAIN
|
||||||
from .coordinator import Coordinator, DeviceUnavailable
|
from .coordinator import Coordinator, DeviceUnavailable
|
||||||
|
|
||||||
PLATFORMS: list[Platform] = [Platform.NUMBER, Platform.SWITCH]
|
PLATFORMS: list[Platform] = [Platform.NUMBER, Platform.SENSOR, Platform.SWITCH]
|
||||||
LOGGER = logging.getLogger(__name__)
|
LOGGER = logging.getLogger(__name__)
|
||||||
TIMEOUT = 20.0
|
TIMEOUT = 20.0
|
||||||
DISCONNECT_DELAY = 5
|
DISCONNECT_DELAY = 5
|
||||||
|
@ -80,7 +80,7 @@ class Coordinator(DataUpdateCoordinator[dict[str, bytes]]):
|
|||||||
) from exception
|
) from exception
|
||||||
return data
|
return data
|
||||||
|
|
||||||
def read_cached(
|
def get_cached(
|
||||||
self, char: Characteristic[CharacteristicType]
|
self, char: Characteristic[CharacteristicType]
|
||||||
) -> CharacteristicType | None:
|
) -> CharacteristicType | None:
|
||||||
"""Read cached characteristic."""
|
"""Read cached characteristic."""
|
||||||
|
125
homeassistant/components/gardena_bluetooth/sensor.py
Normal file
125
homeassistant/components/gardena_bluetooth/sensor.py
Normal file
@ -0,0 +1,125 @@
|
|||||||
|
"""Support for switch entities."""
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
from dataclasses import dataclass, field
|
||||||
|
from datetime import datetime, timedelta, timezone
|
||||||
|
|
||||||
|
from gardena_bluetooth.const import Battery, Valve
|
||||||
|
from gardena_bluetooth.parse import Characteristic
|
||||||
|
|
||||||
|
from homeassistant.components.sensor import (
|
||||||
|
SensorDeviceClass,
|
||||||
|
SensorEntity,
|
||||||
|
SensorEntityDescription,
|
||||||
|
SensorStateClass,
|
||||||
|
)
|
||||||
|
from homeassistant.config_entries import ConfigEntry
|
||||||
|
from homeassistant.const import PERCENTAGE, EntityCategory
|
||||||
|
from homeassistant.core import HomeAssistant
|
||||||
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||||
|
import homeassistant.util.dt as dt_util
|
||||||
|
|
||||||
|
from .const import DOMAIN
|
||||||
|
from .coordinator import Coordinator, GardenaBluetoothEntity
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass
|
||||||
|
class GardenaBluetoothSensorEntityDescription(SensorEntityDescription):
|
||||||
|
"""Description of entity."""
|
||||||
|
|
||||||
|
char: Characteristic = field(default_factory=lambda: Characteristic(""))
|
||||||
|
|
||||||
|
|
||||||
|
DESCRIPTIONS = (
|
||||||
|
GardenaBluetoothSensorEntityDescription(
|
||||||
|
key=Valve.activation_reason.uuid,
|
||||||
|
translation_key="activation_reason",
|
||||||
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
|
entity_category=EntityCategory.DIAGNOSTIC,
|
||||||
|
entity_registry_enabled_default=False,
|
||||||
|
char=Valve.activation_reason,
|
||||||
|
),
|
||||||
|
GardenaBluetoothSensorEntityDescription(
|
||||||
|
key=Battery.battery_level.uuid,
|
||||||
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
|
device_class=SensorDeviceClass.BATTERY,
|
||||||
|
entity_category=EntityCategory.DIAGNOSTIC,
|
||||||
|
native_unit_of_measurement=PERCENTAGE,
|
||||||
|
char=Battery.battery_level,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
async def async_setup_entry(
|
||||||
|
hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
||||||
|
) -> None:
|
||||||
|
"""Set up Gardena Bluetooth sensor based on a config entry."""
|
||||||
|
coordinator: Coordinator = hass.data[DOMAIN][entry.entry_id]
|
||||||
|
entities: list[GardenaBluetoothEntity] = [
|
||||||
|
GardenaBluetoothSensor(coordinator, description)
|
||||||
|
for description in DESCRIPTIONS
|
||||||
|
if description.key in coordinator.characteristics
|
||||||
|
]
|
||||||
|
entities.append(GardenaBluetoothRemainSensor(coordinator))
|
||||||
|
async_add_entities(entities)
|
||||||
|
|
||||||
|
|
||||||
|
class GardenaBluetoothSensor(GardenaBluetoothEntity, SensorEntity):
|
||||||
|
"""Representation of a sensor."""
|
||||||
|
|
||||||
|
entity_description: GardenaBluetoothSensorEntityDescription
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
coordinator: Coordinator,
|
||||||
|
description: GardenaBluetoothSensorEntityDescription,
|
||||||
|
) -> None:
|
||||||
|
"""Initialize the sensor."""
|
||||||
|
super().__init__(coordinator, {description.key})
|
||||||
|
self._attr_native_value = None
|
||||||
|
self._attr_unique_id = f"{coordinator.address}-{description.key}"
|
||||||
|
self.entity_description = description
|
||||||
|
|
||||||
|
def _handle_coordinator_update(self) -> None:
|
||||||
|
value = self.coordinator.get_cached(self.entity_description.char)
|
||||||
|
if isinstance(value, datetime):
|
||||||
|
value = value.replace(
|
||||||
|
tzinfo=dt_util.get_time_zone(self.hass.config.time_zone)
|
||||||
|
)
|
||||||
|
self._attr_native_value = value
|
||||||
|
super()._handle_coordinator_update()
|
||||||
|
|
||||||
|
|
||||||
|
class GardenaBluetoothRemainSensor(GardenaBluetoothEntity, SensorEntity):
|
||||||
|
"""Representation of a sensor."""
|
||||||
|
|
||||||
|
_attr_device_class = SensorDeviceClass.TIMESTAMP
|
||||||
|
_attr_native_value: datetime | None = None
|
||||||
|
_attr_translation_key = "remaining_open_timestamp"
|
||||||
|
|
||||||
|
def __init__(
|
||||||
|
self,
|
||||||
|
coordinator: Coordinator,
|
||||||
|
) -> None:
|
||||||
|
"""Initialize the sensor."""
|
||||||
|
super().__init__(coordinator, {Valve.remaining_open_time.uuid})
|
||||||
|
self._attr_unique_id = f"{coordinator.address}-remaining_open_timestamp"
|
||||||
|
|
||||||
|
def _handle_coordinator_update(self) -> None:
|
||||||
|
value = self.coordinator.get_cached(Valve.remaining_open_time)
|
||||||
|
if not value:
|
||||||
|
self._attr_native_value = None
|
||||||
|
super()._handle_coordinator_update()
|
||||||
|
return
|
||||||
|
|
||||||
|
time = datetime.now(timezone.utc) + timedelta(seconds=value)
|
||||||
|
if not self._attr_native_value:
|
||||||
|
self._attr_native_value = time
|
||||||
|
super()._handle_coordinator_update()
|
||||||
|
return
|
||||||
|
|
||||||
|
error = time - self._attr_native_value
|
||||||
|
if abs(error.total_seconds()) > 10:
|
||||||
|
self._attr_native_value = time
|
||||||
|
super()._handle_coordinator_update()
|
||||||
|
return
|
@ -36,6 +36,14 @@
|
|||||||
"name": "Season pause"
|
"name": "Season pause"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"sensor": {
|
||||||
|
"activation_reason": {
|
||||||
|
"name": "Activation reason"
|
||||||
|
},
|
||||||
|
"remaining_open_timestamp": {
|
||||||
|
"name": "Valve closing"
|
||||||
|
}
|
||||||
|
},
|
||||||
"switch": {
|
"switch": {
|
||||||
"state": {
|
"state": {
|
||||||
"name": "[%key:common::state::open%]"
|
"name": "[%key:common::state::open%]"
|
||||||
|
Loading…
x
Reference in New Issue
Block a user