core/homeassistant/components/watttime/sensor.py

120 lines
3.8 KiB
Python
Raw Normal View History

2021-09-21 04:10:24 +00:00
"""Support for WattTime sensors."""
from __future__ import annotations
from collections.abc import Mapping
from typing import Any, cast
2021-09-21 04:10:24 +00:00
from homeassistant.components.sensor import (
STATE_CLASS_MEASUREMENT,
SensorEntity,
SensorEntityDescription,
)
from homeassistant.config_entries import ConfigEntry
from homeassistant.const import (
ATTR_ATTRIBUTION,
ATTR_LATITUDE,
ATTR_LONGITUDE,
MASS_POUNDS,
PERCENTAGE,
)
from homeassistant.core import HomeAssistant
2021-09-21 04:10:24 +00:00
from homeassistant.helpers.entity_platform import AddEntitiesCallback
from homeassistant.helpers.typing import StateType
2021-09-21 04:10:24 +00:00
from homeassistant.helpers.update_coordinator import (
CoordinatorEntity,
DataUpdateCoordinator,
)
from .const import (
CONF_BALANCING_AUTHORITY,
CONF_BALANCING_AUTHORITY_ABBREV,
CONF_SHOW_ON_MAP,
2021-09-21 04:10:24 +00:00
DATA_COORDINATOR,
DOMAIN,
)
ATTR_BALANCING_AUTHORITY = "balancing_authority"
DEFAULT_ATTRIBUTION = "Pickup data provided by WattTime"
SENSOR_TYPE_REALTIME_EMISSIONS_MOER = "moer"
SENSOR_TYPE_REALTIME_EMISSIONS_PERCENT = "percent"
2021-09-21 04:10:24 +00:00
REALTIME_EMISSIONS_SENSOR_DESCRIPTIONS = (
SensorEntityDescription(
2021-09-21 04:10:24 +00:00
key=SENSOR_TYPE_REALTIME_EMISSIONS_MOER,
name="Marginal Operating Emissions Rate",
icon="mdi:blur",
native_unit_of_measurement=f"{MASS_POUNDS} CO2/MWh",
state_class=STATE_CLASS_MEASUREMENT,
),
SensorEntityDescription(
2021-09-21 04:10:24 +00:00
key=SENSOR_TYPE_REALTIME_EMISSIONS_PERCENT,
name="Relative Marginal Emissions Intensity",
icon="mdi:blur",
native_unit_of_measurement=PERCENTAGE,
state_class=STATE_CLASS_MEASUREMENT,
),
)
async def async_setup_entry(
hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
) -> None:
"""Set up WattTime sensors based on a config entry."""
coordinator = hass.data[DOMAIN][entry.entry_id][DATA_COORDINATOR]
2021-09-21 04:10:24 +00:00
async_add_entities(
[
RealtimeEmissionsSensor(coordinator, entry, description)
2021-09-21 04:10:24 +00:00
for description in REALTIME_EMISSIONS_SENSOR_DESCRIPTIONS
if description.key in coordinator.data
2021-09-21 04:10:24 +00:00
]
)
class RealtimeEmissionsSensor(CoordinatorEntity, SensorEntity):
"""Define a realtime emissions sensor."""
def __init__(
self,
coordinator: DataUpdateCoordinator,
entry: ConfigEntry,
description: SensorEntityDescription,
2021-09-21 04:10:24 +00:00
) -> None:
"""Initialize the sensor."""
super().__init__(coordinator)
self._attr_name = (
f"{description.name} ({entry.data[CONF_BALANCING_AUTHORITY_ABBREV]})"
)
self._attr_unique_id = f"{entry.entry_id}_{description.key}"
self._entry = entry
self.entity_description = description
2021-09-21 04:10:24 +00:00
@property
def extra_state_attributes(self) -> Mapping[str, Any] | None:
"""Return entity specific state attributes."""
attrs = {
2021-09-21 04:10:24 +00:00
ATTR_ATTRIBUTION: DEFAULT_ATTRIBUTION,
ATTR_BALANCING_AUTHORITY: self._entry.data[CONF_BALANCING_AUTHORITY],
2021-09-21 04:10:24 +00:00
}
# Displaying the geography on the map relies upon putting the latitude/longitude
# in the entity attributes with "latitude" and "longitude" as the keys.
# Conversely, we can hide the location on the map by using other keys, like
# "lati" and "long".
if self._entry.options.get(CONF_SHOW_ON_MAP) is not False:
attrs[ATTR_LATITUDE] = self._entry.data[ATTR_LATITUDE]
attrs[ATTR_LONGITUDE] = self._entry.data[ATTR_LONGITUDE]
else:
attrs["lati"] = self._entry.data[ATTR_LATITUDE]
attrs["long"] = self._entry.data[ATTR_LONGITUDE]
return attrs
2021-09-21 04:10:24 +00:00
@property
def native_value(self) -> StateType:
"""Return the value reported by the sensor."""
return cast(StateType, self.coordinator.data[self.entity_description.key])