core/homeassistant/components/lamarzocco/sensor.py

172 lines
5.4 KiB
Python

"""Sensor platform for La Marzocco espresso machines."""
from collections.abc import Callable
from dataclasses import dataclass
from datetime import datetime
from typing import cast
from pylamarzocco.const import ModelName, WidgetType
from pylamarzocco.models import (
BackFlush,
BaseWidgetOutput,
CoffeeAndFlushCounter,
CoffeeBoiler,
SteamBoilerLevel,
SteamBoilerTemperature,
)
from homeassistant.components.sensor import (
SensorDeviceClass,
SensorEntity,
SensorEntityDescription,
SensorStateClass,
)
from homeassistant.const import EntityCategory
from homeassistant.core import HomeAssistant
from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
from homeassistant.helpers.typing import StateType
from .coordinator import LaMarzoccoConfigEntry
from .entity import LaMarzoccoEntity, LaMarzoccoEntityDescription
# Coordinator is used to centralize the data updates
PARALLEL_UPDATES = 0
@dataclass(frozen=True, kw_only=True)
class LaMarzoccoSensorEntityDescription(
LaMarzoccoEntityDescription,
SensorEntityDescription,
):
"""Description of a La Marzocco sensor."""
value_fn: Callable[
[dict[WidgetType, BaseWidgetOutput]], StateType | datetime | None
]
ENTITIES: tuple[LaMarzoccoSensorEntityDescription, ...] = (
LaMarzoccoSensorEntityDescription(
key="coffee_boiler_ready_time",
translation_key="coffee_boiler_ready_time",
device_class=SensorDeviceClass.TIMESTAMP,
value_fn=(
lambda config: cast(
CoffeeBoiler, config[WidgetType.CM_COFFEE_BOILER]
).ready_start_time
),
entity_category=EntityCategory.DIAGNOSTIC,
),
LaMarzoccoSensorEntityDescription(
key="steam_boiler_ready_time",
translation_key="steam_boiler_ready_time",
device_class=SensorDeviceClass.TIMESTAMP,
value_fn=(
lambda config: cast(
SteamBoilerLevel, config[WidgetType.CM_STEAM_BOILER_LEVEL]
).ready_start_time
),
entity_category=EntityCategory.DIAGNOSTIC,
supported_fn=(
lambda coordinator: coordinator.device.dashboard.model_name
in (ModelName.LINEA_MICRA, ModelName.LINEA_MINI_R)
),
),
LaMarzoccoSensorEntityDescription(
key="steam_boiler_ready_time",
translation_key="steam_boiler_ready_time",
device_class=SensorDeviceClass.TIMESTAMP,
value_fn=(
lambda config: cast(
SteamBoilerTemperature, config[WidgetType.CM_STEAM_BOILER_TEMPERATURE]
).ready_start_time
),
entity_category=EntityCategory.DIAGNOSTIC,
supported_fn=(
lambda coordinator: coordinator.device.dashboard.model_name
in (ModelName.GS3_AV, ModelName.GS3_MP, ModelName.LINEA_MINI)
),
),
LaMarzoccoSensorEntityDescription(
key="last_cleaning_time",
translation_key="last_cleaning_time",
device_class=SensorDeviceClass.TIMESTAMP,
value_fn=(
lambda config: cast(
BackFlush, config[WidgetType.CM_BACK_FLUSH]
).last_cleaning_start_time
),
entity_category=EntityCategory.DIAGNOSTIC,
),
)
STATISTIC_ENTITIES: tuple[LaMarzoccoSensorEntityDescription, ...] = (
LaMarzoccoSensorEntityDescription(
key="drink_stats_coffee",
translation_key="total_coffees_made",
state_class=SensorStateClass.TOTAL_INCREASING,
value_fn=(
lambda statistics: cast(
CoffeeAndFlushCounter, statistics[WidgetType.COFFEE_AND_FLUSH_COUNTER]
).total_coffee
),
entity_category=EntityCategory.DIAGNOSTIC,
),
LaMarzoccoSensorEntityDescription(
key="drink_stats_flushing",
translation_key="total_flushes_done",
state_class=SensorStateClass.TOTAL_INCREASING,
value_fn=(
lambda statistics: cast(
CoffeeAndFlushCounter, statistics[WidgetType.COFFEE_AND_FLUSH_COUNTER]
).total_flush
),
entity_category=EntityCategory.DIAGNOSTIC,
),
)
async def async_setup_entry(
hass: HomeAssistant,
entry: LaMarzoccoConfigEntry,
async_add_entities: AddConfigEntryEntitiesCallback,
) -> None:
"""Set up sensor entities."""
coordinator = entry.runtime_data.config_coordinator
entities = [
LaMarzoccoSensorEntity(coordinator, description)
for description in ENTITIES
if description.supported_fn(coordinator)
]
entities.extend(
LaMarzoccoStatisticSensorEntity(coordinator, description)
for description in STATISTIC_ENTITIES
if description.supported_fn(coordinator)
)
async_add_entities(entities)
class LaMarzoccoSensorEntity(LaMarzoccoEntity, SensorEntity):
"""Sensor for La Marzocco."""
entity_description: LaMarzoccoSensorEntityDescription
@property
def native_value(self) -> StateType | datetime | None:
"""Return value of the sensor."""
return self.entity_description.value_fn(
self.coordinator.device.dashboard.config
)
class LaMarzoccoStatisticSensorEntity(LaMarzoccoSensorEntity):
"""Sensor for La Marzocco statistics."""
@property
def native_value(self) -> StateType | datetime | None:
"""Return the value of the sensor."""
return self.entity_description.value_fn(
self.coordinator.device.statistics.widgets
)