372 lines
13 KiB
Python
372 lines
13 KiB
Python
"""Support for Renault sensors."""
|
|
from __future__ import annotations
|
|
|
|
from collections.abc import Callable
|
|
from dataclasses import dataclass
|
|
from datetime import datetime
|
|
from typing import TYPE_CHECKING, Any, Generic, cast
|
|
|
|
from renault_api.kamereon.enums import ChargeState, PlugState
|
|
from renault_api.kamereon.models import (
|
|
KamereonVehicleBatteryStatusData,
|
|
KamereonVehicleCockpitData,
|
|
KamereonVehicleHvacStatusData,
|
|
KamereonVehicleLocationData,
|
|
KamereonVehicleResStateData,
|
|
)
|
|
|
|
from homeassistant.components.sensor import (
|
|
SensorDeviceClass,
|
|
SensorEntity,
|
|
SensorEntityDescription,
|
|
SensorStateClass,
|
|
)
|
|
from homeassistant.config_entries import ConfigEntry
|
|
from homeassistant.const import (
|
|
PERCENTAGE,
|
|
UnitOfEnergy,
|
|
UnitOfLength,
|
|
UnitOfPower,
|
|
UnitOfTemperature,
|
|
UnitOfTime,
|
|
UnitOfVolume,
|
|
)
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
from homeassistant.helpers.typing import StateType
|
|
from homeassistant.util.dt import as_utc, parse_datetime
|
|
|
|
from .const import DOMAIN
|
|
from .coordinator import T
|
|
from .entity import RenaultDataEntity, RenaultDataEntityDescription
|
|
from .renault_hub import RenaultHub
|
|
from .renault_vehicle import RenaultVehicleProxy
|
|
|
|
|
|
@dataclass
|
|
class RenaultSensorRequiredKeysMixin(Generic[T]):
|
|
"""Mixin for required keys."""
|
|
|
|
data_key: str
|
|
entity_class: type[RenaultSensor[T]]
|
|
|
|
|
|
@dataclass
|
|
class RenaultSensorEntityDescription(
|
|
SensorEntityDescription,
|
|
RenaultDataEntityDescription,
|
|
RenaultSensorRequiredKeysMixin[T],
|
|
):
|
|
"""Class describing Renault sensor entities."""
|
|
|
|
icon_lambda: Callable[[RenaultSensor[T]], str] | None = None
|
|
condition_lambda: Callable[[RenaultVehicleProxy], bool] | None = None
|
|
requires_fuel: bool = False
|
|
value_lambda: Callable[[RenaultSensor[T]], StateType | datetime] | None = None
|
|
|
|
|
|
async def async_setup_entry(
|
|
hass: HomeAssistant,
|
|
config_entry: ConfigEntry,
|
|
async_add_entities: AddEntitiesCallback,
|
|
) -> None:
|
|
"""Set up the Renault entities from config entry."""
|
|
proxy: RenaultHub = hass.data[DOMAIN][config_entry.entry_id]
|
|
entities: list[RenaultSensor[Any]] = [
|
|
description.entity_class(vehicle, description)
|
|
for vehicle in proxy.vehicles.values()
|
|
for description in SENSOR_TYPES
|
|
if description.coordinator in vehicle.coordinators
|
|
and (not description.requires_fuel or vehicle.details.uses_fuel())
|
|
and (not description.condition_lambda or description.condition_lambda(vehicle))
|
|
]
|
|
async_add_entities(entities)
|
|
|
|
|
|
class RenaultSensor(RenaultDataEntity[T], SensorEntity):
|
|
"""Mixin for sensor specific attributes."""
|
|
|
|
entity_description: RenaultSensorEntityDescription[T]
|
|
|
|
@property
|
|
def data(self) -> StateType:
|
|
"""Return the state of this entity."""
|
|
return self._get_data_attr(self.entity_description.data_key)
|
|
|
|
@property
|
|
def icon(self) -> str | None:
|
|
"""Icon handling."""
|
|
if self.entity_description.icon_lambda is None:
|
|
return super().icon
|
|
return self.entity_description.icon_lambda(self)
|
|
|
|
@property
|
|
def native_value(self) -> StateType | datetime:
|
|
"""Return the state of this entity."""
|
|
if self.data is None:
|
|
return None
|
|
if self.entity_description.value_lambda is None:
|
|
return self.data
|
|
return self.entity_description.value_lambda(self)
|
|
|
|
|
|
def _get_charging_power(entity: RenaultSensor[T]) -> StateType:
|
|
"""Return the charging_power of this entity."""
|
|
return cast(float, entity.data) / 1000
|
|
|
|
|
|
def _get_charge_state_formatted(entity: RenaultSensor[T]) -> str | None:
|
|
"""Return the charging_status of this entity."""
|
|
data = cast(KamereonVehicleBatteryStatusData, entity.coordinator.data)
|
|
charging_status = data.get_charging_status() if data else None
|
|
return charging_status.name.lower() if charging_status else None
|
|
|
|
|
|
def _get_charge_state_icon(entity: RenaultSensor[T]) -> str:
|
|
"""Return the icon of this entity."""
|
|
if entity.data == ChargeState.CHARGE_IN_PROGRESS.value:
|
|
return "mdi:flash"
|
|
return "mdi:flash-off"
|
|
|
|
|
|
def _get_plug_state_formatted(entity: RenaultSensor[T]) -> str | None:
|
|
"""Return the plug_status of this entity."""
|
|
data = cast(KamereonVehicleBatteryStatusData, entity.coordinator.data)
|
|
plug_status = data.get_plug_status() if data else None
|
|
return plug_status.name.lower() if plug_status else None
|
|
|
|
|
|
def _get_plug_state_icon(entity: RenaultSensor[T]) -> str:
|
|
"""Return the icon of this entity."""
|
|
if entity.data == PlugState.PLUGGED.value:
|
|
return "mdi:power-plug"
|
|
return "mdi:power-plug-off"
|
|
|
|
|
|
def _get_rounded_value(entity: RenaultSensor[T]) -> float:
|
|
"""Return the icon of this entity."""
|
|
return round(cast(float, entity.data))
|
|
|
|
|
|
def _get_utc_value(entity: RenaultSensor[T]) -> datetime:
|
|
"""Return the UTC value of this entity."""
|
|
original_dt = parse_datetime(cast(str, entity.data))
|
|
if TYPE_CHECKING:
|
|
assert original_dt is not None
|
|
return as_utc(original_dt)
|
|
|
|
|
|
SENSOR_TYPES: tuple[RenaultSensorEntityDescription[Any], ...] = (
|
|
RenaultSensorEntityDescription(
|
|
key="battery_level",
|
|
coordinator="battery",
|
|
data_key="batteryLevel",
|
|
device_class=SensorDeviceClass.BATTERY,
|
|
entity_class=RenaultSensor[KamereonVehicleBatteryStatusData],
|
|
native_unit_of_measurement=PERCENTAGE,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
),
|
|
RenaultSensorEntityDescription(
|
|
key="charge_state",
|
|
coordinator="battery",
|
|
data_key="chargingStatus",
|
|
translation_key="charge_state",
|
|
device_class=SensorDeviceClass.ENUM,
|
|
entity_class=RenaultSensor[KamereonVehicleBatteryStatusData],
|
|
icon_lambda=_get_charge_state_icon,
|
|
options=[
|
|
"not_in_charge",
|
|
"waiting_for_a_planned_charge",
|
|
"charge_ended",
|
|
"waiting_for_current_charge",
|
|
"energy_flap_opened",
|
|
"charge_in_progress",
|
|
"charge_error",
|
|
"unavailable",
|
|
],
|
|
value_lambda=_get_charge_state_formatted,
|
|
),
|
|
RenaultSensorEntityDescription(
|
|
key="charging_remaining_time",
|
|
coordinator="battery",
|
|
data_key="chargingRemainingTime",
|
|
device_class=SensorDeviceClass.DURATION,
|
|
entity_class=RenaultSensor[KamereonVehicleBatteryStatusData],
|
|
icon="mdi:timer",
|
|
native_unit_of_measurement=UnitOfTime.MINUTES,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
translation_key="charging_remaining_time",
|
|
),
|
|
RenaultSensorEntityDescription(
|
|
# For vehicles that DO NOT report charging power in watts, this seems to
|
|
# correspond to the maximum power that would be admissible by the car based
|
|
# on the battery state, regardless of the type of charger.
|
|
key="charging_power",
|
|
condition_lambda=lambda a: not a.details.reports_charging_power_in_watts(),
|
|
coordinator="battery",
|
|
data_key="chargingInstantaneousPower",
|
|
device_class=SensorDeviceClass.POWER,
|
|
entity_class=RenaultSensor[KamereonVehicleBatteryStatusData],
|
|
native_unit_of_measurement=UnitOfPower.KILO_WATT,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
translation_key="admissible_charging_power",
|
|
),
|
|
RenaultSensorEntityDescription(
|
|
# For vehicles that DO report charging power in watts, this is the power
|
|
# effectively being transferred to the car.
|
|
key="charging_power",
|
|
condition_lambda=lambda a: a.details.reports_charging_power_in_watts(),
|
|
coordinator="battery",
|
|
data_key="chargingInstantaneousPower",
|
|
device_class=SensorDeviceClass.POWER,
|
|
entity_class=RenaultSensor[KamereonVehicleBatteryStatusData],
|
|
native_unit_of_measurement=UnitOfPower.KILO_WATT,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
value_lambda=_get_charging_power,
|
|
translation_key="charging_power",
|
|
),
|
|
RenaultSensorEntityDescription(
|
|
key="plug_state",
|
|
coordinator="battery",
|
|
data_key="plugStatus",
|
|
translation_key="plug_state",
|
|
device_class=SensorDeviceClass.ENUM,
|
|
entity_class=RenaultSensor[KamereonVehicleBatteryStatusData],
|
|
icon_lambda=_get_plug_state_icon,
|
|
options=["unplugged", "plugged", "plug_error", "plug_unknown"],
|
|
value_lambda=_get_plug_state_formatted,
|
|
),
|
|
RenaultSensorEntityDescription(
|
|
key="battery_autonomy",
|
|
coordinator="battery",
|
|
data_key="batteryAutonomy",
|
|
device_class=SensorDeviceClass.DISTANCE,
|
|
entity_class=RenaultSensor[KamereonVehicleBatteryStatusData],
|
|
icon="mdi:ev-station",
|
|
native_unit_of_measurement=UnitOfLength.KILOMETERS,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
translation_key="battery_autonomy",
|
|
),
|
|
RenaultSensorEntityDescription(
|
|
key="battery_available_energy",
|
|
coordinator="battery",
|
|
data_key="batteryAvailableEnergy",
|
|
entity_class=RenaultSensor[KamereonVehicleBatteryStatusData],
|
|
device_class=SensorDeviceClass.ENERGY,
|
|
native_unit_of_measurement=UnitOfEnergy.KILO_WATT_HOUR,
|
|
state_class=SensorStateClass.TOTAL,
|
|
translation_key="battery_available_energy",
|
|
),
|
|
RenaultSensorEntityDescription(
|
|
key="battery_temperature",
|
|
coordinator="battery",
|
|
data_key="batteryTemperature",
|
|
device_class=SensorDeviceClass.TEMPERATURE,
|
|
entity_class=RenaultSensor[KamereonVehicleBatteryStatusData],
|
|
native_unit_of_measurement=UnitOfTemperature.CELSIUS,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
translation_key="battery_temperature",
|
|
),
|
|
RenaultSensorEntityDescription(
|
|
key="battery_last_activity",
|
|
coordinator="battery",
|
|
device_class=SensorDeviceClass.TIMESTAMP,
|
|
data_key="timestamp",
|
|
entity_class=RenaultSensor[KamereonVehicleBatteryStatusData],
|
|
entity_registry_enabled_default=False,
|
|
value_lambda=_get_utc_value,
|
|
translation_key="battery_last_activity",
|
|
),
|
|
RenaultSensorEntityDescription(
|
|
key="mileage",
|
|
coordinator="cockpit",
|
|
data_key="totalMileage",
|
|
device_class=SensorDeviceClass.DISTANCE,
|
|
entity_class=RenaultSensor[KamereonVehicleCockpitData],
|
|
icon="mdi:sign-direction",
|
|
native_unit_of_measurement=UnitOfLength.KILOMETERS,
|
|
state_class=SensorStateClass.TOTAL_INCREASING,
|
|
value_lambda=_get_rounded_value,
|
|
translation_key="mileage",
|
|
),
|
|
RenaultSensorEntityDescription(
|
|
key="fuel_autonomy",
|
|
coordinator="cockpit",
|
|
data_key="fuelAutonomy",
|
|
device_class=SensorDeviceClass.DISTANCE,
|
|
entity_class=RenaultSensor[KamereonVehicleCockpitData],
|
|
icon="mdi:gas-station",
|
|
native_unit_of_measurement=UnitOfLength.KILOMETERS,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
requires_fuel=True,
|
|
value_lambda=_get_rounded_value,
|
|
translation_key="fuel_autonomy",
|
|
),
|
|
RenaultSensorEntityDescription(
|
|
key="fuel_quantity",
|
|
coordinator="cockpit",
|
|
data_key="fuelQuantity",
|
|
device_class=SensorDeviceClass.VOLUME,
|
|
entity_class=RenaultSensor[KamereonVehicleCockpitData],
|
|
icon="mdi:fuel",
|
|
native_unit_of_measurement=UnitOfVolume.LITERS,
|
|
state_class=SensorStateClass.TOTAL,
|
|
requires_fuel=True,
|
|
value_lambda=_get_rounded_value,
|
|
translation_key="fuel_quantity",
|
|
),
|
|
RenaultSensorEntityDescription(
|
|
key="outside_temperature",
|
|
coordinator="hvac_status",
|
|
device_class=SensorDeviceClass.TEMPERATURE,
|
|
data_key="externalTemperature",
|
|
entity_class=RenaultSensor[KamereonVehicleHvacStatusData],
|
|
native_unit_of_measurement=UnitOfTemperature.CELSIUS,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
translation_key="outside_temperature",
|
|
),
|
|
RenaultSensorEntityDescription(
|
|
key="hvac_soc_threshold",
|
|
coordinator="hvac_status",
|
|
data_key="socThreshold",
|
|
entity_class=RenaultSensor[KamereonVehicleHvacStatusData],
|
|
native_unit_of_measurement=PERCENTAGE,
|
|
translation_key="hvac_soc_threshold",
|
|
),
|
|
RenaultSensorEntityDescription(
|
|
key="hvac_last_activity",
|
|
coordinator="hvac_status",
|
|
device_class=SensorDeviceClass.TIMESTAMP,
|
|
data_key="lastUpdateTime",
|
|
entity_class=RenaultSensor[KamereonVehicleHvacStatusData],
|
|
entity_registry_enabled_default=False,
|
|
translation_key="hvac_last_activity",
|
|
value_lambda=_get_utc_value,
|
|
),
|
|
RenaultSensorEntityDescription(
|
|
key="location_last_activity",
|
|
coordinator="location",
|
|
device_class=SensorDeviceClass.TIMESTAMP,
|
|
data_key="lastUpdateTime",
|
|
entity_class=RenaultSensor[KamereonVehicleLocationData],
|
|
entity_registry_enabled_default=False,
|
|
translation_key="location_last_activity",
|
|
value_lambda=_get_utc_value,
|
|
),
|
|
RenaultSensorEntityDescription(
|
|
key="res_state",
|
|
coordinator="res_state",
|
|
data_key="details",
|
|
entity_class=RenaultSensor[KamereonVehicleResStateData],
|
|
translation_key="res_state",
|
|
),
|
|
RenaultSensorEntityDescription(
|
|
key="res_state_code",
|
|
coordinator="res_state",
|
|
data_key="code",
|
|
entity_class=RenaultSensor[KamereonVehicleResStateData],
|
|
entity_registry_enabled_default=False,
|
|
translation_key="res_state_code",
|
|
),
|
|
)
|