786 lines
28 KiB
Python
786 lines
28 KiB
Python
"""Sensor platform for Teslemetry integration."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from collections.abc import Callable
|
|
from dataclasses import dataclass
|
|
from datetime import timedelta
|
|
|
|
from propcache import cached_property
|
|
from teslemetry_stream import Signal
|
|
|
|
from homeassistant.components.sensor import (
|
|
RestoreSensor,
|
|
SensorDeviceClass,
|
|
SensorEntity,
|
|
SensorEntityDescription,
|
|
SensorStateClass,
|
|
)
|
|
from homeassistant.const import (
|
|
PERCENTAGE,
|
|
EntityCategory,
|
|
UnitOfElectricCurrent,
|
|
UnitOfElectricPotential,
|
|
UnitOfEnergy,
|
|
UnitOfLength,
|
|
UnitOfPower,
|
|
UnitOfPressure,
|
|
UnitOfSpeed,
|
|
UnitOfTemperature,
|
|
UnitOfTime,
|
|
)
|
|
from homeassistant.core import HomeAssistant
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
|
from homeassistant.helpers.typing import StateType
|
|
from homeassistant.util import dt as dt_util
|
|
from homeassistant.util.variance import ignore_variance
|
|
|
|
from . import TeslemetryConfigEntry
|
|
from .const import ENERGY_HISTORY_FIELDS
|
|
from .entity import (
|
|
TeslemetryEnergyHistoryEntity,
|
|
TeslemetryEnergyInfoEntity,
|
|
TeslemetryEnergyLiveEntity,
|
|
TeslemetryVehicleEntity,
|
|
TeslemetryVehicleStreamEntity,
|
|
TeslemetryWallConnectorEntity,
|
|
)
|
|
from .models import TeslemetryEnergyData, TeslemetryVehicleData
|
|
|
|
PARALLEL_UPDATES = 0
|
|
|
|
CHARGE_STATES = {
|
|
"Starting": "starting",
|
|
"Charging": "charging",
|
|
"Stopped": "stopped",
|
|
"Complete": "complete",
|
|
"Disconnected": "disconnected",
|
|
"NoPower": "no_power",
|
|
}
|
|
|
|
SHIFT_STATES = {"P": "p", "D": "d", "R": "r", "N": "n"}
|
|
|
|
|
|
@dataclass(frozen=True, kw_only=True)
|
|
class TeslemetryVehicleSensorEntityDescription(SensorEntityDescription):
|
|
"""Describes Teslemetry Sensor entity."""
|
|
|
|
polling: bool = False
|
|
polling_value_fn: Callable[[StateType], StateType] = lambda x: x
|
|
polling_available_fn: Callable[[StateType], bool] = lambda x: x is not None
|
|
streaming_key: Signal | None = None
|
|
streaming_value_fn: Callable[[StateType], StateType] = lambda x: x
|
|
streaming_firmware: str = "2024.26"
|
|
|
|
|
|
VEHICLE_DESCRIPTIONS: tuple[TeslemetryVehicleSensorEntityDescription, ...] = (
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="charge_state_charging_state",
|
|
polling=True,
|
|
streaming_key=Signal.DETAILED_CHARGE_STATE,
|
|
polling_value_fn=lambda value: CHARGE_STATES.get(str(value)),
|
|
streaming_value_fn=lambda value: CHARGE_STATES.get(
|
|
str(value).replace("DetailedChargeState", "")
|
|
),
|
|
options=list(CHARGE_STATES.values()),
|
|
device_class=SensorDeviceClass.ENUM,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="charge_state_battery_level",
|
|
polling=True,
|
|
streaming_key=Signal.BATTERY_LEVEL,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=PERCENTAGE,
|
|
device_class=SensorDeviceClass.BATTERY,
|
|
suggested_display_precision=1,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="charge_state_usable_battery_level",
|
|
polling=True,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=PERCENTAGE,
|
|
device_class=SensorDeviceClass.BATTERY,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="charge_state_charge_energy_added",
|
|
polling=True,
|
|
streaming_key=Signal.AC_CHARGING_ENERGY_IN,
|
|
state_class=SensorStateClass.TOTAL_INCREASING,
|
|
native_unit_of_measurement=UnitOfEnergy.KILO_WATT_HOUR,
|
|
device_class=SensorDeviceClass.ENERGY,
|
|
suggested_display_precision=1,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="charge_state_charger_power",
|
|
polling=True,
|
|
streaming_key=Signal.AC_CHARGING_POWER,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfPower.KILO_WATT,
|
|
device_class=SensorDeviceClass.POWER,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="charge_state_charger_voltage",
|
|
polling=True,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfElectricPotential.VOLT,
|
|
device_class=SensorDeviceClass.VOLTAGE,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="charge_state_charger_actual_current",
|
|
polling=True,
|
|
streaming_key=Signal.CHARGE_AMPS,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfElectricCurrent.AMPERE,
|
|
device_class=SensorDeviceClass.CURRENT,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="charge_state_charge_rate",
|
|
polling=True,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfSpeed.MILES_PER_HOUR,
|
|
device_class=SensorDeviceClass.SPEED,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="charge_state_conn_charge_cable",
|
|
polling=True,
|
|
streaming_key=Signal.CHARGING_CABLE_TYPE,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="charge_state_fast_charger_type",
|
|
polling=True,
|
|
streaming_key=Signal.FAST_CHARGER_TYPE,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="charge_state_battery_range",
|
|
polling=True,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfLength.MILES,
|
|
device_class=SensorDeviceClass.DISTANCE,
|
|
suggested_display_precision=1,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="charge_state_est_battery_range",
|
|
polling=True,
|
|
streaming_key=Signal.EST_BATTERY_RANGE,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfLength.MILES,
|
|
device_class=SensorDeviceClass.DISTANCE,
|
|
suggested_display_precision=1,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="charge_state_ideal_battery_range",
|
|
polling=True,
|
|
streaming_key=Signal.IDEAL_BATTERY_RANGE,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfLength.MILES,
|
|
device_class=SensorDeviceClass.DISTANCE,
|
|
suggested_display_precision=1,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="drive_state_speed",
|
|
polling=True,
|
|
polling_value_fn=lambda value: value or 0,
|
|
streaming_key=Signal.VEHICLE_SPEED,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfSpeed.MILES_PER_HOUR,
|
|
device_class=SensorDeviceClass.SPEED,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="drive_state_power",
|
|
polling=True,
|
|
polling_value_fn=lambda value: value or 0,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfPower.KILO_WATT,
|
|
device_class=SensorDeviceClass.POWER,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="drive_state_shift_state",
|
|
polling=True,
|
|
polling_available_fn=lambda x: True,
|
|
polling_value_fn=lambda x: SHIFT_STATES.get(str(x), "p"),
|
|
streaming_key=Signal.GEAR,
|
|
streaming_value_fn=lambda x: SHIFT_STATES.get(str(x)),
|
|
options=list(SHIFT_STATES.values()),
|
|
device_class=SensorDeviceClass.ENUM,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="vehicle_state_odometer",
|
|
polling=True,
|
|
streaming_key=Signal.ODOMETER,
|
|
state_class=SensorStateClass.TOTAL_INCREASING,
|
|
native_unit_of_measurement=UnitOfLength.MILES,
|
|
device_class=SensorDeviceClass.DISTANCE,
|
|
suggested_display_precision=0,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="vehicle_state_tpms_pressure_fl",
|
|
polling=True,
|
|
streaming_key=Signal.TPMS_PRESSURE_FL,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfPressure.BAR,
|
|
suggested_unit_of_measurement=UnitOfPressure.PSI,
|
|
device_class=SensorDeviceClass.PRESSURE,
|
|
suggested_display_precision=1,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="vehicle_state_tpms_pressure_fr",
|
|
polling=True,
|
|
streaming_key=Signal.TPMS_PRESSURE_FR,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfPressure.BAR,
|
|
suggested_unit_of_measurement=UnitOfPressure.PSI,
|
|
device_class=SensorDeviceClass.PRESSURE,
|
|
suggested_display_precision=1,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="vehicle_state_tpms_pressure_rl",
|
|
polling=True,
|
|
streaming_key=Signal.TPMS_PRESSURE_RL,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfPressure.BAR,
|
|
suggested_unit_of_measurement=UnitOfPressure.PSI,
|
|
device_class=SensorDeviceClass.PRESSURE,
|
|
suggested_display_precision=1,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="vehicle_state_tpms_pressure_rr",
|
|
polling=True,
|
|
streaming_key=Signal.TPMS_PRESSURE_RR,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfPressure.BAR,
|
|
suggested_unit_of_measurement=UnitOfPressure.PSI,
|
|
device_class=SensorDeviceClass.PRESSURE,
|
|
suggested_display_precision=1,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="climate_state_inside_temp",
|
|
polling=True,
|
|
streaming_key=Signal.INSIDE_TEMP,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfTemperature.CELSIUS,
|
|
device_class=SensorDeviceClass.TEMPERATURE,
|
|
suggested_display_precision=1,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="climate_state_outside_temp",
|
|
polling=True,
|
|
streaming_key=Signal.OUTSIDE_TEMP,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfTemperature.CELSIUS,
|
|
device_class=SensorDeviceClass.TEMPERATURE,
|
|
suggested_display_precision=1,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="climate_state_driver_temp_setting",
|
|
polling=True,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfTemperature.CELSIUS,
|
|
device_class=SensorDeviceClass.TEMPERATURE,
|
|
suggested_display_precision=1,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="climate_state_passenger_temp_setting",
|
|
polling=True,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfTemperature.CELSIUS,
|
|
device_class=SensorDeviceClass.TEMPERATURE,
|
|
suggested_display_precision=1,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="drive_state_active_route_traffic_minutes_delay",
|
|
polling=True,
|
|
streaming_key=Signal.ROUTE_TRAFFIC_MINUTES_DELAY,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfTime.MINUTES,
|
|
device_class=SensorDeviceClass.DURATION,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="drive_state_active_route_energy_at_arrival",
|
|
polling=True,
|
|
streaming_key=Signal.EXPECTED_ENERGY_PERCENT_AT_TRIP_ARRIVAL,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=PERCENTAGE,
|
|
device_class=SensorDeviceClass.BATTERY,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetryVehicleSensorEntityDescription(
|
|
key="drive_state_active_route_miles_to_arrival",
|
|
polling=True,
|
|
streaming_key=Signal.MILES_TO_ARRIVAL,
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfLength.MILES,
|
|
device_class=SensorDeviceClass.DISTANCE,
|
|
),
|
|
)
|
|
|
|
|
|
@dataclass(frozen=True, kw_only=True)
|
|
class TeslemetryTimeEntityDescription(SensorEntityDescription):
|
|
"""Describes Teslemetry Sensor entity."""
|
|
|
|
variance: int
|
|
streaming_key: Signal
|
|
streaming_firmware: str = "2024.26"
|
|
|
|
|
|
VEHICLE_TIME_DESCRIPTIONS: tuple[TeslemetryTimeEntityDescription, ...] = (
|
|
TeslemetryTimeEntityDescription(
|
|
key="charge_state_minutes_to_full_charge",
|
|
streaming_key=Signal.TIME_TO_FULL_CHARGE,
|
|
device_class=SensorDeviceClass.TIMESTAMP,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
variance=4,
|
|
),
|
|
TeslemetryTimeEntityDescription(
|
|
key="drive_state_active_route_minutes_to_arrival",
|
|
streaming_key=Signal.MINUTES_TO_ARRIVAL,
|
|
device_class=SensorDeviceClass.TIMESTAMP,
|
|
variance=1,
|
|
),
|
|
)
|
|
|
|
ENERGY_LIVE_DESCRIPTIONS: tuple[SensorEntityDescription, ...] = (
|
|
SensorEntityDescription(
|
|
key="solar_power",
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfPower.WATT,
|
|
suggested_unit_of_measurement=UnitOfPower.KILO_WATT,
|
|
suggested_display_precision=2,
|
|
device_class=SensorDeviceClass.POWER,
|
|
),
|
|
SensorEntityDescription(
|
|
key="energy_left",
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
|
|
suggested_unit_of_measurement=UnitOfEnergy.KILO_WATT_HOUR,
|
|
suggested_display_precision=2,
|
|
device_class=SensorDeviceClass.ENERGY_STORAGE,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
),
|
|
SensorEntityDescription(
|
|
key="total_pack_energy",
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
|
|
suggested_unit_of_measurement=UnitOfEnergy.KILO_WATT_HOUR,
|
|
suggested_display_precision=2,
|
|
device_class=SensorDeviceClass.ENERGY_STORAGE,
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
SensorEntityDescription(
|
|
key="percentage_charged",
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=PERCENTAGE,
|
|
device_class=SensorDeviceClass.BATTERY,
|
|
suggested_display_precision=2,
|
|
),
|
|
SensorEntityDescription(
|
|
key="battery_power",
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfPower.WATT,
|
|
suggested_unit_of_measurement=UnitOfPower.KILO_WATT,
|
|
suggested_display_precision=2,
|
|
device_class=SensorDeviceClass.POWER,
|
|
),
|
|
SensorEntityDescription(
|
|
key="load_power",
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfPower.WATT,
|
|
suggested_unit_of_measurement=UnitOfPower.KILO_WATT,
|
|
suggested_display_precision=2,
|
|
device_class=SensorDeviceClass.POWER,
|
|
),
|
|
SensorEntityDescription(
|
|
key="grid_power",
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfPower.WATT,
|
|
suggested_unit_of_measurement=UnitOfPower.KILO_WATT,
|
|
suggested_display_precision=2,
|
|
device_class=SensorDeviceClass.POWER,
|
|
),
|
|
SensorEntityDescription(
|
|
key="grid_services_power",
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfPower.WATT,
|
|
suggested_unit_of_measurement=UnitOfPower.KILO_WATT,
|
|
suggested_display_precision=2,
|
|
device_class=SensorDeviceClass.POWER,
|
|
),
|
|
SensorEntityDescription(
|
|
key="generator_power",
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfPower.WATT,
|
|
suggested_unit_of_measurement=UnitOfPower.KILO_WATT,
|
|
suggested_display_precision=2,
|
|
device_class=SensorDeviceClass.POWER,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
SensorEntityDescription(
|
|
key="island_status",
|
|
device_class=SensorDeviceClass.ENUM,
|
|
options=[
|
|
"on_grid",
|
|
"off_grid",
|
|
"off_grid_intentional",
|
|
"off_grid_unintentional",
|
|
"island_status_unknown",
|
|
],
|
|
),
|
|
)
|
|
|
|
|
|
@dataclass(frozen=True, kw_only=True)
|
|
class TeslemetrySensorEntityDescription(SensorEntityDescription):
|
|
"""Describes Teslemetry Sensor entity."""
|
|
|
|
value_fn: Callable[[StateType], StateType] = lambda x: x
|
|
|
|
|
|
WALL_CONNECTOR_DESCRIPTIONS: tuple[TeslemetrySensorEntityDescription, ...] = (
|
|
TeslemetrySensorEntityDescription(
|
|
key="wall_connector_state",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetrySensorEntityDescription(
|
|
key="wall_connector_fault_state",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
entity_registry_enabled_default=False,
|
|
),
|
|
TeslemetrySensorEntityDescription(
|
|
key="wall_connector_power",
|
|
state_class=SensorStateClass.MEASUREMENT,
|
|
native_unit_of_measurement=UnitOfPower.WATT,
|
|
suggested_unit_of_measurement=UnitOfPower.KILO_WATT,
|
|
suggested_display_precision=2,
|
|
device_class=SensorDeviceClass.POWER,
|
|
),
|
|
TeslemetrySensorEntityDescription(
|
|
key="vin",
|
|
value_fn=lambda vin: vin or "disconnected",
|
|
),
|
|
)
|
|
|
|
ENERGY_INFO_DESCRIPTIONS: tuple[SensorEntityDescription, ...] = (
|
|
SensorEntityDescription(
|
|
key="vpp_backup_reserve_percent",
|
|
entity_category=EntityCategory.DIAGNOSTIC,
|
|
device_class=SensorDeviceClass.BATTERY,
|
|
native_unit_of_measurement=PERCENTAGE,
|
|
),
|
|
SensorEntityDescription(key="version"),
|
|
)
|
|
|
|
ENERGY_HISTORY_DESCRIPTIONS: tuple[SensorEntityDescription, ...] = tuple(
|
|
SensorEntityDescription(
|
|
key=key,
|
|
device_class=SensorDeviceClass.ENERGY,
|
|
native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
|
|
suggested_unit_of_measurement=UnitOfEnergy.KILO_WATT_HOUR,
|
|
suggested_display_precision=2,
|
|
state_class=SensorStateClass.TOTAL_INCREASING,
|
|
entity_registry_enabled_default=(
|
|
key.startswith("total") or key == "grid_energy_imported"
|
|
),
|
|
)
|
|
for key in ENERGY_HISTORY_FIELDS
|
|
)
|
|
|
|
|
|
async def async_setup_entry(
|
|
hass: HomeAssistant,
|
|
entry: TeslemetryConfigEntry,
|
|
async_add_entities: AddEntitiesCallback,
|
|
) -> None:
|
|
"""Set up the Teslemetry sensor platform from a config entry."""
|
|
|
|
entities: list[SensorEntity] = []
|
|
for vehicle in entry.runtime_data.vehicles:
|
|
for description in VEHICLE_DESCRIPTIONS:
|
|
if (
|
|
not vehicle.api.pre2021
|
|
and description.streaming_key
|
|
and vehicle.firmware >= description.streaming_firmware
|
|
):
|
|
entities.append(TeslemetryStreamSensorEntity(vehicle, description))
|
|
elif description.polling:
|
|
entities.append(TeslemetryVehicleSensorEntity(vehicle, description))
|
|
|
|
for time_description in VEHICLE_TIME_DESCRIPTIONS:
|
|
if (
|
|
not vehicle.api.pre2021
|
|
and vehicle.firmware >= time_description.streaming_firmware
|
|
):
|
|
entities.append(
|
|
TeslemetryStreamTimeSensorEntity(vehicle, time_description)
|
|
)
|
|
else:
|
|
entities.append(
|
|
TeslemetryVehicleTimeSensorEntity(vehicle, time_description)
|
|
)
|
|
|
|
entities.extend(
|
|
TeslemetryEnergyLiveSensorEntity(energysite, description)
|
|
for energysite in entry.runtime_data.energysites
|
|
if energysite.live_coordinator
|
|
for description in ENERGY_LIVE_DESCRIPTIONS
|
|
if description.key in energysite.live_coordinator.data
|
|
)
|
|
|
|
entities.extend(
|
|
TeslemetryWallConnectorSensorEntity(energysite, din, description)
|
|
for energysite in entry.runtime_data.energysites
|
|
if energysite.live_coordinator
|
|
for din in energysite.live_coordinator.data.get("wall_connectors", {})
|
|
for description in WALL_CONNECTOR_DESCRIPTIONS
|
|
)
|
|
|
|
entities.extend(
|
|
TeslemetryEnergyInfoSensorEntity(energysite, description)
|
|
for energysite in entry.runtime_data.energysites
|
|
for description in ENERGY_INFO_DESCRIPTIONS
|
|
if description.key in energysite.info_coordinator.data
|
|
)
|
|
|
|
entities.extend(
|
|
TeslemetryEnergyHistorySensorEntity(energysite, description)
|
|
for energysite in entry.runtime_data.energysites
|
|
for description in ENERGY_HISTORY_DESCRIPTIONS
|
|
if energysite.history_coordinator is not None
|
|
)
|
|
|
|
async_add_entities(entities)
|
|
|
|
|
|
class TeslemetryStreamSensorEntity(TeslemetryVehicleStreamEntity, RestoreSensor):
|
|
"""Base class for Teslemetry vehicle streaming sensors."""
|
|
|
|
entity_description: TeslemetryVehicleSensorEntityDescription
|
|
|
|
def __init__(
|
|
self,
|
|
data: TeslemetryVehicleData,
|
|
description: TeslemetryVehicleSensorEntityDescription,
|
|
) -> None:
|
|
"""Initialize the sensor."""
|
|
self.entity_description = description
|
|
assert description.streaming_key
|
|
super().__init__(data, description.key, description.streaming_key)
|
|
|
|
async def async_added_to_hass(self) -> None:
|
|
"""Handle entity which will be added."""
|
|
await super().async_added_to_hass()
|
|
|
|
if (sensor_data := await self.async_get_last_sensor_data()) is not None:
|
|
self._attr_native_value = sensor_data.native_value
|
|
|
|
@cached_property
|
|
def available(self) -> bool:
|
|
"""Return True if entity is available."""
|
|
return self.stream.connected
|
|
|
|
def _async_value_from_stream(self, value) -> None:
|
|
"""Update the value of the entity."""
|
|
if value is None:
|
|
self._attr_native_value = None
|
|
else:
|
|
self._attr_native_value = self.entity_description.streaming_value_fn(value)
|
|
|
|
|
|
class TeslemetryVehicleSensorEntity(TeslemetryVehicleEntity, SensorEntity):
|
|
"""Base class for Teslemetry vehicle metric sensors."""
|
|
|
|
entity_description: TeslemetryVehicleSensorEntityDescription
|
|
|
|
def __init__(
|
|
self,
|
|
data: TeslemetryVehicleData,
|
|
description: TeslemetryVehicleSensorEntityDescription,
|
|
) -> None:
|
|
"""Initialize the sensor."""
|
|
self.entity_description = description
|
|
super().__init__(data, description.key)
|
|
|
|
def _async_update_attrs(self) -> None:
|
|
"""Update the attributes of the sensor."""
|
|
if self.entity_description.polling_available_fn(self._value):
|
|
self._attr_available = True
|
|
self._attr_native_value = self.entity_description.polling_value_fn(
|
|
self._value
|
|
)
|
|
else:
|
|
self._attr_available = False
|
|
self._attr_native_value = None
|
|
|
|
|
|
class TeslemetryStreamTimeSensorEntity(TeslemetryVehicleStreamEntity, SensorEntity):
|
|
"""Base class for Teslemetry vehicle streaming sensors."""
|
|
|
|
entity_description: TeslemetryTimeEntityDescription
|
|
|
|
def __init__(
|
|
self,
|
|
data: TeslemetryVehicleData,
|
|
description: TeslemetryTimeEntityDescription,
|
|
) -> None:
|
|
"""Initialize the sensor."""
|
|
self.entity_description = description
|
|
self._get_timestamp = ignore_variance(
|
|
func=lambda value: dt_util.now() + timedelta(minutes=value),
|
|
ignored_variance=timedelta(minutes=description.variance),
|
|
)
|
|
assert description.streaming_key
|
|
super().__init__(data, description.key, description.streaming_key)
|
|
|
|
@cached_property
|
|
def available(self) -> bool:
|
|
"""Return True if entity is available."""
|
|
return self.stream.connected
|
|
|
|
def _async_value_from_stream(self, value) -> None:
|
|
"""Update the value of the entity."""
|
|
if value is None:
|
|
self._attr_native_value = None
|
|
else:
|
|
self._attr_native_value = self._get_timestamp(value)
|
|
|
|
|
|
class TeslemetryVehicleTimeSensorEntity(TeslemetryVehicleEntity, SensorEntity):
|
|
"""Base class for Teslemetry vehicle time sensors."""
|
|
|
|
entity_description: TeslemetryTimeEntityDescription
|
|
|
|
def __init__(
|
|
self,
|
|
data: TeslemetryVehicleData,
|
|
description: TeslemetryTimeEntityDescription,
|
|
) -> None:
|
|
"""Initialize the sensor."""
|
|
self.entity_description = description
|
|
self._get_timestamp = ignore_variance(
|
|
func=lambda value: dt_util.now() + timedelta(minutes=value),
|
|
ignored_variance=timedelta(minutes=description.variance),
|
|
)
|
|
|
|
super().__init__(data, description.key)
|
|
|
|
def _async_update_attrs(self) -> None:
|
|
"""Update the attributes of the sensor."""
|
|
self._attr_available = isinstance(self._value, int | float) and self._value > 0
|
|
if self._attr_available:
|
|
self._attr_native_value = self._get_timestamp(self._value)
|
|
|
|
|
|
class TeslemetryEnergyLiveSensorEntity(TeslemetryEnergyLiveEntity, SensorEntity):
|
|
"""Base class for Teslemetry energy site metric sensors."""
|
|
|
|
entity_description: SensorEntityDescription
|
|
|
|
def __init__(
|
|
self,
|
|
data: TeslemetryEnergyData,
|
|
description: SensorEntityDescription,
|
|
) -> None:
|
|
"""Initialize the sensor."""
|
|
self.entity_description = description
|
|
super().__init__(data, description.key)
|
|
|
|
def _async_update_attrs(self) -> None:
|
|
"""Update the attributes of the sensor."""
|
|
self._attr_available = not self.is_none
|
|
self._attr_native_value = self._value
|
|
|
|
|
|
class TeslemetryWallConnectorSensorEntity(TeslemetryWallConnectorEntity, SensorEntity):
|
|
"""Base class for Teslemetry energy site metric sensors."""
|
|
|
|
entity_description: TeslemetrySensorEntityDescription
|
|
|
|
def __init__(
|
|
self,
|
|
data: TeslemetryEnergyData,
|
|
din: str,
|
|
description: TeslemetrySensorEntityDescription,
|
|
) -> None:
|
|
"""Initialize the sensor."""
|
|
self.entity_description = description
|
|
super().__init__(
|
|
data,
|
|
din,
|
|
description.key,
|
|
)
|
|
|
|
def _async_update_attrs(self) -> None:
|
|
"""Update the attributes of the sensor."""
|
|
if self.exists:
|
|
self._attr_native_value = self.entity_description.value_fn(self._value)
|
|
|
|
|
|
class TeslemetryEnergyInfoSensorEntity(TeslemetryEnergyInfoEntity, SensorEntity):
|
|
"""Base class for Teslemetry energy site metric sensors."""
|
|
|
|
entity_description: SensorEntityDescription
|
|
|
|
def __init__(
|
|
self,
|
|
data: TeslemetryEnergyData,
|
|
description: SensorEntityDescription,
|
|
) -> None:
|
|
"""Initialize the sensor."""
|
|
self.entity_description = description
|
|
super().__init__(data, description.key)
|
|
|
|
def _async_update_attrs(self) -> None:
|
|
"""Update the attributes of the sensor."""
|
|
self._attr_available = not self.is_none
|
|
self._attr_native_value = self._value
|
|
|
|
|
|
class TeslemetryEnergyHistorySensorEntity(TeslemetryEnergyHistoryEntity, SensorEntity):
|
|
"""Base class for Tesla Fleet energy site metric sensors."""
|
|
|
|
entity_description: SensorEntityDescription
|
|
|
|
def __init__(
|
|
self,
|
|
data: TeslemetryEnergyData,
|
|
description: SensorEntityDescription,
|
|
) -> None:
|
|
"""Initialize the sensor."""
|
|
self.entity_description = description
|
|
super().__init__(data, description.key)
|
|
|
|
def _async_update_attrs(self) -> None:
|
|
"""Update the attributes of the sensor."""
|
|
self._attr_native_value = self._value
|