1
0
mirror of https://github.com/home-assistant/core.git synced 2026-06-01 21:24:17 +01:00
Files
core/homeassistant/components/solaredge/sensor.py
T

615 lines
22 KiB
Python

"""Support for SolarEdge Monitoring API."""
from dataclasses import dataclass
from typing import Any
from aiosolaredge import SolarEdge
from homeassistant.components.sensor import (
SensorDeviceClass,
SensorEntity,
SensorEntityDescription,
SensorStateClass,
)
from homeassistant.const import PERCENTAGE, UnitOfEnergy, UnitOfPower
from homeassistant.core import HomeAssistant
from homeassistant.helpers.device_registry import DeviceInfo
from homeassistant.helpers.entity_platform import AddConfigEntryEntitiesCallback
from homeassistant.helpers.update_coordinator import (
CoordinatorEntity,
DataUpdateCoordinator,
)
from .const import CONF_SITE_ID, DATA_API_CLIENT, DOMAIN, LOGGER
from .coordinator import (
SolarEdgeDataService,
SolarEdgeDetailsDataService,
SolarEdgeEnergyDetailsService,
SolarEdgeInventoryDataService,
SolarEdgeOverviewDataService,
SolarEdgePowerFlowDataService,
SolarEdgeStorageDataService,
)
from .types import SolarEdgeConfigEntry
@dataclass(frozen=True, kw_only=True)
class SolarEdgeSensorEntityDescription(SensorEntityDescription):
"""Sensor entity description for SolarEdge."""
json_key: str
SENSOR_TYPES = [
SolarEdgeSensorEntityDescription(
key="lifetime_energy",
json_key="lifeTimeData",
translation_key="lifetime_energy",
state_class=SensorStateClass.TOTAL_INCREASING,
native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
device_class=SensorDeviceClass.ENERGY,
),
SolarEdgeSensorEntityDescription(
key="energy_this_year",
json_key="lastYearData",
translation_key="energy_this_year",
entity_registry_enabled_default=False,
state_class=SensorStateClass.TOTAL_INCREASING,
native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
device_class=SensorDeviceClass.ENERGY,
),
SolarEdgeSensorEntityDescription(
key="energy_this_month",
json_key="lastMonthData",
translation_key="energy_this_month",
entity_registry_enabled_default=False,
state_class=SensorStateClass.TOTAL_INCREASING,
native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
device_class=SensorDeviceClass.ENERGY,
),
SolarEdgeSensorEntityDescription(
key="energy_today",
json_key="lastDayData",
translation_key="energy_today",
entity_registry_enabled_default=False,
state_class=SensorStateClass.TOTAL_INCREASING,
native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
device_class=SensorDeviceClass.ENERGY,
),
SolarEdgeSensorEntityDescription(
key="current_power",
json_key="currentPower",
translation_key="current_power",
state_class=SensorStateClass.MEASUREMENT,
native_unit_of_measurement=UnitOfPower.WATT,
device_class=SensorDeviceClass.POWER,
),
SolarEdgeSensorEntityDescription(
key="site_details",
json_key="status",
translation_key="site_details",
entity_registry_enabled_default=False,
),
SolarEdgeSensorEntityDescription(
key="meters",
json_key="meters",
translation_key="meters",
entity_registry_enabled_default=False,
),
SolarEdgeSensorEntityDescription(
key="sensors",
json_key="sensors",
translation_key="sensors",
entity_registry_enabled_default=False,
),
SolarEdgeSensorEntityDescription(
key="gateways",
json_key="gateways",
translation_key="gateways",
entity_registry_enabled_default=False,
),
SolarEdgeSensorEntityDescription(
key="batteries",
json_key="batteries",
translation_key="batteries",
entity_registry_enabled_default=False,
),
SolarEdgeSensorEntityDescription(
key="inverters",
json_key="inverters",
translation_key="inverters",
entity_registry_enabled_default=False,
),
SolarEdgeSensorEntityDescription(
key="power_consumption",
json_key="LOAD",
translation_key="power_consumption",
entity_registry_enabled_default=False,
state_class=SensorStateClass.MEASUREMENT,
device_class=SensorDeviceClass.POWER,
),
SolarEdgeSensorEntityDescription(
key="solar_power",
json_key="PV",
translation_key="solar_power",
entity_registry_enabled_default=False,
state_class=SensorStateClass.MEASUREMENT,
device_class=SensorDeviceClass.POWER,
),
SolarEdgeSensorEntityDescription(
key="grid_power",
json_key="GRID",
translation_key="grid_power",
entity_registry_enabled_default=False,
state_class=SensorStateClass.MEASUREMENT,
device_class=SensorDeviceClass.POWER,
),
SolarEdgeSensorEntityDescription(
key="storage_power",
json_key="STORAGE",
translation_key="storage_power",
entity_registry_enabled_default=False,
state_class=SensorStateClass.MEASUREMENT,
device_class=SensorDeviceClass.POWER,
),
SolarEdgeSensorEntityDescription(
key="grid_flow_direction",
json_key="grid_flow_direction",
translation_key="grid_flow_direction",
entity_registry_enabled_default=False,
device_class=SensorDeviceClass.ENUM,
options=["export", "import"],
),
SolarEdgeSensorEntityDescription(
key="storage_flow_direction",
json_key="storage_flow_direction",
translation_key="storage_flow_direction",
entity_registry_enabled_default=False,
device_class=SensorDeviceClass.ENUM,
options=["charge", "discharge"],
),
SolarEdgeSensorEntityDescription(
key="purchased_energy",
json_key="Purchased",
translation_key="purchased_energy",
entity_registry_enabled_default=False,
state_class=SensorStateClass.TOTAL_INCREASING,
native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
device_class=SensorDeviceClass.ENERGY,
),
SolarEdgeSensorEntityDescription(
key="production_energy",
json_key="Production",
translation_key="production_energy",
entity_registry_enabled_default=False,
state_class=SensorStateClass.TOTAL_INCREASING,
native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
device_class=SensorDeviceClass.ENERGY,
),
SolarEdgeSensorEntityDescription(
key="consumption_energy",
json_key="Consumption",
translation_key="consumption_energy",
entity_registry_enabled_default=False,
state_class=SensorStateClass.TOTAL_INCREASING,
native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
device_class=SensorDeviceClass.ENERGY,
),
SolarEdgeSensorEntityDescription(
key="selfconsumption_energy",
json_key="SelfConsumption",
translation_key="selfconsumption_energy",
entity_registry_enabled_default=False,
state_class=SensorStateClass.TOTAL_INCREASING,
native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
device_class=SensorDeviceClass.ENERGY,
),
SolarEdgeSensorEntityDescription(
key="feedin_energy",
json_key="FeedIn",
translation_key="feedin_energy",
entity_registry_enabled_default=False,
state_class=SensorStateClass.TOTAL_INCREASING,
native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
device_class=SensorDeviceClass.ENERGY,
),
SolarEdgeSensorEntityDescription(
key="storage_level",
json_key="storage_level",
translation_key="storage_level",
entity_registry_enabled_default=False,
state_class=SensorStateClass.MEASUREMENT,
native_unit_of_measurement=PERCENTAGE,
device_class=SensorDeviceClass.BATTERY,
),
SolarEdgeSensorEntityDescription(
key="storage_charge_energy",
json_key="charge_energy",
translation_key="storage_charge_energy",
entity_registry_enabled_default=False,
state_class=SensorStateClass.TOTAL_INCREASING,
native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
device_class=SensorDeviceClass.ENERGY,
),
SolarEdgeSensorEntityDescription(
key="storage_discharge_energy",
json_key="discharge_energy",
translation_key="storage_discharge_energy",
entity_registry_enabled_default=False,
state_class=SensorStateClass.TOTAL_INCREASING,
native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
device_class=SensorDeviceClass.ENERGY,
),
]
# Per-battery sensor descriptions, created dynamically per serial number
BATTERY_SENSOR_TYPES = [
SolarEdgeSensorEntityDescription(
key="battery_charge_energy",
json_key="charge_energy",
translation_key="battery_charge_energy",
entity_registry_enabled_default=False,
state_class=SensorStateClass.TOTAL_INCREASING,
native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
device_class=SensorDeviceClass.ENERGY,
),
SolarEdgeSensorEntityDescription(
key="battery_discharge_energy",
json_key="discharge_energy",
translation_key="battery_discharge_energy",
entity_registry_enabled_default=False,
state_class=SensorStateClass.TOTAL_INCREASING,
native_unit_of_measurement=UnitOfEnergy.WATT_HOUR,
device_class=SensorDeviceClass.ENERGY,
),
SolarEdgeSensorEntityDescription(
key="battery_state_of_charge",
json_key="state_of_charge",
translation_key="battery_state_of_charge",
entity_registry_enabled_default=False,
state_class=SensorStateClass.MEASUREMENT,
native_unit_of_measurement=PERCENTAGE,
device_class=SensorDeviceClass.BATTERY,
),
SolarEdgeSensorEntityDescription(
key="battery_power",
json_key="power",
translation_key="battery_power",
entity_registry_enabled_default=False,
state_class=SensorStateClass.MEASUREMENT,
native_unit_of_measurement=UnitOfPower.WATT,
device_class=SensorDeviceClass.POWER,
),
]
async def async_setup_entry(
hass: HomeAssistant,
entry: SolarEdgeConfigEntry,
async_add_entities: AddConfigEntryEntitiesCallback,
) -> None:
"""Add an solarEdge entry."""
# Add sensor entities only if API key is configured
if DATA_API_CLIENT not in entry.runtime_data:
return
api = entry.runtime_data[DATA_API_CLIENT]
sensor_factory = SolarEdgeSensorFactory(hass, entry, entry.data[CONF_SITE_ID], api)
# Set up and refresh base services first
for service in sensor_factory.all_services:
service.async_setup()
await service.coordinator.async_refresh()
entities: list[SolarEdgeSensorEntity] = []
# Set up storage sensors only if inventory shows batteries are present
storage_result = sensor_factory.setup_storage_sensors()
if storage_result is not None:
if storage_result:
await sensor_factory.storage_service.coordinator.async_refresh()
entities.extend(storage_result)
else:
# Inventory fetch failed, register listener to retry when data arrives
def on_inventory_update() -> None:
"""Handle inventory update to set up storage sensors."""
result = sensor_factory.setup_storage_sensors()
if result is not None:
if result:
hass.async_create_task(
sensor_factory.storage_service.coordinator.async_refresh()
)
async_add_entities(result)
# Success or confirmed no batteries - stop listening
unsub()
unsub = sensor_factory.inventory_service.coordinator.async_add_listener(
on_inventory_update
)
entry.async_on_unload(unsub)
for sensor_type in SENSOR_TYPES:
if sensor_type.key in ("storage_charge_energy", "storage_discharge_energy"):
continue
entities.append(sensor_factory.create_sensor(sensor_type))
async_add_entities(entities)
class SolarEdgeSensorFactory:
"""Factory which creates sensors based on the sensor_key."""
def __init__(
self,
hass: HomeAssistant,
config_entry: SolarEdgeConfigEntry,
site_id: str,
api: SolarEdge,
) -> None:
"""Initialize the factory."""
details = SolarEdgeDetailsDataService(hass, config_entry, api, site_id)
overview = SolarEdgeOverviewDataService(hass, config_entry, api, site_id)
inventory = SolarEdgeInventoryDataService(hass, config_entry, api, site_id)
flow = SolarEdgePowerFlowDataService(hass, config_entry, api, site_id)
energy = SolarEdgeEnergyDetailsService(hass, config_entry, api, site_id)
storage = SolarEdgeStorageDataService(hass, config_entry, api, site_id)
self.all_services: list[SolarEdgeDataService] = [
details,
overview,
inventory,
flow,
energy,
]
self.inventory_service = inventory
self.storage_service = storage
self.services: dict[
str,
tuple[
type[SolarEdgeSensorEntity | SolarEdgeOverviewSensor],
SolarEdgeDataService,
],
] = {"site_details": (SolarEdgeDetailsSensor, details)}
for key in (
"lifetime_energy",
"energy_this_year",
"energy_this_month",
"energy_today",
"current_power",
):
self.services[key] = (SolarEdgeOverviewSensor, overview)
for key in ("meters", "sensors", "gateways", "batteries", "inverters"):
self.services[key] = (SolarEdgeInventorySensor, inventory)
for key in ("power_consumption", "solar_power", "grid_power", "storage_power"):
self.services[key] = (SolarEdgePowerFlowSensor, flow)
for key in ("storage_level", "grid_flow_direction", "storage_flow_direction"):
self.services[key] = (SolarEdgeOverviewSensor, flow)
for key in (
"purchased_energy",
"production_energy",
"feedin_energy",
"consumption_energy",
"selfconsumption_energy",
):
self.services[key] = (SolarEdgeEnergyDetailsSensor, energy)
def setup_storage_sensors(
self,
) -> list[SolarEdgeSensorEntity] | None:
"""Set up storage sensors if batteries are available.
Returns:
list: Storage sensor entities to add (empty if no batteries)
None: Inventory fetch failed, should retry later
"""
# Check if inventory data was successfully fetched
if not self.inventory_service.coordinator.last_update_success:
LOGGER.debug("Inventory data not available, will retry later")
return None
battery_attr = self.inventory_service.attributes.get("batteries", {})
inventory_batteries = battery_attr.get("batteries", [])
if not inventory_batteries:
LOGGER.debug("No batteries found in inventory, skipping storage sensors")
return []
# Set up storage service and add to services
self.storage_service.async_setup()
self.all_services.append(self.storage_service)
for key in ("storage_charge_energy", "storage_discharge_energy"):
self.services[key] = (SolarEdgeStorageDataSensor, self.storage_service)
# Create aggregate storage sensors
storage_entities: list[SolarEdgeSensorEntity] = [
self.create_sensor(sensor_type)
for sensor_type in SENSOR_TYPES
if sensor_type.key in ("storage_charge_energy", "storage_discharge_energy")
]
# Create per-battery entities
for battery in inventory_batteries:
serial = battery.get("SN") or battery.get("serialNumber")
if not serial:
LOGGER.debug("Skipping battery without serial number in inventory")
continue
storage_entities.extend(
SolarEdgeBatterySensor(sensor_type, self.storage_service, serial)
for sensor_type in BATTERY_SENSOR_TYPES
)
LOGGER.debug(
"Storage sensors enabled, found %d batteries", len(inventory_batteries)
)
return storage_entities
def create_sensor(
self, sensor_type: SolarEdgeSensorEntityDescription
) -> SolarEdgeSensorEntity:
"""Create and return a sensor based on the sensor_key."""
sensor_class, service = self.services[sensor_type.key]
return sensor_class(sensor_type, service)
class SolarEdgeSensorEntity(
CoordinatorEntity[DataUpdateCoordinator[None]], SensorEntity
):
"""Abstract class for a solaredge sensor."""
_attr_has_entity_name = True
entity_description: SolarEdgeSensorEntityDescription
def __init__(
self,
description: SolarEdgeSensorEntityDescription,
data_service: SolarEdgeDataService,
) -> None:
"""Initialize the sensor."""
super().__init__(data_service.coordinator)
self.entity_description = description
self.data_service = data_service
self._attr_unique_id = f"{data_service.site_id}_{description.key}"
self._attr_device_info = DeviceInfo(
identifiers={(DOMAIN, data_service.site_id)}, manufacturer="SolarEdge"
)
class SolarEdgeOverviewSensor(SolarEdgeSensorEntity):
"""Representation of an SolarEdge Monitoring API overview sensor."""
@property
def native_value(self) -> str | None:
"""Return the state of the sensor."""
return self.data_service.data.get(self.entity_description.json_key)
class SolarEdgeDetailsSensor(SolarEdgeSensorEntity):
"""Representation of an SolarEdge Monitoring API details sensor."""
@property
def extra_state_attributes(self) -> dict[str, Any]:
"""Return the state attributes."""
return self.data_service.attributes
@property
def native_value(self) -> str | None:
"""Return the state of the sensor."""
return self.data_service.data.get(self.entity_description.json_key)
@property
def unique_id(self) -> str | None:
"""Return a unique ID."""
if not self.data_service.site_id:
return None
return str(self.data_service.site_id)
class SolarEdgeInventorySensor(SolarEdgeSensorEntity):
"""Representation of an SolarEdge Monitoring API inventory sensor."""
@property
def extra_state_attributes(self) -> dict[str, Any] | None:
"""Return the state attributes."""
return self.data_service.attributes.get(self.entity_description.json_key)
@property
def native_value(self) -> str | None:
"""Return the state of the sensor."""
return self.data_service.data.get(self.entity_description.json_key)
class SolarEdgeEnergyDetailsSensor(SolarEdgeSensorEntity):
"""Representation of an SolarEdge Monitoring API power flow sensor."""
def __init__(
self,
sensor_type: SolarEdgeSensorEntityDescription,
data_service: SolarEdgeEnergyDetailsService,
) -> None:
"""Initialize the power flow sensor."""
super().__init__(sensor_type, data_service)
self._attr_native_unit_of_measurement = data_service.unit
@property
def extra_state_attributes(self) -> dict[str, Any] | None:
"""Return the state attributes."""
return self.data_service.attributes.get(self.entity_description.json_key)
@property
def native_value(self) -> str | None:
"""Return the state of the sensor."""
return self.data_service.data.get(self.entity_description.json_key)
class SolarEdgePowerFlowSensor(SolarEdgeSensorEntity):
"""Representation of an SolarEdge Monitoring API power flow sensor."""
_attr_device_class = SensorDeviceClass.POWER
def __init__(
self,
description: SolarEdgeSensorEntityDescription,
data_service: SolarEdgePowerFlowDataService,
) -> None:
"""Initialize the power flow sensor."""
super().__init__(description, data_service)
self._attr_native_unit_of_measurement = data_service.unit
@property
def extra_state_attributes(self) -> dict[str, Any] | None:
"""Return the state attributes."""
return self.data_service.attributes.get(self.entity_description.json_key)
@property
def native_value(self) -> str | None:
"""Return the state of the sensor."""
return self.data_service.data.get(self.entity_description.json_key)
class SolarEdgeStorageDataSensor(SolarEdgeSensorEntity):
"""Representation of an SolarEdge aggregate storage data sensor."""
@property
def native_value(self) -> float | None:
"""Return the state of the sensor."""
return self.data_service.data.get(self.entity_description.json_key)
class SolarEdgeBatterySensor(SolarEdgeSensorEntity):
"""Representation of a per-battery SolarEdge sensor."""
def __init__(
self,
description: SolarEdgeSensorEntityDescription,
data_service: SolarEdgeStorageDataService,
serial: str,
) -> None:
"""Initialize the per-battery sensor."""
super().__init__(description, data_service)
self._serial = serial
self._attr_unique_id = f"{data_service.site_id}_{serial}_{description.key}"
self._attr_device_info = DeviceInfo(
identifiers={(DOMAIN, f"{data_service.site_id}_{serial}")},
manufacturer="SolarEdge",
name=f"Battery {serial}",
serial_number=serial,
via_device=(DOMAIN, data_service.site_id),
)
@property
def native_value(self) -> float | None:
"""Return the state of the sensor."""
return self.data_service.data.get(
f"{self._serial}_{self.entity_description.json_key}"
)