Files
HomeAssistant/custom_components/nordpool_planner/sensor.py
T
2025-09-11 10:47:34 +03:00

222 lines
6.9 KiB
Python

"""Binary sensor definitions."""
from __future__ import annotations
import logging
from homeassistant.components.sensor import (
RestoreSensor,
SensorDeviceClass,
SensorEntity,
SensorEntityDescription,
)
from homeassistant.config_entries import ConfigEntry
from homeassistant.const import STATE_UNAVAILABLE, STATE_UNKNOWN, EntityCategory
from homeassistant.core import HomeAssistant
from . import NordpoolPlanner, NordpoolPlannerEntity
from .const import (
CONF_HEALTH_ENTITY,
CONF_HIGH_COST_ENTITY,
CONF_LOW_COST_ENTITY,
CONF_STARTS_AT_ENTITY,
CONF_USED_HOURS_LOW_ENTITY,
DOMAIN,
PlannerStates,
)
_LOGGER = logging.getLogger(__name__)
CONF_LOW_COST_STARTS_AT_ENTITY = (
CONF_LOW_COST_ENTITY.replace("_entity", "") + "_" + CONF_STARTS_AT_ENTITY
)
CONF_HIGH_COST_STARTS_AT_ENTITY = (
CONF_HIGH_COST_ENTITY.replace("_entity", "") + "_" + CONF_STARTS_AT_ENTITY
)
LOW_COST_START_AT_ENTITY_DESCRIPTION = SensorEntityDescription(
key=CONF_LOW_COST_STARTS_AT_ENTITY,
device_class=SensorDeviceClass.TIMESTAMP,
)
HIGH_COST_START_AT_ENTITY_DESCRIPTION = SensorEntityDescription(
key=CONF_HIGH_COST_STARTS_AT_ENTITY,
device_class=SensorDeviceClass.TIMESTAMP,
)
USED_HOURS_LOW_ENTITY_DESCRIPTION = SensorEntityDescription(
key=CONF_USED_HOURS_LOW_ENTITY,
device_class=SensorDeviceClass.DURATION,
entity_category=EntityCategory.DIAGNOSTIC,
)
HEALTH_ENTITY_DESCRIPTION = SensorEntityDescription(
key=CONF_HEALTH_ENTITY,
device_class=SensorDeviceClass.ENUM,
entity_category=EntityCategory.DIAGNOSTIC,
options=[e.name for e in PlannerStates],
)
async def async_setup_entry(
hass: HomeAssistant, config_entry: ConfigEntry, async_add_entities
):
"""Create state sensor entities for platform."""
planner: NordpoolPlanner = hass.data[DOMAIN][config_entry.entry_id]
entities = []
if config_entry.data.get(CONF_STARTS_AT_ENTITY):
if config_entry.data.get(CONF_LOW_COST_ENTITY):
entities.append(
NordpoolPlannerStartAtSensor(
planner,
entity_description=LOW_COST_START_AT_ENTITY_DESCRIPTION,
)
)
if config_entry.data.get(CONF_HIGH_COST_ENTITY):
entities.append(
NordpoolPlannerStartAtSensor(
planner,
entity_description=HIGH_COST_START_AT_ENTITY_DESCRIPTION,
)
)
if config_entry.data.get(CONF_USED_HOURS_LOW_ENTITY):
entities.append(
NordpoolPlannerUsedHoursSensor(
planner,
entity_description=USED_HOURS_LOW_ENTITY_DESCRIPTION,
)
)
if config_entry.data.get(CONF_HEALTH_ENTITY):
entities.append(
NordpoolPlannerHealthSensor(
planner,
entity_description=HEALTH_ENTITY_DESCRIPTION,
)
)
async_add_entities(entities)
return True
class NordpoolPlannerSensor(NordpoolPlannerEntity, SensorEntity):
"""Generic state sensor."""
_attr_icon = "mdi:flash"
def __init__(
self,
planner,
entity_description: SensorEntityDescription,
) -> None:
"""Initialize the entity."""
super().__init__(planner)
self.entity_description = entity_description
self._attr_name = (
self._planner.name
+ " "
+ entity_description.key.replace("_entity", "").replace("_", " ")
)
self._attr_unique_id = (
("nordpool_planner_" + self._attr_name)
.lower()
.replace(".", "")
.replace(" ", "_")
)
async def async_added_to_hass(self) -> None:
"""Load the last known state when added to hass."""
await super().async_added_to_hass()
self._planner.register_output_listener_entity(self, self.entity_description.key)
class NordpoolPlannerStartAtSensor(NordpoolPlannerSensor):
"""Start at specific sensor."""
@property
def native_value(self):
"""Output state."""
state = STATE_UNKNOWN
# TODO: This can be made nicer to get value from states in dictionary in planner
if self.entity_description.key == CONF_LOW_COST_STARTS_AT_ENTITY:
if self._planner.low_cost_state.starts_at not in [
STATE_UNKNOWN,
STATE_UNAVAILABLE,
]:
state = self._planner.low_cost_state.starts_at
if self.entity_description.key == CONF_HIGH_COST_STARTS_AT_ENTITY:
if self._planner.high_cost_state.starts_at not in [
STATE_UNKNOWN,
STATE_UNAVAILABLE,
]:
state = self._planner.high_cost_state.starts_at
_LOGGER.debug(
'Returning state "%s" of sensor "%s"',
state,
self.unique_id,
)
return state
# @property
# def extra_state_attributes(self):
# """Extra state attributes."""
# state_attributes = {
# "cost_at": STATE_UNKNOWN,
# "now_cost_rate": STATE_UNKNOWN,
# }
# # TODO: This can be made nicer to get value from states in dictionary in planner
# if self.entity_description.key == CONF_LOW_COST_STARTS_AT_ENTITY:
# state_attributes = {
# "cost_at": self._planner.low_cost_state.cost_at,
# "now_cost_rate": self._planner.low_cost_state.now_cost_rate,
# }
# _LOGGER.debug(
# 'Returning extra state attributes "%s" of sensor "%s"',
# state_attributes,
# self.unique_id,
# )
# return state_attributes
class NordpoolPlannerUsedHoursSensor(NordpoolPlannerSensor, RestoreSensor):
"""Start at specific sensor."""
async def async_added_to_hass(self) -> None:
"""Restore last state."""
await super().async_added_to_hass()
if (
(last_state := await self.async_get_last_state()) is not None
and last_state.state not in (STATE_UNKNOWN, STATE_UNAVAILABLE)
and last_state.state.isdigit()
# and (extra_data := await self.async_get_last_sensor_data()) is not None
):
self._planner.low_hours = int(last_state.state)
else:
self._planner.low_hours = 0
@property
def native_value(self):
"""Output state."""
return self._planner.low_hours
class NordpoolPlannerHealthSensor(NordpoolPlannerSensor, RestoreSensor):
"""Start at specific sensor."""
@property
def native_value(self):
"""Output state."""
return self._planner.planner_status.status.name
@property
def extra_state_attributes(self):
"""Extra state attributes."""
return {
"running_state": self._planner.planner_status.running_text,
"config_state": self._planner.planner_status.config_text,
}