mirror of https://github.com/home-assistant/core
197 lines
6.0 KiB
Python
197 lines
6.0 KiB
Python
"""Support for Tractive sensors."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from collections.abc import Callable
|
|
from dataclasses import dataclass
|
|
from typing import Any
|
|
|
|
from homeassistant.components.sensor import (
|
|
SensorDeviceClass,
|
|
SensorEntity,
|
|
SensorEntityDescription,
|
|
SensorStateClass,
|
|
)
|
|
from homeassistant.const import (
|
|
ATTR_BATTERY_LEVEL,
|
|
PERCENTAGE,
|
|
EntityCategory,
|
|
UnitOfTime,
|
|
)
|
|
from homeassistant.core import HomeAssistant, callback
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
from homeassistant.helpers.typing import StateType
|
|
|
|
from . import Trackables, TractiveClient, TractiveConfigEntry
|
|
from .const import (
|
|
ATTR_ACTIVITY_LABEL,
|
|
ATTR_CALORIES,
|
|
ATTR_DAILY_GOAL,
|
|
ATTR_MINUTES_ACTIVE,
|
|
ATTR_MINUTES_DAY_SLEEP,
|
|
ATTR_MINUTES_NIGHT_SLEEP,
|
|
ATTR_MINUTES_REST,
|
|
ATTR_SLEEP_LABEL,
|
|
ATTR_TRACKER_STATE,
|
|
TRACKER_HARDWARE_STATUS_UPDATED,
|
|
TRACKER_WELLNESS_STATUS_UPDATED,
|
|
)
|
|
from .entity import TractiveEntity
|
|
|
|
|
|
@dataclass(frozen=True, kw_only=True)
|
|
class TractiveSensorEntityDescription(SensorEntityDescription):
|
|
"""Class describing Tractive sensor entities."""
|
|
|
|
signal_prefix: str
|
|
|
|
hardware_sensor: bool = False
|
|
value_fn: Callable[[StateType], StateType] = lambda state: state
|
|
|
|
|
|
class TractiveSensor(TractiveEntity, SensorEntity):
|
|
"""Tractive sensor."""
|
|
|
|
entity_description: TractiveSensorEntityDescription
|
|
|
|
def __init__(
|
|
self,
|
|
client: TractiveClient,
|
|
item: Trackables,
|
|
description: TractiveSensorEntityDescription,
|
|
) -> None:
|
|
"""Initialize sensor entity."""
|
|
if description.hardware_sensor:
|
|
dispatcher_signal = (
|
|
f"{description.signal_prefix}-{item.tracker_details['_id']}"
|
|
)
|
|
else:
|
|
dispatcher_signal = f"{description.signal_prefix}-{item.trackable['_id']}"
|
|
super().__init__(
|
|
client, item.trackable, item.tracker_details, dispatcher_signal
|
|
)
|
|
|
|
self._attr_unique_id = f"{item.trackable['_id']}_{description.key}"
|
|
self._attr_available = False
|
|
self.entity_description = description
|
|
|
|
@callback
|
|
def handle_status_update(self, event: dict[str, Any]) -> None:
|
|
"""Handle status update."""
|
|
self._attr_native_value = self.entity_description.value_fn(
|
|
event[self.entity_description.key]
|
|
)
|
|
|
|
super().handle_status_update(event)
|
|
|
|
|
|
SENSOR_TYPES: tuple[TractiveSensorEntityDescription, ...] = (
|
|
TractiveSensorEntityDescription(
|
|
key=ATTR_BATTERY_LEVEL,
|
|
translation_key="tracker_battery_level",
|
|
native_unit_of_measurement=PERCENTAGE,
|
|
device_class=SensorDeviceClass.BATTERY,
|
|
signal_prefix=TRACKER_HARDWARE_STATUS_UPDATED,
|
|
hardware_sensor=True,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
),
|
|
TractiveSensorEntityDescription(
|
|
key=ATTR_TRACKER_STATE,
|
|
translation_key="tracker_state",
|
|
signal_prefix=TRACKER_HARDWARE_STATUS_UPDATED,
|
|
hardware_sensor=True,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
device_class=SensorDeviceClass.ENUM,
|
|
options=[
|
|
"inaccurate_position",
|
|
"not_reporting",
|
|
"operational",
|
|
"system_shutdown_user",
|
|
"system_startup",
|
|
],
|
|
),
|
|
TractiveSensorEntityDescription(
|
|
key=ATTR_MINUTES_ACTIVE,
|
|
translation_key="activity_time",
|
|
native_unit_of_measurement=UnitOfTime.MINUTES,
|
|
signal_prefix=TRACKER_WELLNESS_STATUS_UPDATED,
|
|
state_class=SensorStateClass.TOTAL,
|
|
),
|
|
TractiveSensorEntityDescription(
|
|
key=ATTR_MINUTES_REST,
|
|
translation_key="rest_time",
|
|
native_unit_of_measurement=UnitOfTime.MINUTES,
|
|
signal_prefix=TRACKER_WELLNESS_STATUS_UPDATED,
|
|
state_class=SensorStateClass.TOTAL,
|
|
),
|
|
TractiveSensorEntityDescription(
|
|
key=ATTR_CALORIES,
|
|
translation_key="calories",
|
|
native_unit_of_measurement="kcal",
|
|
signal_prefix=TRACKER_WELLNESS_STATUS_UPDATED,
|
|
state_class=SensorStateClass.TOTAL,
|
|
),
|
|
TractiveSensorEntityDescription(
|
|
key=ATTR_DAILY_GOAL,
|
|
translation_key="daily_goal",
|
|
native_unit_of_measurement=UnitOfTime.MINUTES,
|
|
signal_prefix=TRACKER_WELLNESS_STATUS_UPDATED,
|
|
),
|
|
TractiveSensorEntityDescription(
|
|
key=ATTR_MINUTES_DAY_SLEEP,
|
|
translation_key="minutes_day_sleep",
|
|
native_unit_of_measurement=UnitOfTime.MINUTES,
|
|
signal_prefix=TRACKER_WELLNESS_STATUS_UPDATED,
|
|
state_class=SensorStateClass.TOTAL,
|
|
),
|
|
TractiveSensorEntityDescription(
|
|
key=ATTR_MINUTES_NIGHT_SLEEP,
|
|
translation_key="minutes_night_sleep",
|
|
native_unit_of_measurement=UnitOfTime.MINUTES,
|
|
signal_prefix=TRACKER_WELLNESS_STATUS_UPDATED,
|
|
state_class=SensorStateClass.TOTAL,
|
|
),
|
|
TractiveSensorEntityDescription(
|
|
key=ATTR_SLEEP_LABEL,
|
|
translation_key="sleep",
|
|
signal_prefix=TRACKER_WELLNESS_STATUS_UPDATED,
|
|
value_fn=lambda state: state.lower() if isinstance(state, str) else state,
|
|
device_class=SensorDeviceClass.ENUM,
|
|
options=[
|
|
"good",
|
|
"low",
|
|
"ok",
|
|
],
|
|
),
|
|
TractiveSensorEntityDescription(
|
|
key=ATTR_ACTIVITY_LABEL,
|
|
translation_key="activity",
|
|
signal_prefix=TRACKER_WELLNESS_STATUS_UPDATED,
|
|
value_fn=lambda state: state.lower() if isinstance(state, str) else state,
|
|
device_class=SensorDeviceClass.ENUM,
|
|
options=[
|
|
"good",
|
|
"low",
|
|
"ok",
|
|
],
|
|
),
|
|
)
|
|
|
|
|
|
async def async_setup_entry(
|
|
hass: HomeAssistant,
|
|
entry: TractiveConfigEntry,
|
|
async_add_entities: AddEntitiesCallback,
|
|
) -> None:
|
|
"""Set up Tractive device trackers."""
|
|
client = entry.runtime_data.client
|
|
trackables = entry.runtime_data.trackables
|
|
|
|
entities = [
|
|
TractiveSensor(client, item, description)
|
|
for description in SENSOR_TYPES
|
|
for item in trackables
|
|
]
|
|
|
|
async_add_entities(entities)
|