496 lines
17 KiB
Python
496 lines
17 KiB
Python
"""Support for Tibber sensors."""
|
|
from __future__ import annotations
|
|
|
|
import asyncio
|
|
from dataclasses import dataclass
|
|
from datetime import timedelta
|
|
from enum import Enum
|
|
import logging
|
|
from random import randrange
|
|
|
|
import aiohttp
|
|
|
|
from homeassistant.components.sensor import (
|
|
DEVICE_CLASS_CURRENT,
|
|
DEVICE_CLASS_ENERGY,
|
|
DEVICE_CLASS_MONETARY,
|
|
DEVICE_CLASS_POWER,
|
|
DEVICE_CLASS_POWER_FACTOR,
|
|
DEVICE_CLASS_SIGNAL_STRENGTH,
|
|
DEVICE_CLASS_VOLTAGE,
|
|
STATE_CLASS_MEASUREMENT,
|
|
SensorEntity,
|
|
SensorEntityDescription,
|
|
)
|
|
from homeassistant.const import (
|
|
ELECTRIC_CURRENT_AMPERE,
|
|
ELECTRIC_POTENTIAL_VOLT,
|
|
ENERGY_KILO_WATT_HOUR,
|
|
PERCENTAGE,
|
|
POWER_WATT,
|
|
SIGNAL_STRENGTH_DECIBELS,
|
|
)
|
|
from homeassistant.core import callback
|
|
from homeassistant.exceptions import PlatformNotReady
|
|
from homeassistant.helpers.device_registry import async_get as async_get_dev_reg
|
|
from homeassistant.helpers.dispatcher import (
|
|
async_dispatcher_connect,
|
|
async_dispatcher_send,
|
|
)
|
|
from homeassistant.helpers.entity_registry import async_get as async_get_entity_reg
|
|
from homeassistant.util import Throttle, dt as dt_util
|
|
|
|
from .const import DOMAIN as TIBBER_DOMAIN, MANUFACTURER
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
ICON = "mdi:currency-usd"
|
|
SCAN_INTERVAL = timedelta(minutes=1)
|
|
MIN_TIME_BETWEEN_UPDATES = timedelta(minutes=5)
|
|
PARALLEL_UPDATES = 0
|
|
SIGNAL_UPDATE_ENTITY = "tibber_rt_update_{}"
|
|
|
|
|
|
class ResetType(Enum):
|
|
"""Data reset type."""
|
|
|
|
HOURLY = "hourly"
|
|
DAILY = "daily"
|
|
NEVER = "never"
|
|
|
|
|
|
@dataclass
|
|
class TibberSensorEntityDescription(SensorEntityDescription):
|
|
"""Describes Tibber sensor entity."""
|
|
|
|
reset_type: ResetType | None = None
|
|
|
|
|
|
RT_SENSOR_MAP: dict[str, TibberSensorEntityDescription] = {
|
|
"averagePower": TibberSensorEntityDescription(
|
|
key="averagePower",
|
|
name="average power",
|
|
device_class=DEVICE_CLASS_POWER,
|
|
unit_of_measurement=POWER_WATT,
|
|
),
|
|
"power": TibberSensorEntityDescription(
|
|
key="power",
|
|
name="power",
|
|
device_class=DEVICE_CLASS_POWER,
|
|
unit_of_measurement=POWER_WATT,
|
|
),
|
|
"powerProduction": TibberSensorEntityDescription(
|
|
key="powerProduction",
|
|
name="power production",
|
|
device_class=DEVICE_CLASS_POWER,
|
|
unit_of_measurement=POWER_WATT,
|
|
),
|
|
"minPower": TibberSensorEntityDescription(
|
|
key="minPower",
|
|
name="min power",
|
|
device_class=DEVICE_CLASS_POWER,
|
|
unit_of_measurement=POWER_WATT,
|
|
),
|
|
"maxPower": TibberSensorEntityDescription(
|
|
key="maxPower",
|
|
name="max power",
|
|
device_class=DEVICE_CLASS_POWER,
|
|
unit_of_measurement=POWER_WATT,
|
|
),
|
|
"accumulatedConsumption": TibberSensorEntityDescription(
|
|
key="accumulatedConsumption",
|
|
name="accumulated consumption",
|
|
device_class=DEVICE_CLASS_ENERGY,
|
|
unit_of_measurement=ENERGY_KILO_WATT_HOUR,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
reset_type=ResetType.DAILY,
|
|
),
|
|
"accumulatedConsumptionLastHour": TibberSensorEntityDescription(
|
|
key="accumulatedConsumptionLastHour",
|
|
name="accumulated consumption current hour",
|
|
device_class=DEVICE_CLASS_ENERGY,
|
|
unit_of_measurement=ENERGY_KILO_WATT_HOUR,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
reset_type=ResetType.HOURLY,
|
|
),
|
|
"accumulatedProduction": TibberSensorEntityDescription(
|
|
key="accumulatedProduction",
|
|
name="accumulated production",
|
|
device_class=DEVICE_CLASS_ENERGY,
|
|
unit_of_measurement=ENERGY_KILO_WATT_HOUR,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
reset_type=ResetType.DAILY,
|
|
),
|
|
"accumulatedProductionLastHour": TibberSensorEntityDescription(
|
|
key="accumulatedProductionLastHour",
|
|
name="accumulated production current hour",
|
|
device_class=DEVICE_CLASS_ENERGY,
|
|
unit_of_measurement=ENERGY_KILO_WATT_HOUR,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
reset_type=ResetType.HOURLY,
|
|
),
|
|
"lastMeterConsumption": TibberSensorEntityDescription(
|
|
key="lastMeterConsumption",
|
|
name="last meter consumption",
|
|
device_class=DEVICE_CLASS_ENERGY,
|
|
unit_of_measurement=ENERGY_KILO_WATT_HOUR,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
),
|
|
"lastMeterProduction": TibberSensorEntityDescription(
|
|
key="lastMeterProduction",
|
|
name="last meter production",
|
|
device_class=DEVICE_CLASS_ENERGY,
|
|
unit_of_measurement=ENERGY_KILO_WATT_HOUR,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
),
|
|
"voltagePhase1": TibberSensorEntityDescription(
|
|
key="voltagePhase1",
|
|
name="voltage phase1",
|
|
device_class=DEVICE_CLASS_VOLTAGE,
|
|
unit_of_measurement=ELECTRIC_POTENTIAL_VOLT,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
),
|
|
"voltagePhase2": TibberSensorEntityDescription(
|
|
key="voltagePhase2",
|
|
name="voltage phase2",
|
|
device_class=DEVICE_CLASS_VOLTAGE,
|
|
unit_of_measurement=ELECTRIC_POTENTIAL_VOLT,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
),
|
|
"voltagePhase3": TibberSensorEntityDescription(
|
|
key="voltagePhase3",
|
|
name="voltage phase3",
|
|
device_class=DEVICE_CLASS_VOLTAGE,
|
|
unit_of_measurement=ELECTRIC_POTENTIAL_VOLT,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
),
|
|
"currentL1": TibberSensorEntityDescription(
|
|
key="currentL1",
|
|
name="current L1",
|
|
device_class=DEVICE_CLASS_CURRENT,
|
|
unit_of_measurement=ELECTRIC_CURRENT_AMPERE,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
),
|
|
"currentL2": TibberSensorEntityDescription(
|
|
key="currentL2",
|
|
name="current L2",
|
|
device_class=DEVICE_CLASS_CURRENT,
|
|
unit_of_measurement=ELECTRIC_CURRENT_AMPERE,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
),
|
|
"currentL3": TibberSensorEntityDescription(
|
|
key="currentL3",
|
|
name="current L3",
|
|
device_class=DEVICE_CLASS_CURRENT,
|
|
unit_of_measurement=ELECTRIC_CURRENT_AMPERE,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
),
|
|
"signalStrength": TibberSensorEntityDescription(
|
|
key="signalStrength",
|
|
name="signal strength",
|
|
device_class=DEVICE_CLASS_SIGNAL_STRENGTH,
|
|
unit_of_measurement=SIGNAL_STRENGTH_DECIBELS,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
),
|
|
"accumulatedReward": TibberSensorEntityDescription(
|
|
key="accumulatedReward",
|
|
name="accumulated reward",
|
|
device_class=DEVICE_CLASS_MONETARY,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
reset_type=ResetType.DAILY,
|
|
),
|
|
"accumulatedCost": TibberSensorEntityDescription(
|
|
key="accumulatedCost",
|
|
name="accumulated cost",
|
|
device_class=DEVICE_CLASS_MONETARY,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
reset_type=ResetType.DAILY,
|
|
),
|
|
"powerFactor": TibberSensorEntityDescription(
|
|
key="powerFactor",
|
|
name="power factor",
|
|
device_class=DEVICE_CLASS_POWER_FACTOR,
|
|
unit_of_measurement=PERCENTAGE,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
),
|
|
}
|
|
|
|
|
|
async def async_setup_entry(hass, entry, async_add_entities):
|
|
"""Set up the Tibber sensor."""
|
|
|
|
tibber_connection = hass.data.get(TIBBER_DOMAIN)
|
|
|
|
entity_registry = async_get_entity_reg(hass)
|
|
device_registry = async_get_dev_reg(hass)
|
|
|
|
entities = []
|
|
for home in tibber_connection.get_homes(only_active=False):
|
|
try:
|
|
await home.update_info()
|
|
except asyncio.TimeoutError as err:
|
|
_LOGGER.error("Timeout connecting to Tibber home: %s ", err)
|
|
raise PlatformNotReady() from err
|
|
except aiohttp.ClientError as err:
|
|
_LOGGER.error("Error connecting to Tibber home: %s ", err)
|
|
raise PlatformNotReady() from err
|
|
|
|
if home.has_active_subscription:
|
|
entities.append(TibberSensorElPrice(home))
|
|
if home.has_real_time_consumption:
|
|
await home.rt_subscribe(
|
|
TibberRtDataHandler(async_add_entities, home, hass).async_callback
|
|
)
|
|
|
|
# migrate
|
|
old_id = home.info["viewer"]["home"]["meteringPointData"]["consumptionEan"]
|
|
if old_id is None:
|
|
continue
|
|
|
|
# migrate to new device ids
|
|
old_entity_id = entity_registry.async_get_entity_id(
|
|
"sensor", TIBBER_DOMAIN, old_id
|
|
)
|
|
if old_entity_id is not None:
|
|
entity_registry.async_update_entity(
|
|
old_entity_id, new_unique_id=home.home_id
|
|
)
|
|
|
|
# migrate to new device ids
|
|
device_entry = device_registry.async_get_device({(TIBBER_DOMAIN, old_id)})
|
|
if device_entry and entry.entry_id in device_entry.config_entries:
|
|
device_registry.async_update_device(
|
|
device_entry.id, new_identifiers={(TIBBER_DOMAIN, home.home_id)}
|
|
)
|
|
|
|
async_add_entities(entities, True)
|
|
|
|
|
|
class TibberSensor(SensorEntity):
|
|
"""Representation of a generic Tibber sensor."""
|
|
|
|
def __init__(self, tibber_home):
|
|
"""Initialize the sensor."""
|
|
self._tibber_home = tibber_home
|
|
self._home_name = tibber_home.info["viewer"]["home"]["appNickname"]
|
|
self._device_name = None
|
|
if self._home_name is None:
|
|
self._home_name = tibber_home.info["viewer"]["home"]["address"].get(
|
|
"address1", ""
|
|
)
|
|
self._model = None
|
|
|
|
@property
|
|
def device_id(self):
|
|
"""Return the ID of the physical device this sensor is part of."""
|
|
return self._tibber_home.home_id
|
|
|
|
@property
|
|
def device_info(self):
|
|
"""Return the device_info of the device."""
|
|
device_info = {
|
|
"identifiers": {(TIBBER_DOMAIN, self.device_id)},
|
|
"name": self._device_name,
|
|
"manufacturer": MANUFACTURER,
|
|
}
|
|
if self._model is not None:
|
|
device_info["model"] = self._model
|
|
return device_info
|
|
|
|
|
|
class TibberSensorElPrice(TibberSensor):
|
|
"""Representation of a Tibber sensor for el price."""
|
|
|
|
def __init__(self, tibber_home):
|
|
"""Initialize the sensor."""
|
|
super().__init__(tibber_home)
|
|
self._last_updated = None
|
|
self._spread_load_constant = randrange(5000)
|
|
|
|
self._attr_available = False
|
|
self._attr_extra_state_attributes = {
|
|
"app_nickname": None,
|
|
"grid_company": None,
|
|
"estimated_annual_consumption": None,
|
|
"price_level": None,
|
|
"max_price": None,
|
|
"avg_price": None,
|
|
"min_price": None,
|
|
"off_peak_1": None,
|
|
"peak": None,
|
|
"off_peak_2": None,
|
|
}
|
|
self._attr_icon = ICON
|
|
self._attr_name = f"Electricity price {self._home_name}"
|
|
self._attr_unique_id = f"{self._tibber_home.home_id}"
|
|
self._model = "Price Sensor"
|
|
|
|
self._device_name = self._attr_name
|
|
|
|
async def async_update(self):
|
|
"""Get the latest data and updates the states."""
|
|
now = dt_util.now()
|
|
if (
|
|
not self._tibber_home.last_data_timestamp
|
|
or (self._tibber_home.last_data_timestamp - now).total_seconds()
|
|
< 5 * 3600 + self._spread_load_constant
|
|
or not self.available
|
|
):
|
|
_LOGGER.debug("Asking for new data")
|
|
await self._fetch_data()
|
|
|
|
elif (
|
|
self._tibber_home.current_price_total
|
|
and self._last_updated
|
|
and self._last_updated.hour == now.hour
|
|
and self._tibber_home.last_data_timestamp
|
|
):
|
|
return
|
|
|
|
res = self._tibber_home.current_price_data()
|
|
self._attr_state, price_level, self._last_updated = res
|
|
self._attr_extra_state_attributes["price_level"] = price_level
|
|
|
|
attrs = self._tibber_home.current_attributes()
|
|
self._attr_extra_state_attributes.update(attrs)
|
|
self._attr_available = self._attr_state is not None
|
|
self._attr_unit_of_measurement = self._tibber_home.price_unit
|
|
|
|
@Throttle(MIN_TIME_BETWEEN_UPDATES)
|
|
async def _fetch_data(self):
|
|
_LOGGER.debug("Fetching data")
|
|
try:
|
|
await self._tibber_home.update_info_and_price_info()
|
|
except (asyncio.TimeoutError, aiohttp.ClientError):
|
|
return
|
|
data = self._tibber_home.info["viewer"]["home"]
|
|
self._attr_extra_state_attributes["app_nickname"] = data["appNickname"]
|
|
self._attr_extra_state_attributes["grid_company"] = data["meteringPointData"][
|
|
"gridCompany"
|
|
]
|
|
self._attr_extra_state_attributes["estimated_annual_consumption"] = data[
|
|
"meteringPointData"
|
|
]["estimatedAnnualConsumption"]
|
|
|
|
|
|
class TibberSensorRT(TibberSensor):
|
|
"""Representation of a Tibber sensor for real time consumption."""
|
|
|
|
_attr_should_poll = False
|
|
entity_description: TibberSensorEntityDescription
|
|
|
|
def __init__(
|
|
self,
|
|
tibber_home,
|
|
description: TibberSensorEntityDescription,
|
|
initial_state,
|
|
):
|
|
"""Initialize the sensor."""
|
|
super().__init__(tibber_home)
|
|
self.entity_description = description
|
|
self._model = "Tibber Pulse"
|
|
self._device_name = f"{self._model} {self._home_name}"
|
|
|
|
self._attr_name = f"{description.name} {self._home_name}"
|
|
self._attr_state = initial_state
|
|
self._attr_unique_id = f"{self._tibber_home.home_id}_rt_{description.name}"
|
|
|
|
if description.name in ("accumulated cost", "accumulated reward"):
|
|
self._attr_unit_of_measurement = tibber_home.currency
|
|
if description.reset_type == ResetType.NEVER:
|
|
self._attr_last_reset = dt_util.utc_from_timestamp(0)
|
|
elif description.reset_type == ResetType.DAILY:
|
|
self._attr_last_reset = dt_util.as_utc(
|
|
dt_util.now().replace(hour=0, minute=0, second=0, microsecond=0)
|
|
)
|
|
elif description.reset_type == ResetType.HOURLY:
|
|
self._attr_last_reset = dt_util.as_utc(
|
|
dt_util.now().replace(minute=0, second=0, microsecond=0)
|
|
)
|
|
else:
|
|
self._attr_last_reset = None
|
|
|
|
async def async_added_to_hass(self):
|
|
"""Start listen for real time data."""
|
|
self.async_on_remove(
|
|
async_dispatcher_connect(
|
|
self.hass,
|
|
SIGNAL_UPDATE_ENTITY.format(self.unique_id),
|
|
self._set_state,
|
|
)
|
|
)
|
|
|
|
@property
|
|
def available(self):
|
|
"""Return True if entity is available."""
|
|
return self._tibber_home.rt_subscription_running
|
|
|
|
@callback
|
|
def _set_state(self, state, timestamp):
|
|
"""Set sensor state."""
|
|
if (
|
|
state < self._attr_state
|
|
and self.entity_description.reset_type == ResetType.DAILY
|
|
):
|
|
self._attr_last_reset = dt_util.as_utc(
|
|
timestamp.replace(hour=0, minute=0, second=0, microsecond=0)
|
|
)
|
|
if (
|
|
state < self._attr_state
|
|
and self.entity_description.reset_type == ResetType.HOURLY
|
|
):
|
|
self._attr_last_reset = dt_util.as_utc(
|
|
timestamp.replace(minute=0, second=0, microsecond=0)
|
|
)
|
|
self._attr_state = state
|
|
self.async_write_ha_state()
|
|
|
|
|
|
class TibberRtDataHandler:
|
|
"""Handle Tibber realtime data."""
|
|
|
|
def __init__(self, async_add_entities, tibber_home, hass):
|
|
"""Initialize the data handler."""
|
|
self._async_add_entities = async_add_entities
|
|
self._tibber_home = tibber_home
|
|
self.hass = hass
|
|
self._entities = {}
|
|
|
|
async def async_callback(self, payload):
|
|
"""Handle received data."""
|
|
errors = payload.get("errors")
|
|
if errors:
|
|
_LOGGER.error(errors[0])
|
|
return
|
|
data = payload.get("data")
|
|
if data is None:
|
|
return
|
|
live_measurement = data.get("liveMeasurement")
|
|
if live_measurement is None:
|
|
return
|
|
|
|
timestamp = dt_util.parse_datetime(live_measurement.pop("timestamp"))
|
|
new_entities = []
|
|
for sensor_type, state in live_measurement.items():
|
|
if state is None or sensor_type not in RT_SENSOR_MAP:
|
|
continue
|
|
if sensor_type == "powerFactor":
|
|
state *= 100.0
|
|
if sensor_type in self._entities:
|
|
async_dispatcher_send(
|
|
self.hass,
|
|
SIGNAL_UPDATE_ENTITY.format(self._entities[sensor_type]),
|
|
state,
|
|
timestamp,
|
|
)
|
|
else:
|
|
entity = TibberSensorRT(
|
|
self._tibber_home,
|
|
RT_SENSOR_MAP[sensor_type],
|
|
state,
|
|
)
|
|
new_entities.append(entity)
|
|
self._entities[sensor_type] = entity.unique_id
|
|
if new_entities:
|
|
self._async_add_entities(new_entities)
|