2019-02-13 20:21:14 +00:00
|
|
|
"""Support for monitoring a Sense energy sensor."""
|
2021-08-18 14:31:10 +00:00
|
|
|
from homeassistant.components.sensor import (
|
|
|
|
STATE_CLASS_MEASUREMENT,
|
|
|
|
STATE_CLASS_TOTAL_INCREASING,
|
|
|
|
SensorEntity,
|
|
|
|
)
|
2020-04-04 20:51:00 +00:00
|
|
|
from homeassistant.const import (
|
|
|
|
ATTR_ATTRIBUTION,
|
2021-07-29 04:58:28 +00:00
|
|
|
DEVICE_CLASS_ENERGY,
|
2020-04-04 20:51:00 +00:00
|
|
|
DEVICE_CLASS_POWER,
|
2021-07-20 18:06:23 +00:00
|
|
|
ELECTRIC_POTENTIAL_VOLT,
|
2020-04-04 20:51:00 +00:00
|
|
|
ENERGY_KILO_WATT_HOUR,
|
2021-08-30 15:01:26 +00:00
|
|
|
PERCENTAGE,
|
2020-04-04 20:51:00 +00:00
|
|
|
POWER_WATT,
|
|
|
|
)
|
2020-02-28 05:23:47 +00:00
|
|
|
from homeassistant.core import callback
|
|
|
|
from homeassistant.helpers.dispatcher import async_dispatcher_connect
|
2018-02-28 22:29:24 +00:00
|
|
|
|
2020-02-28 05:23:47 +00:00
|
|
|
from .const import (
|
|
|
|
ACTIVE_NAME,
|
|
|
|
ACTIVE_TYPE,
|
2020-04-04 20:51:00 +00:00
|
|
|
ATTRIBUTION,
|
2020-02-28 05:23:47 +00:00
|
|
|
CONSUMPTION_ID,
|
|
|
|
CONSUMPTION_NAME,
|
|
|
|
DOMAIN,
|
2021-08-30 15:01:26 +00:00
|
|
|
FROM_GRID_ID,
|
|
|
|
FROM_GRID_NAME,
|
2020-02-28 05:23:47 +00:00
|
|
|
ICON,
|
|
|
|
MDI_ICONS,
|
2021-08-30 15:01:26 +00:00
|
|
|
NET_PRODUCTION_ID,
|
|
|
|
NET_PRODUCTION_NAME,
|
2020-02-28 05:23:47 +00:00
|
|
|
PRODUCTION_ID,
|
|
|
|
PRODUCTION_NAME,
|
2021-08-30 15:01:26 +00:00
|
|
|
PRODUCTION_PCT_ID,
|
|
|
|
PRODUCTION_PCT_NAME,
|
2020-02-28 05:23:47 +00:00
|
|
|
SENSE_DATA,
|
|
|
|
SENSE_DEVICE_UPDATE,
|
|
|
|
SENSE_DEVICES_DATA,
|
|
|
|
SENSE_DISCOVERED_DEVICES_DATA,
|
2020-04-14 00:08:37 +00:00
|
|
|
SENSE_TRENDS_COORDINATOR,
|
2021-08-30 15:01:26 +00:00
|
|
|
SOLAR_POWERED_ID,
|
|
|
|
SOLAR_POWERED_NAME,
|
|
|
|
TO_GRID_ID,
|
|
|
|
TO_GRID_NAME,
|
2020-02-28 05:23:47 +00:00
|
|
|
)
|
2018-12-01 18:27:21 +00:00
|
|
|
|
2018-02-28 22:29:24 +00:00
|
|
|
|
2018-07-20 08:45:20 +00:00
|
|
|
class SensorConfig:
|
2018-12-01 18:27:21 +00:00
|
|
|
"""Data structure holding sensor configuration."""
|
2018-02-28 22:29:24 +00:00
|
|
|
|
|
|
|
def __init__(self, name, sensor_type):
|
|
|
|
"""Sensor name and type to pass to API."""
|
|
|
|
self.name = name
|
|
|
|
self.sensor_type = sensor_type
|
|
|
|
|
|
|
|
|
|
|
|
# Sensor types/ranges
|
2020-02-28 05:23:47 +00:00
|
|
|
ACTIVE_SENSOR_TYPE = SensorConfig(ACTIVE_NAME, ACTIVE_TYPE)
|
|
|
|
|
|
|
|
# Sensor types/ranges
|
|
|
|
TRENDS_SENSOR_TYPES = {
|
2019-07-31 19:25:30 +00:00
|
|
|
"daily": SensorConfig("Daily", "DAY"),
|
|
|
|
"weekly": SensorConfig("Weekly", "WEEK"),
|
|
|
|
"monthly": SensorConfig("Monthly", "MONTH"),
|
|
|
|
"yearly": SensorConfig("Yearly", "YEAR"),
|
2018-12-01 18:27:21 +00:00
|
|
|
}
|
2018-02-28 22:29:24 +00:00
|
|
|
|
|
|
|
# Production/consumption variants
|
2021-08-30 15:01:26 +00:00
|
|
|
SENSOR_VARIANTS = [(PRODUCTION_ID, PRODUCTION_NAME), (CONSUMPTION_ID, CONSUMPTION_NAME)]
|
|
|
|
|
|
|
|
# Trend production/consumption variants
|
|
|
|
TREND_SENSOR_VARIANTS = SENSOR_VARIANTS + [
|
|
|
|
(PRODUCTION_PCT_ID, PRODUCTION_PCT_NAME),
|
|
|
|
(NET_PRODUCTION_ID, NET_PRODUCTION_NAME),
|
|
|
|
(FROM_GRID_ID, FROM_GRID_NAME),
|
|
|
|
(TO_GRID_ID, TO_GRID_NAME),
|
|
|
|
(SOLAR_POWERED_ID, SOLAR_POWERED_NAME),
|
|
|
|
]
|
2020-02-28 05:23:47 +00:00
|
|
|
|
|
|
|
|
|
|
|
def sense_to_mdi(sense_icon):
|
|
|
|
"""Convert sense icon to mdi icon."""
|
|
|
|
return "mdi:{}".format(MDI_ICONS.get(sense_icon, "power-plug"))
|
2018-02-28 22:29:24 +00:00
|
|
|
|
|
|
|
|
2020-02-25 23:37:41 +00:00
|
|
|
async def async_setup_entry(hass, config_entry, async_add_entities):
|
2018-02-28 22:29:24 +00:00
|
|
|
"""Set up the Sense sensor."""
|
2020-02-25 23:37:41 +00:00
|
|
|
data = hass.data[DOMAIN][config_entry.entry_id][SENSE_DATA]
|
2020-02-28 05:23:47 +00:00
|
|
|
sense_devices_data = hass.data[DOMAIN][config_entry.entry_id][SENSE_DEVICES_DATA]
|
2020-04-14 00:08:37 +00:00
|
|
|
trends_coordinator = hass.data[DOMAIN][config_entry.entry_id][
|
|
|
|
SENSE_TRENDS_COORDINATOR
|
|
|
|
]
|
2018-02-28 22:29:24 +00:00
|
|
|
|
2020-04-14 00:08:37 +00:00
|
|
|
# Request only in case it takes longer
|
|
|
|
# than 60s
|
|
|
|
await trends_coordinator.async_request_refresh()
|
2018-02-28 22:29:24 +00:00
|
|
|
|
2020-02-25 23:37:41 +00:00
|
|
|
sense_monitor_id = data.sense_monitor_id
|
2020-02-28 05:23:47 +00:00
|
|
|
sense_devices = hass.data[DOMAIN][config_entry.entry_id][
|
|
|
|
SENSE_DISCOVERED_DEVICES_DATA
|
|
|
|
]
|
|
|
|
|
|
|
|
devices = [
|
|
|
|
SenseEnergyDevice(sense_devices_data, device, sense_monitor_id)
|
|
|
|
for device in sense_devices
|
|
|
|
if device["tags"]["DeviceListAllowed"] == "true"
|
|
|
|
]
|
|
|
|
|
2021-08-30 15:01:26 +00:00
|
|
|
for variant_id, variant_name in SENSOR_VARIANTS:
|
2020-02-28 05:23:47 +00:00
|
|
|
name = ACTIVE_SENSOR_TYPE.name
|
|
|
|
sensor_type = ACTIVE_SENSOR_TYPE.sensor_type
|
|
|
|
|
2021-08-30 15:01:26 +00:00
|
|
|
unique_id = f"{sense_monitor_id}-active-{variant_id}"
|
2020-02-28 05:23:47 +00:00
|
|
|
devices.append(
|
|
|
|
SenseActiveSensor(
|
2021-08-30 15:01:26 +00:00
|
|
|
data,
|
|
|
|
name,
|
|
|
|
sensor_type,
|
|
|
|
sense_monitor_id,
|
|
|
|
variant_id,
|
|
|
|
variant_name,
|
|
|
|
unique_id,
|
2020-02-28 05:23:47 +00:00
|
|
|
)
|
|
|
|
)
|
2020-02-25 23:37:41 +00:00
|
|
|
|
2020-10-08 16:08:37 +00:00
|
|
|
for i in range(len(data.active_voltage)):
|
|
|
|
devices.append(SenseVoltageSensor(data, i, sense_monitor_id))
|
|
|
|
|
2021-07-15 04:44:57 +00:00
|
|
|
for type_id, typ in TRENDS_SENSOR_TYPES.items():
|
2021-08-30 15:01:26 +00:00
|
|
|
for variant_id, variant_name in TREND_SENSOR_VARIANTS:
|
2018-11-02 09:13:14 +00:00
|
|
|
name = typ.name
|
|
|
|
sensor_type = typ.sensor_type
|
2020-02-25 23:37:41 +00:00
|
|
|
|
2021-08-30 15:01:26 +00:00
|
|
|
unique_id = f"{sense_monitor_id}-{type_id}-{variant_id}"
|
2020-02-25 23:37:41 +00:00
|
|
|
devices.append(
|
2020-02-28 05:23:47 +00:00
|
|
|
SenseTrendsSensor(
|
|
|
|
data,
|
|
|
|
name,
|
|
|
|
sensor_type,
|
2021-08-30 15:01:26 +00:00
|
|
|
variant_id,
|
|
|
|
variant_name,
|
2020-04-14 00:08:37 +00:00
|
|
|
trends_coordinator,
|
2020-02-28 05:23:47 +00:00
|
|
|
unique_id,
|
2020-02-25 23:37:41 +00:00
|
|
|
)
|
|
|
|
)
|
2018-02-28 22:29:24 +00:00
|
|
|
|
2019-03-12 13:44:53 +00:00
|
|
|
async_add_entities(devices)
|
2018-02-28 22:29:24 +00:00
|
|
|
|
|
|
|
|
2021-03-22 18:54:14 +00:00
|
|
|
class SenseActiveSensor(SensorEntity):
|
2020-02-28 05:23:47 +00:00
|
|
|
"""Implementation of a Sense energy sensor."""
|
|
|
|
|
2021-06-23 06:56:11 +00:00
|
|
|
_attr_icon = ICON
|
2021-08-12 15:40:55 +00:00
|
|
|
_attr_native_unit_of_measurement = POWER_WATT
|
2021-06-23 06:56:11 +00:00
|
|
|
_attr_extra_state_attributes = {ATTR_ATTRIBUTION: ATTRIBUTION}
|
|
|
|
_attr_should_poll = False
|
|
|
|
_attr_available = False
|
|
|
|
_attr_state_class = STATE_CLASS_MEASUREMENT
|
|
|
|
|
2020-02-28 05:23:47 +00:00
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
data,
|
|
|
|
name,
|
|
|
|
sensor_type,
|
|
|
|
sense_monitor_id,
|
2021-08-30 15:01:26 +00:00
|
|
|
variant_id,
|
|
|
|
variant_name,
|
2020-02-28 05:23:47 +00:00
|
|
|
unique_id,
|
|
|
|
):
|
|
|
|
"""Initialize the Sense sensor."""
|
2021-08-30 15:01:26 +00:00
|
|
|
self._attr_name = f"{name} {variant_name}"
|
2021-06-23 06:56:11 +00:00
|
|
|
self._attr_unique_id = unique_id
|
2020-02-28 05:23:47 +00:00
|
|
|
self._data = data
|
|
|
|
self._sense_monitor_id = sense_monitor_id
|
|
|
|
self._sensor_type = sensor_type
|
2021-08-30 15:01:26 +00:00
|
|
|
self._variant_id = variant_id
|
|
|
|
self._variant_name = variant_name
|
2020-02-28 05:23:47 +00:00
|
|
|
|
|
|
|
async def async_added_to_hass(self):
|
|
|
|
"""Register callbacks."""
|
2020-04-14 00:08:37 +00:00
|
|
|
self.async_on_remove(
|
|
|
|
async_dispatcher_connect(
|
|
|
|
self.hass,
|
|
|
|
f"{SENSE_DEVICE_UPDATE}-{self._sense_monitor_id}",
|
|
|
|
self._async_update_from_data,
|
|
|
|
)
|
2020-02-28 05:23:47 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _async_update_from_data(self):
|
|
|
|
"""Update the sensor from the data. Must not do I/O."""
|
2020-10-24 21:35:21 +00:00
|
|
|
new_state = round(
|
2020-02-28 05:23:47 +00:00
|
|
|
self._data.active_solar_power
|
2021-08-30 15:01:26 +00:00
|
|
|
if self._variant_id == PRODUCTION_ID
|
2020-02-28 05:23:47 +00:00
|
|
|
else self._data.active_power
|
|
|
|
)
|
2021-08-12 15:40:55 +00:00
|
|
|
if self._attr_available and self._attr_native_value == new_state:
|
2020-10-24 21:35:21 +00:00
|
|
|
return
|
2021-08-12 15:40:55 +00:00
|
|
|
self._attr_native_value = new_state
|
2021-06-23 06:56:11 +00:00
|
|
|
self._attr_available = True
|
2020-02-28 05:23:47 +00:00
|
|
|
self.async_write_ha_state()
|
|
|
|
|
|
|
|
|
2021-03-22 18:54:14 +00:00
|
|
|
class SenseVoltageSensor(SensorEntity):
|
2020-10-08 16:08:37 +00:00
|
|
|
"""Implementation of a Sense energy voltage sensor."""
|
|
|
|
|
2021-08-12 15:40:55 +00:00
|
|
|
_attr_native_unit_of_measurement = ELECTRIC_POTENTIAL_VOLT
|
2021-06-23 06:56:11 +00:00
|
|
|
_attr_extra_state_attributes = {ATTR_ATTRIBUTION: ATTRIBUTION}
|
|
|
|
_attr_icon = ICON
|
|
|
|
_attr_should_poll = False
|
|
|
|
_attr_available = False
|
|
|
|
|
2020-10-08 16:08:37 +00:00
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
data,
|
|
|
|
index,
|
|
|
|
sense_monitor_id,
|
|
|
|
):
|
|
|
|
"""Initialize the Sense sensor."""
|
|
|
|
line_num = index + 1
|
2021-06-23 06:56:11 +00:00
|
|
|
self._attr_name = f"L{line_num} Voltage"
|
|
|
|
self._attr_unique_id = f"{sense_monitor_id}-L{line_num}"
|
2020-10-08 16:08:37 +00:00
|
|
|
self._data = data
|
|
|
|
self._sense_monitor_id = sense_monitor_id
|
|
|
|
self._voltage_index = index
|
|
|
|
|
|
|
|
async def async_added_to_hass(self):
|
|
|
|
"""Register callbacks."""
|
|
|
|
self.async_on_remove(
|
|
|
|
async_dispatcher_connect(
|
|
|
|
self.hass,
|
|
|
|
f"{SENSE_DEVICE_UPDATE}-{self._sense_monitor_id}",
|
|
|
|
self._async_update_from_data,
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _async_update_from_data(self):
|
|
|
|
"""Update the sensor from the data. Must not do I/O."""
|
2020-10-24 21:35:21 +00:00
|
|
|
new_state = round(self._data.active_voltage[self._voltage_index], 1)
|
2021-08-12 15:40:55 +00:00
|
|
|
if self._attr_available and self._attr_native_value == new_state:
|
2020-10-24 21:35:21 +00:00
|
|
|
return
|
2021-06-23 06:56:11 +00:00
|
|
|
self._attr_available = True
|
2021-08-12 15:40:55 +00:00
|
|
|
self._attr_native_value = new_state
|
2020-10-08 16:08:37 +00:00
|
|
|
self.async_write_ha_state()
|
|
|
|
|
|
|
|
|
2021-03-22 18:54:14 +00:00
|
|
|
class SenseTrendsSensor(SensorEntity):
|
2018-02-28 22:29:24 +00:00
|
|
|
"""Implementation of a Sense energy sensor."""
|
|
|
|
|
2021-07-29 04:58:28 +00:00
|
|
|
_attr_device_class = DEVICE_CLASS_ENERGY
|
2021-08-18 14:31:10 +00:00
|
|
|
_attr_state_class = STATE_CLASS_TOTAL_INCREASING
|
2021-08-12 15:40:55 +00:00
|
|
|
_attr_native_unit_of_measurement = ENERGY_KILO_WATT_HOUR
|
2021-06-23 06:56:11 +00:00
|
|
|
_attr_extra_state_attributes = {ATTR_ATTRIBUTION: ATTRIBUTION}
|
|
|
|
_attr_icon = ICON
|
|
|
|
_attr_should_poll = False
|
|
|
|
|
2020-02-25 23:37:41 +00:00
|
|
|
def __init__(
|
2020-08-27 11:56:20 +00:00
|
|
|
self,
|
|
|
|
data,
|
|
|
|
name,
|
|
|
|
sensor_type,
|
2021-08-30 15:01:26 +00:00
|
|
|
variant_id,
|
|
|
|
variant_name,
|
2020-08-27 11:56:20 +00:00
|
|
|
trends_coordinator,
|
|
|
|
unique_id,
|
2020-02-25 23:37:41 +00:00
|
|
|
):
|
2018-12-01 18:27:21 +00:00
|
|
|
"""Initialize the Sense sensor."""
|
2021-08-30 15:01:26 +00:00
|
|
|
self._attr_name = f"{name} {variant_name}"
|
2021-06-23 06:56:11 +00:00
|
|
|
self._attr_unique_id = unique_id
|
2018-02-28 22:29:24 +00:00
|
|
|
self._data = data
|
|
|
|
self._sensor_type = sensor_type
|
2020-04-14 00:08:37 +00:00
|
|
|
self._coordinator = trends_coordinator
|
2021-08-30 15:01:26 +00:00
|
|
|
self._variant_id = variant_id
|
2020-04-14 00:08:37 +00:00
|
|
|
self._had_any_update = False
|
2018-02-28 22:29:24 +00:00
|
|
|
|
2021-08-30 15:01:26 +00:00
|
|
|
if variant_id in [PRODUCTION_PCT_ID, SOLAR_POWERED_ID]:
|
|
|
|
self._attr_native_unit_of_measurement = PERCENTAGE
|
|
|
|
self._attr_entity_registry_enabled_default = False
|
|
|
|
self._attr_state_class = None
|
|
|
|
self._attr_device_class = None
|
|
|
|
|
2018-02-28 22:29:24 +00:00
|
|
|
@property
|
2021-08-12 15:40:55 +00:00
|
|
|
def native_value(self):
|
2018-02-28 22:29:24 +00:00
|
|
|
"""Return the state of the sensor."""
|
2021-08-30 15:01:26 +00:00
|
|
|
return round(self._data.get_trend(self._sensor_type, self._variant_id), 1)
|
2018-02-28 22:29:24 +00:00
|
|
|
|
2020-02-25 23:37:41 +00:00
|
|
|
@property
|
|
|
|
def available(self):
|
2020-04-14 00:08:37 +00:00
|
|
|
"""Return if entity is available."""
|
|
|
|
return self._had_any_update and self._coordinator.last_update_success
|
2020-02-25 23:37:41 +00:00
|
|
|
|
2020-04-14 00:08:37 +00:00
|
|
|
@callback
|
|
|
|
def _async_update(self):
|
|
|
|
"""Track if we had an update so we do not report zero data."""
|
|
|
|
self._had_any_update = True
|
|
|
|
self.async_write_ha_state()
|
|
|
|
|
2019-03-12 13:44:53 +00:00
|
|
|
async def async_update(self):
|
2020-04-14 00:08:37 +00:00
|
|
|
"""Update the entity.
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2020-04-14 00:08:37 +00:00
|
|
|
Only used by the generic entity update service.
|
|
|
|
"""
|
|
|
|
await self._coordinator.async_request_refresh()
|
2018-02-28 22:29:24 +00:00
|
|
|
|
2020-04-14 00:08:37 +00:00
|
|
|
async def async_added_to_hass(self):
|
|
|
|
"""When entity is added to hass."""
|
|
|
|
self.async_on_remove(self._coordinator.async_add_listener(self._async_update))
|
2020-02-28 05:23:47 +00:00
|
|
|
|
|
|
|
|
2021-03-22 18:54:14 +00:00
|
|
|
class SenseEnergyDevice(SensorEntity):
|
2020-02-28 05:23:47 +00:00
|
|
|
"""Implementation of a Sense energy device."""
|
|
|
|
|
2021-06-23 06:56:11 +00:00
|
|
|
_attr_available = False
|
|
|
|
_attr_state_class = STATE_CLASS_MEASUREMENT
|
2021-08-12 15:40:55 +00:00
|
|
|
_attr_native_unit_of_measurement = POWER_WATT
|
2021-06-23 06:56:11 +00:00
|
|
|
_attr_extra_state_attributes = {ATTR_ATTRIBUTION: ATTRIBUTION}
|
|
|
|
_attr_device_class = DEVICE_CLASS_POWER
|
|
|
|
_attr_should_poll = False
|
|
|
|
|
2020-02-28 05:23:47 +00:00
|
|
|
def __init__(self, sense_devices_data, device, sense_monitor_id):
|
|
|
|
"""Initialize the Sense binary sensor."""
|
2021-06-23 06:56:11 +00:00
|
|
|
self._attr_name = f"{device['name']} {CONSUMPTION_NAME}"
|
2020-02-28 05:23:47 +00:00
|
|
|
self._id = device["id"]
|
|
|
|
self._sense_monitor_id = sense_monitor_id
|
2021-06-23 06:56:11 +00:00
|
|
|
self._attr_unique_id = f"{sense_monitor_id}-{self._id}-{CONSUMPTION_ID}"
|
|
|
|
self._attr_icon = sense_to_mdi(device["icon"])
|
2020-02-28 05:23:47 +00:00
|
|
|
self._sense_devices_data = sense_devices_data
|
|
|
|
|
|
|
|
async def async_added_to_hass(self):
|
|
|
|
"""Register callbacks."""
|
2020-04-14 00:08:37 +00:00
|
|
|
self.async_on_remove(
|
|
|
|
async_dispatcher_connect(
|
|
|
|
self.hass,
|
|
|
|
f"{SENSE_DEVICE_UPDATE}-{self._sense_monitor_id}",
|
|
|
|
self._async_update_from_data,
|
|
|
|
)
|
2020-02-28 05:23:47 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _async_update_from_data(self):
|
|
|
|
"""Get the latest data, update state. Must not do I/O."""
|
|
|
|
device_data = self._sense_devices_data.get_device_by_id(self._id)
|
|
|
|
if not device_data or "w" not in device_data:
|
2020-10-24 21:35:21 +00:00
|
|
|
new_state = 0
|
2020-02-28 05:23:47 +00:00
|
|
|
else:
|
2020-10-24 21:35:21 +00:00
|
|
|
new_state = int(device_data["w"])
|
2021-08-12 15:40:55 +00:00
|
|
|
if self._attr_available and self._attr_native_value == new_state:
|
2020-10-24 21:35:21 +00:00
|
|
|
return
|
2021-08-12 15:40:55 +00:00
|
|
|
self._attr_native_value = new_state
|
2021-06-23 06:56:11 +00:00
|
|
|
self._attr_available = True
|
2020-02-28 05:23:47 +00:00
|
|
|
self.async_write_ha_state()
|