Home Assistant Unofficial Reference 2024.12.1
sensor.py
Go to the documentation of this file.
1 """Utility meter from sensors providing raw data."""
2 
3 from __future__ import annotations
4 
5 from collections.abc import Mapping
6 from dataclasses import dataclass
7 from datetime import datetime, timedelta
8 from decimal import Decimal, DecimalException, InvalidOperation
9 import logging
10 from typing import Any, Self
11 
12 from cronsim import CronSim
13 import voluptuous as vol
14 
16  ATTR_LAST_RESET,
17  DEVICE_CLASS_UNITS,
18  RestoreSensor,
19  SensorDeviceClass,
20  SensorExtraStoredData,
21  SensorStateClass,
22 )
23 from homeassistant.components.sensor.recorder import _suggest_report_issue
24 from homeassistant.config_entries import ConfigEntry
25 from homeassistant.const import (
26  ATTR_DEVICE_CLASS,
27  ATTR_UNIT_OF_MEASUREMENT,
28  CONF_NAME,
29  CONF_UNIQUE_ID,
30  EVENT_CORE_CONFIG_UPDATE,
31  STATE_UNAVAILABLE,
32  STATE_UNKNOWN,
33 )
34 from homeassistant.core import (
35  Event,
36  EventStateChangedData,
37  HomeAssistant,
38  State,
39  callback,
40 )
41 from homeassistant.helpers import entity_platform, entity_registry as er
42 from homeassistant.helpers.device import async_device_info_to_link_from_entity
43 from homeassistant.helpers.dispatcher import async_dispatcher_connect
44 from homeassistant.helpers.entity_platform import AddEntitiesCallback
45 from homeassistant.helpers.event import (
46  async_track_point_in_time,
47  async_track_state_change_event,
48 )
49 from homeassistant.helpers.start import async_at_started
50 from homeassistant.helpers.template import is_number
51 from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
52 from homeassistant.util import slugify
53 import homeassistant.util.dt as dt_util
54 from homeassistant.util.enum import try_parse_enum
55 
56 from .const import (
57  ATTR_NEXT_RESET,
58  ATTR_VALUE,
59  BIMONTHLY,
60  CONF_CRON_PATTERN,
61  CONF_METER,
62  CONF_METER_DELTA_VALUES,
63  CONF_METER_NET_CONSUMPTION,
64  CONF_METER_OFFSET,
65  CONF_METER_PERIODICALLY_RESETTING,
66  CONF_METER_TYPE,
67  CONF_SENSOR_ALWAYS_AVAILABLE,
68  CONF_SOURCE_SENSOR,
69  CONF_TARIFF,
70  CONF_TARIFF_ENTITY,
71  CONF_TARIFFS,
72  DAILY,
73  DATA_TARIFF_SENSORS,
74  DATA_UTILITY,
75  HOURLY,
76  MONTHLY,
77  QUARTER_HOURLY,
78  QUARTERLY,
79  SERVICE_CALIBRATE_METER,
80  SIGNAL_RESET_METER,
81  WEEKLY,
82  YEARLY,
83 )
84 
85 PERIOD2CRON = {
86  QUARTER_HOURLY: "{minute}/15 * * * *",
87  HOURLY: "{minute} * * * *",
88  DAILY: "{minute} {hour} * * *",
89  WEEKLY: "{minute} {hour} * * {day}",
90  MONTHLY: "{minute} {hour} {day} * *",
91  BIMONTHLY: "{minute} {hour} {day} */2 *",
92  QUARTERLY: "{minute} {hour} {day} */3 *",
93  YEARLY: "{minute} {hour} {day} 1/12 *",
94 }
95 
96 _LOGGER = logging.getLogger(__name__)
97 
98 ATTR_SOURCE_ID = "source"
99 ATTR_STATUS = "status"
100 ATTR_PERIOD = "meter_period"
101 ATTR_LAST_PERIOD = "last_period"
102 ATTR_LAST_VALID_STATE = "last_valid_state"
103 ATTR_TARIFF = "tariff"
104 
105 PRECISION = 3
106 PAUSED = "paused"
107 COLLECTING = "collecting"
108 
109 
111  """Validate value is a number."""
112  if is_number(value):
113  return value
114  raise vol.Invalid("Value is not a number")
115 
116 
118  hass: HomeAssistant,
119  config_entry: ConfigEntry,
120  async_add_entities: AddEntitiesCallback,
121 ) -> None:
122  """Initialize Utility Meter config entry."""
123  entry_id = config_entry.entry_id
124  registry = er.async_get(hass)
125  # Validate + resolve entity registry id to entity_id
126  source_entity_id = er.async_validate_entity_id(
127  registry, config_entry.options[CONF_SOURCE_SENSOR]
128  )
129 
131  hass,
132  source_entity_id,
133  )
134 
135  cron_pattern = None
136  delta_values = config_entry.options[CONF_METER_DELTA_VALUES]
137  meter_offset = timedelta(days=config_entry.options[CONF_METER_OFFSET])
138  meter_type = config_entry.options[CONF_METER_TYPE]
139  if meter_type == "none":
140  meter_type = None
141  name = config_entry.title
142  net_consumption = config_entry.options[CONF_METER_NET_CONSUMPTION]
143  periodically_resetting = config_entry.options[CONF_METER_PERIODICALLY_RESETTING]
144  tariff_entity = hass.data[DATA_UTILITY][entry_id][CONF_TARIFF_ENTITY]
145  sensor_always_available = config_entry.options.get(
146  CONF_SENSOR_ALWAYS_AVAILABLE, False
147  )
148 
149  meters = []
150  tariffs = config_entry.options[CONF_TARIFFS]
151 
152  if not tariffs:
153  # Add single sensor, not gated by a tariff selector
154  meter_sensor = UtilityMeterSensor(
155  cron_pattern=cron_pattern,
156  delta_values=delta_values,
157  meter_offset=meter_offset,
158  meter_type=meter_type,
159  name=name,
160  net_consumption=net_consumption,
161  parent_meter=entry_id,
162  periodically_resetting=periodically_resetting,
163  source_entity=source_entity_id,
164  tariff_entity=tariff_entity,
165  tariff=None,
166  unique_id=entry_id,
167  device_info=device_info,
168  sensor_always_available=sensor_always_available,
169  )
170  meters.append(meter_sensor)
171  hass.data[DATA_UTILITY][entry_id][DATA_TARIFF_SENSORS].append(meter_sensor)
172  else:
173  # Add sensors for each tariff
174  for tariff in tariffs:
175  meter_sensor = UtilityMeterSensor(
176  cron_pattern=cron_pattern,
177  delta_values=delta_values,
178  meter_offset=meter_offset,
179  meter_type=meter_type,
180  name=f"{name} {tariff}",
181  net_consumption=net_consumption,
182  parent_meter=entry_id,
183  periodically_resetting=periodically_resetting,
184  source_entity=source_entity_id,
185  tariff_entity=tariff_entity,
186  tariff=tariff,
187  unique_id=f"{entry_id}_{tariff}",
188  device_info=device_info,
189  sensor_always_available=sensor_always_available,
190  )
191  meters.append(meter_sensor)
192  hass.data[DATA_UTILITY][entry_id][DATA_TARIFF_SENSORS].append(meter_sensor)
193 
194  async_add_entities(meters)
195 
196  platform = entity_platform.async_get_current_platform()
197 
198  platform.async_register_entity_service(
199  SERVICE_CALIBRATE_METER,
200  {vol.Required(ATTR_VALUE): validate_is_number},
201  "async_calibrate",
202  )
203 
204 
206  hass: HomeAssistant,
207  config: ConfigType,
208  async_add_entities: AddEntitiesCallback,
209  discovery_info: DiscoveryInfoType | None = None,
210 ) -> None:
211  """Set up the utility meter sensor."""
212  if discovery_info is None:
213  _LOGGER.error(
214  "This platform is not available to configure "
215  "from 'sensor:' in configuration.yaml"
216  )
217  return
218 
219  meters = []
220  for conf in discovery_info.values():
221  meter = conf[CONF_METER]
222  conf_meter_source = hass.data[DATA_UTILITY][meter][CONF_SOURCE_SENSOR]
223  conf_meter_unique_id = hass.data[DATA_UTILITY][meter].get(CONF_UNIQUE_ID)
224  conf_sensor_tariff = conf.get(CONF_TARIFF, "single_tariff")
225  conf_sensor_unique_id = (
226  f"{conf_meter_unique_id}_{conf_sensor_tariff}"
227  if conf_meter_unique_id
228  else None
229  )
230  conf_meter_name = hass.data[DATA_UTILITY][meter].get(CONF_NAME, meter)
231  conf_sensor_tariff = conf.get(CONF_TARIFF)
232 
233  suggested_entity_id = None
234  if conf_sensor_tariff:
235  conf_sensor_name = f"{conf_meter_name} {conf_sensor_tariff}"
236  slug = slugify(f"{meter} {conf_sensor_tariff}")
237  suggested_entity_id = f"sensor.{slug}"
238  else:
239  conf_sensor_name = conf_meter_name
240 
241  conf_meter_type = hass.data[DATA_UTILITY][meter].get(CONF_METER_TYPE)
242  conf_meter_offset = hass.data[DATA_UTILITY][meter][CONF_METER_OFFSET]
243  conf_meter_delta_values = hass.data[DATA_UTILITY][meter][
244  CONF_METER_DELTA_VALUES
245  ]
246  conf_meter_net_consumption = hass.data[DATA_UTILITY][meter][
247  CONF_METER_NET_CONSUMPTION
248  ]
249  conf_meter_periodically_resetting = hass.data[DATA_UTILITY][meter][
250  CONF_METER_PERIODICALLY_RESETTING
251  ]
252  conf_meter_tariff_entity = hass.data[DATA_UTILITY][meter].get(
253  CONF_TARIFF_ENTITY
254  )
255  conf_cron_pattern = hass.data[DATA_UTILITY][meter].get(CONF_CRON_PATTERN)
256  conf_sensor_always_available = hass.data[DATA_UTILITY][meter][
257  CONF_SENSOR_ALWAYS_AVAILABLE
258  ]
259  meter_sensor = UtilityMeterSensor(
260  cron_pattern=conf_cron_pattern,
261  delta_values=conf_meter_delta_values,
262  meter_offset=conf_meter_offset,
263  meter_type=conf_meter_type,
264  name=conf_sensor_name,
265  net_consumption=conf_meter_net_consumption,
266  parent_meter=meter,
267  periodically_resetting=conf_meter_periodically_resetting,
268  source_entity=conf_meter_source,
269  tariff_entity=conf_meter_tariff_entity,
270  tariff=conf_sensor_tariff,
271  unique_id=conf_sensor_unique_id,
272  suggested_entity_id=suggested_entity_id,
273  sensor_always_available=conf_sensor_always_available,
274  )
275  meters.append(meter_sensor)
276 
277  hass.data[DATA_UTILITY][meter][DATA_TARIFF_SENSORS].append(meter_sensor)
278 
279  async_add_entities(meters)
280 
281  platform = entity_platform.async_get_current_platform()
282 
283  platform.async_register_entity_service(
284  SERVICE_CALIBRATE_METER,
285  {vol.Required(ATTR_VALUE): validate_is_number},
286  "async_calibrate",
287  )
288 
289 
290 @dataclass
292  """Object to hold extra stored data."""
293 
294  last_period: Decimal
295  last_reset: datetime | None
296  last_valid_state: Decimal | None
297  status: str
298  input_device_class: SensorDeviceClass | None
299 
300  def as_dict(self) -> dict[str, Any]:
301  """Return a dict representation of the utility sensor data."""
302  data = super().as_dict()
303  data["last_period"] = str(self.last_period)
304  if isinstance(self.last_reset, (datetime)):
305  data["last_reset"] = self.last_reset.isoformat()
306  data["last_valid_state"] = (
307  str(self.last_valid_state) if self.last_valid_state else None
308  )
309  data["status"] = self.status
310  data["input_device_class"] = str(self.input_device_class)
311 
312  return data
313 
314  @classmethod
315  def from_dict(cls, restored: dict[str, Any]) -> Self | None:
316  """Initialize a stored sensor state from a dict."""
317  extra = SensorExtraStoredData.from_dict(restored)
318  if extra is None:
319  return None
320 
321  try:
322  last_period: Decimal = Decimal(restored["last_period"])
323  last_reset: datetime | None = dt_util.parse_datetime(restored["last_reset"])
324  last_valid_state: Decimal | None = (
325  Decimal(restored["last_valid_state"])
326  if restored.get("last_valid_state")
327  else None
328  )
329  status: str = restored["status"]
330  input_device_class = try_parse_enum(
331  SensorDeviceClass, restored.get("input_device_class")
332  )
333  except KeyError:
334  # restored is a dict, but does not have all values
335  return None
336  except InvalidOperation:
337  # last_period is corrupted
338  return None
339 
340  return cls(
341  extra.native_value,
342  extra.native_unit_of_measurement,
343  last_period,
344  last_reset,
345  last_valid_state,
346  status,
347  input_device_class,
348  )
349 
350 
352  """Representation of an utility meter sensor."""
353 
354  _attr_translation_key = "utility_meter"
355  _attr_should_poll = False
356  _unrecorded_attributes = frozenset({ATTR_NEXT_RESET})
357 
358  def __init__(
359  self,
360  *,
361  cron_pattern,
362  delta_values,
363  meter_offset,
364  meter_type,
365  name,
366  net_consumption,
367  parent_meter,
368  periodically_resetting,
369  source_entity,
370  tariff_entity,
371  tariff,
372  unique_id,
373  sensor_always_available,
374  suggested_entity_id=None,
375  device_info=None,
376  ):
377  """Initialize the Utility Meter sensor."""
378  self._attr_unique_id_attr_unique_id = unique_id
379  self._attr_device_info_attr_device_info = device_info
380  self.entity_identity_identity_id = suggested_entity_id
381  self._parent_meter_parent_meter = parent_meter
382  self._sensor_source_id_sensor_source_id = source_entity
383  self._last_period_last_period = Decimal(0)
384  self._last_reset_last_reset = dt_util.utcnow()
385  self._last_valid_state_last_valid_state = None
386  self._collecting_collecting = None
387  self._attr_name_attr_name = name
388  self._input_device_class_input_device_class = None
389  self._attr_native_unit_of_measurement_attr_native_unit_of_measurement = None
390  self._period_period = meter_type
391  if meter_type is not None:
392  # For backwards compatibility reasons we convert the period and offset into a cron pattern
393  self._cron_pattern_cron_pattern = PERIOD2CRON[meter_type].format(
394  minute=meter_offset.seconds % 3600 // 60,
395  hour=meter_offset.seconds // 3600,
396  day=meter_offset.days + 1,
397  )
398  _LOGGER.debug("CRON pattern: %s", self._cron_pattern_cron_pattern)
399  else:
400  self._cron_pattern_cron_pattern = cron_pattern
401  self._sensor_always_available_sensor_always_available = sensor_always_available
402  self._sensor_delta_values_sensor_delta_values = delta_values
403  self._sensor_net_consumption_sensor_net_consumption = net_consumption
404  self._sensor_periodically_resetting_sensor_periodically_resetting = periodically_resetting
405  self._tariff_tariff = tariff
406  self._tariff_entity_tariff_entity = tariff_entity
407  self._next_reset_next_reset = None
408  self._current_tz_current_tz = None
409  self._config_scheduler_config_scheduler()
410 
411  def _config_scheduler(self):
412  self.schedulerscheduler = (
413  CronSim(
414  self._cron_pattern_cron_pattern,
415  dt_util.now(
416  dt_util.get_default_time_zone()
417  ), # we need timezone for DST purposes (see issue #102984)
418  )
419  if self._cron_pattern_cron_pattern
420  else None
421  )
422 
423  def start(self, attributes: Mapping[str, Any]) -> None:
424  """Initialize unit and state upon source initial update."""
425  self._input_device_class_input_device_class = attributes.get(ATTR_DEVICE_CLASS)
426  self._attr_native_unit_of_measurement_attr_native_unit_of_measurement = attributes.get(ATTR_UNIT_OF_MEASUREMENT)
427  self._attr_native_value_attr_native_value = 0
428  self.async_write_ha_stateasync_write_ha_state()
429 
430  @staticmethod
431  def _validate_state(state: State | None) -> Decimal | None:
432  """Parse the state as a Decimal if available. Throws DecimalException if the state is not a number."""
433  try:
434  return (
435  None
436  if state is None or state.state in [STATE_UNAVAILABLE, STATE_UNKNOWN]
437  else Decimal(state.state)
438  )
439  except DecimalException:
440  return None
441 
443  self, old_state: State | None, new_state: State
444  ) -> Decimal | None:
445  """Calculate the adjustment based on the old and new state."""
446 
447  # First check if the new_state is valid (see discussion in PR #88446)
448  if (new_state_val := self._validate_state_validate_state(new_state)) is None:
449  _LOGGER.warning("Invalid state %s", new_state.state)
450  return None
451 
452  if self._sensor_delta_values_sensor_delta_values:
453  return new_state_val
454 
455  if (
456  not self._sensor_periodically_resetting_sensor_periodically_resetting
457  and self._last_valid_state_last_valid_state is not None
458  ): # Fallback to old_state if sensor is periodically resetting but last_valid_state is None
459  return new_state_val - self._last_valid_state_last_valid_state
460 
461  if (old_state_val := self._validate_state_validate_state(old_state)) is not None:
462  return new_state_val - old_state_val
463 
464  _LOGGER.debug(
465  "%s received an invalid state change coming from %s (%s > %s)",
466  self.namename,
467  self._sensor_source_id_sensor_source_id,
468  old_state.state if old_state else None,
469  new_state_val,
470  )
471  return None
472 
473  @callback
474  def async_reading(self, event: Event[EventStateChangedData]) -> None:
475  """Handle the sensor state changes."""
476  if (
477  source_state := self.hasshass.states.get(self._sensor_source_id_sensor_source_id)
478  ) is None or source_state.state == STATE_UNAVAILABLE:
479  if not self._sensor_always_available_sensor_always_available:
480  self._attr_available_attr_available = False
481  self.async_write_ha_stateasync_write_ha_state()
482  return
483 
484  self._attr_available_attr_available = True
485 
486  old_state = event.data["old_state"]
487  new_state = event.data["new_state"]
488  if new_state is None:
489  return
490  new_state_attributes: Mapping[str, Any] = new_state.attributes or {}
491 
492  # First check if the new_state is valid (see discussion in PR #88446)
493  if (new_state_val := self._validate_state_validate_state(new_state)) is None:
494  _LOGGER.warning(
495  "%s received an invalid new state from %s : %s",
496  self.namename,
497  self._sensor_source_id_sensor_source_id,
498  new_state.state,
499  )
500  return
501 
502  if self.native_valuenative_value is None:
503  # First state update initializes the utility_meter sensors
504  for sensor in self.hasshass.data[DATA_UTILITY][self._parent_meter_parent_meter][
505  DATA_TARIFF_SENSORS
506  ]:
507  sensor.start(new_state_attributes)
508  if self.native_unit_of_measurementnative_unit_of_measurement is None:
509  _LOGGER.warning(
510  "Source sensor %s has no unit of measurement. Please %s",
511  self._sensor_source_id_sensor_source_id,
512  _suggest_report_issue(self.hasshass, self._sensor_source_id_sensor_source_id),
513  )
514 
515  if (
516  adjustment := self.calculate_adjustmentcalculate_adjustment(old_state, new_state)
517  ) is not None and (self._sensor_net_consumption_sensor_net_consumption or adjustment >= 0):
518  # If net_consumption is off, the adjustment must be non-negative
519  self._attr_native_value_attr_native_value += adjustment # type: ignore[operator] # self._attr_native_value will be set to by the start function if it is None, therefore it always has a valid Decimal value at this line
520 
521  self._input_device_class_input_device_class = new_state_attributes.get(ATTR_DEVICE_CLASS)
522  self._attr_native_unit_of_measurement_attr_native_unit_of_measurement = new_state_attributes.get(
523  ATTR_UNIT_OF_MEASUREMENT
524  )
525  self._last_valid_state_last_valid_state = new_state_val
526  self.async_write_ha_stateasync_write_ha_state()
527 
528  @callback
529  def async_tariff_change(self, event: Event[EventStateChangedData]) -> None:
530  """Handle tariff changes."""
531  if (new_state := event.data["new_state"]) is None:
532  return
533 
534  self._change_status_change_status(new_state.state)
535 
536  def _change_status(self, tariff: str) -> None:
537  if self._tariff_tariff == tariff:
539  self.hasshass, [self._sensor_source_id_sensor_source_id], self.async_readingasync_reading
540  )
541  else:
542  if self._collecting_collecting:
543  self._collecting_collecting()
544  self._collecting_collecting = None
545 
546  # Reset the last_valid_state during state change because if the last state before the tariff change was invalid,
547  # there is no way to know how much "adjustment" counts for which tariff. Therefore, we set the last_valid_state
548  # to None and let the fallback mechanism handle the case that the old state was valid
549  self._last_valid_state_last_valid_state = None
550 
551  _LOGGER.debug(
552  "%s - %s - source <%s>",
553  self.namename,
554  COLLECTING if self._collecting_collecting is not None else PAUSED,
555  self._sensor_source_id_sensor_source_id,
556  )
557 
558  self.async_write_ha_stateasync_write_ha_state()
559 
560  async def _program_reset(self):
561  """Program the reset of the utility meter."""
562  if self.schedulerscheduler:
563  self._next_reset_next_reset = next(self.schedulerscheduler)
564 
565  _LOGGER.debug("Next reset of %s is %s", self.entity_identity_identity_id, self._next_reset_next_reset)
566  self.async_on_removeasync_on_remove(
568  self.hasshass,
569  self._async_reset_meter_async_reset_meter,
570  self._next_reset_next_reset,
571  )
572  )
573  self.async_write_ha_stateasync_write_ha_state()
574 
575  async def _async_reset_meter(self, event):
576  """Reset the utility meter status."""
577 
578  await self._program_reset_program_reset()
579 
580  await self.async_reset_meterasync_reset_meter(self._tariff_entity_tariff_entity)
581 
582  async def async_reset_meter(self, entity_id):
583  """Reset meter."""
584  if self._tariff_entity_tariff_entity is not None and self._tariff_entity_tariff_entity != entity_id:
585  return
586  if (
587  self._tariff_entity_tariff_entity is None
588  and entity_id is not None
589  and self.entity_identity_identity_id != entity_id
590  ):
591  return
592  _LOGGER.debug("Reset utility meter <%s>", self.entity_identity_identity_id)
593  self._last_reset_last_reset = dt_util.utcnow()
594  self._last_period_last_period = (
595  Decimal(self.native_valuenative_value) if self.native_valuenative_value else Decimal(0)
596  )
597  self._attr_native_value_attr_native_value = 0
598  self.async_write_ha_stateasync_write_ha_state()
599 
600  async def async_calibrate(self, value):
601  """Calibrate the Utility Meter with a given value."""
602  _LOGGER.debug("Calibrate %s = %s type(%s)", self.namename, value, type(value))
603  self._attr_native_value_attr_native_value = Decimal(str(value))
604  self.async_write_ha_stateasync_write_ha_state()
605 
606  async def async_added_to_hass(self):
607  """Handle entity which will be added."""
608  await super().async_added_to_hass()
609 
610  # track current timezone in case it changes
611  # and we need to reconfigure the scheduler
612  self._current_tz_current_tz = self.hasshass.config.time_zone
613 
614  await self._program_reset_program_reset()
615 
616  self.async_on_removeasync_on_remove(
618  self.hasshass, SIGNAL_RESET_METER, self.async_reset_meterasync_reset_meter
619  )
620  )
621 
622  if (last_sensor_data := await self.async_get_last_sensor_dataasync_get_last_sensor_dataasync_get_last_sensor_data()) is not None:
623  self._attr_native_value_attr_native_value = last_sensor_data.native_value
624  self._input_device_class_input_device_class = last_sensor_data.input_device_class
625  self._attr_native_unit_of_measurement_attr_native_unit_of_measurement = (
626  last_sensor_data.native_unit_of_measurement
627  )
628  self._last_period_last_period = last_sensor_data.last_period
629  self._last_reset_last_reset = last_sensor_data.last_reset
630  self._last_valid_state_last_valid_state = last_sensor_data.last_valid_state
631  if last_sensor_data.status == COLLECTING:
632  # Null lambda to allow cancelling the collection on tariff change
633  self._collecting_collecting = lambda: None
634 
635  @callback
636  def async_source_tracking(event):
637  """Wait for source to be ready, then start meter."""
638  if self._tariff_entity_tariff_entity is not None:
639  _LOGGER.debug(
640  "<%s> tracks utility meter %s", self.namename, self._tariff_entity_tariff_entity
641  )
642  self.async_on_removeasync_on_remove(
644  self.hasshass, [self._tariff_entity_tariff_entity], self.async_tariff_changeasync_tariff_change
645  )
646  )
647 
648  tariff_entity_state = self.hasshass.states.get(self._tariff_entity_tariff_entity)
649  if not tariff_entity_state:
650  # The utility meter is not yet added
651  return
652 
653  self._change_status_change_status(tariff_entity_state.state)
654  return
655 
656  _LOGGER.debug(
657  "<%s> collecting %s from %s",
658  self.namename,
659  self.native_unit_of_measurementnative_unit_of_measurement,
660  self._sensor_source_id_sensor_source_id,
661  )
663  self.hasshass, [self._sensor_source_id_sensor_source_id], self.async_readingasync_reading
664  )
665 
666  self.async_on_removeasync_on_remove(async_at_started(self.hasshass, async_source_tracking))
667 
668  async def async_track_time_zone(event):
669  """Reconfigure Scheduler after time zone changes."""
670 
671  if self._current_tz_current_tz != self.hasshass.config.time_zone:
672  self._current_tz_current_tz = self.hasshass.config.time_zone
673 
674  self._config_scheduler_config_scheduler()
675  await self._program_reset_program_reset()
676 
677  self.async_on_removeasync_on_remove(
678  self.hasshass.bus.async_listen(EVENT_CORE_CONFIG_UPDATE, async_track_time_zone)
679  )
680 
681  async def async_will_remove_from_hass(self) -> None:
682  """Run when entity will be removed from hass."""
683  if self._collecting_collecting:
684  self._collecting_collecting()
685  self._collecting_collecting = None
686 
687  @property
688  def device_class(self):
689  """Return the device class of the sensor."""
690  if self._input_device_class_input_device_class is not None:
691  return self._input_device_class_input_device_class
692  if (
693  self.native_unit_of_measurementnative_unit_of_measurement
694  in DEVICE_CLASS_UNITS[SensorDeviceClass.ENERGY]
695  ):
696  return SensorDeviceClass.ENERGY
697  return None
698 
699  @property
700  def state_class(self):
701  """Return the device class of the sensor."""
702  return (
703  SensorStateClass.TOTAL
704  if self._sensor_net_consumption_sensor_net_consumption
705  else SensorStateClass.TOTAL_INCREASING
706  )
707 
708  @property
710  """Return the state attributes of the sensor."""
711  state_attr = {
712  ATTR_STATUS: PAUSED if self._collecting_collecting is None else COLLECTING,
713  ATTR_LAST_PERIOD: str(self._last_period_last_period),
714  ATTR_LAST_VALID_STATE: str(self._last_valid_state_last_valid_state),
715  }
716  if self._tariff_tariff is not None:
717  state_attr[ATTR_TARIFF] = self._tariff_tariff
718  # last_reset in utility meter was used before last_reset was added for long term
719  # statistics in base sensor. base sensor only supports last reset
720  # sensors with state_class set to total.
721  # To avoid a breaking change we set last_reset directly
722  # in extra state attributes.
723  if last_reset := self._last_reset_last_reset:
724  state_attr[ATTR_LAST_RESET] = last_reset.isoformat()
725  if self._next_reset_next_reset is not None:
726  state_attr[ATTR_NEXT_RESET] = self._next_reset_next_reset.isoformat()
727 
728  return state_attr
729 
730  @property
731  def extra_restore_state_data(self) -> UtilitySensorExtraStoredData:
732  """Return sensor specific state data to be restored."""
734  self.native_valuenative_value,
735  self.native_unit_of_measurementnative_unit_of_measurement,
736  self._last_period_last_period,
737  self._last_reset_last_reset,
738  self._last_valid_state_last_valid_state,
739  PAUSED if self._collecting_collecting is None else COLLECTING,
740  self._input_device_class_input_device_class,
741  )
742 
743  async def async_get_last_sensor_data(self) -> UtilitySensorExtraStoredData | None:
744  """Restore Utility Meter Sensor Extra Stored Data."""
745  if (restored_last_extra_data := await self.async_get_last_extra_data()) is None:
746  return None
747 
748  return UtilitySensorExtraStoredData.from_dict(
749  restored_last_extra_data.as_dict()
750  )
SensorExtraStoredData|None async_get_last_sensor_data(self)
Definition: __init__.py:934
StateType|date|datetime|Decimal native_value(self)
Definition: __init__.py:460
UtilitySensorExtraStoredData extra_restore_state_data(self)
Definition: sensor.py:731
None start(self, Mapping[str, Any] attributes)
Definition: sensor.py:423
None async_tariff_change(self, Event[EventStateChangedData] event)
Definition: sensor.py:529
None async_reading(self, Event[EventStateChangedData] event)
Definition: sensor.py:474
UtilitySensorExtraStoredData|None async_get_last_sensor_data(self)
Definition: sensor.py:743
def __init__(self, *cron_pattern, delta_values, meter_offset, meter_type, name, net_consumption, parent_meter, periodically_resetting, source_entity, tariff_entity, tariff, unique_id, sensor_always_available, suggested_entity_id=None, device_info=None)
Definition: sensor.py:376
Decimal|None calculate_adjustment(self, State|None old_state, State new_state)
Definition: sensor.py:444
None async_on_remove(self, CALLBACK_TYPE func)
Definition: entity.py:1331
str|UndefinedType|None name(self)
Definition: entity.py:738
web.Response get(self, web.Request request, str config_key)
Definition: view.py:88
None async_setup_entry(HomeAssistant hass, ConfigEntry config_entry, AddEntitiesCallback async_add_entities)
Definition: sensor.py:121
None async_setup_platform(HomeAssistant hass, ConfigType config, AddEntitiesCallback async_add_entities, DiscoveryInfoType|None discovery_info=None)
Definition: sensor.py:210
dr.DeviceInfo|None async_device_info_to_link_from_entity(HomeAssistant hass, str entity_id_or_uuid)
Definition: device.py:28
Callable[[], None] async_dispatcher_connect(HomeAssistant hass, str signal, Callable[..., Any] target)
Definition: dispatcher.py:103
CALLBACK_TYPE async_track_state_change_event(HomeAssistant hass, str|Iterable[str] entity_ids, Callable[[Event[EventStateChangedData]], Any] action, HassJobType|None job_type=None)
Definition: event.py:314
CALLBACK_TYPE async_track_point_in_time(HomeAssistant hass, HassJob[[datetime], Coroutine[Any, Any, None]|None]|Callable[[datetime], Coroutine[Any, Any, None]|None] action, datetime point_in_time)
Definition: event.py:1462
CALLBACK_TYPE async_at_started(HomeAssistant hass, Callable[[HomeAssistant], Coroutine[Any, Any, None]|None] at_start_cb)
Definition: start.py:80