Home Assistant Unofficial Reference 2024.12.1
sensor.py
Go to the documentation of this file.
1 """Feed Entity Manager Sensor support for GeoNet NZ Volcano Feeds."""
2 
3 from __future__ import annotations
4 
5 import logging
6 
7 from homeassistant.components.sensor import SensorEntity
8 from homeassistant.config_entries import ConfigEntry
9 from homeassistant.const import ATTR_LATITUDE, ATTR_LONGITUDE, UnitOfLength
10 from homeassistant.core import HomeAssistant, callback
11 from homeassistant.helpers.dispatcher import async_dispatcher_connect
12 from homeassistant.helpers.entity_platform import AddEntitiesCallback
13 from homeassistant.util import dt as dt_util
14 from homeassistant.util.unit_conversion import DistanceConverter
15 
16 from .const import (
17  ATTR_ACTIVITY,
18  ATTR_DISTANCE,
19  ATTR_EXTERNAL_ID,
20  ATTR_HAZARDS,
21  DEFAULT_ICON,
22  DOMAIN,
23  FEED,
24  IMPERIAL_UNITS,
25 )
26 
27 _LOGGER = logging.getLogger(__name__)
28 
29 ATTR_LAST_UPDATE = "feed_last_update"
30 ATTR_LAST_UPDATE_SUCCESSFUL = "feed_last_update_successful"
31 
32 
34  hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
35 ) -> None:
36  """Set up the GeoNet NZ Volcano Feed platform."""
37  manager = hass.data[DOMAIN][FEED][entry.entry_id]
38 
39  @callback
40  def async_add_sensor(feed_manager, external_id, unit_system):
41  """Add sensor entity from feed."""
42  new_entity = GeonetnzVolcanoSensor(
43  entry.entry_id, feed_manager, external_id, unit_system
44  )
45  _LOGGER.debug("Adding sensor %s", new_entity)
46  async_add_entities([new_entity], True)
47 
48  manager.listeners.append(
50  hass, manager.async_event_new_entity(), async_add_sensor
51  )
52  )
53  hass.async_create_task(manager.async_update())
54  _LOGGER.debug("Sensor setup done")
55 
56 
58  """Represents an external event with GeoNet NZ Volcano feed data."""
59 
60  _attr_should_poll = False
61 
62  def __init__(self, config_entry_id, feed_manager, external_id, unit_system):
63  """Initialize entity with data from feed entry."""
64  self._config_entry_id_config_entry_id = config_entry_id
65  self._feed_manager_feed_manager = feed_manager
66  self._external_id_external_id = external_id
67  self._attr_unique_id_attr_unique_id = f"{config_entry_id}_{external_id}"
68  self._unit_system_unit_system = unit_system
69  self._title_title = None
70  self._distance_distance = None
71  self._latitude_latitude = None
72  self._longitude_longitude = None
73  self._attribution_attribution = None
74  self._alert_level_alert_level = None
75  self._activity_activity = None
76  self._hazards_hazards = None
77  self._feed_last_update_feed_last_update = None
78  self._feed_last_update_successful_feed_last_update_successful = None
79  self._remove_signal_update_remove_signal_update = None
80 
81  async def async_added_to_hass(self) -> None:
82  """Call when entity is added to hass."""
83  self._remove_signal_update_remove_signal_update = async_dispatcher_connect(
84  self.hasshass,
85  f"geonetnz_volcano_update_{self._external_id}",
86  self._update_callback_update_callback,
87  )
88 
89  async def async_will_remove_from_hass(self) -> None:
90  """Call when entity will be removed from hass."""
91  if self._remove_signal_update_remove_signal_update:
92  self._remove_signal_update_remove_signal_update()
93 
94  @callback
95  def _update_callback(self):
96  """Call update method."""
97  self.async_schedule_update_ha_stateasync_schedule_update_ha_state(True)
98 
99  async def async_update(self) -> None:
100  """Update this entity from the data held in the feed manager."""
101  _LOGGER.debug("Updating %s", self._external_id_external_id)
102  feed_entry = self._feed_manager_feed_manager.get_entry(self._external_id_external_id)
103  last_update = self._feed_manager_feed_manager.last_update()
104  last_update_successful = self._feed_manager_feed_manager.last_update_successful()
105  if feed_entry:
106  self._update_from_feed_update_from_feed(feed_entry, last_update, last_update_successful)
107 
108  def _update_from_feed(self, feed_entry, last_update, last_update_successful):
109  """Update the internal state from the provided feed entry."""
110  self._title_title = feed_entry.title
111  # Convert distance if not metric system.
112  if self._unit_system_unit_system == IMPERIAL_UNITS:
113  self._distance_distance = round(
114  DistanceConverter.convert(
115  feed_entry.distance_to_home,
116  UnitOfLength.KILOMETERS,
117  UnitOfLength.MILES,
118  ),
119  1,
120  )
121  else:
122  self._distance_distance = round(feed_entry.distance_to_home, 1)
123  self._latitude_latitude = round(feed_entry.coordinates[0], 5)
124  self._longitude_longitude = round(feed_entry.coordinates[1], 5)
125  self._attr_attribution_attr_attribution = feed_entry.attribution
126  self._alert_level_alert_level = feed_entry.alert_level
127  self._activity_activity = feed_entry.activity
128  self._hazards_hazards = feed_entry.hazards
129  self._feed_last_update_feed_last_update = dt_util.as_utc(last_update) if last_update else None
130  self._feed_last_update_successful_feed_last_update_successful = (
131  dt_util.as_utc(last_update_successful) if last_update_successful else None
132  )
133 
134  @property
135  def native_value(self):
136  """Return the state of the sensor."""
137  return self._alert_level_alert_level
138 
139  @property
140  def icon(self):
141  """Return the icon to use in the frontend, if any."""
142  return DEFAULT_ICON
143 
144  @property
145  def name(self) -> str | None:
146  """Return the name of the entity."""
147  return f"Volcano {self._title}"
148 
149  @property
151  """Return the unit of measurement."""
152  return "alert level"
153 
154  @property
156  """Return the device state attributes."""
157  return {
158  key: value
159  for key, value in (
160  (ATTR_EXTERNAL_ID, self._external_id_external_id),
161  (ATTR_ACTIVITY, self._activity_activity),
162  (ATTR_HAZARDS, self._hazards_hazards),
163  (ATTR_LONGITUDE, self._longitude_longitude),
164  (ATTR_LATITUDE, self._latitude_latitude),
165  (ATTR_DISTANCE, self._distance_distance),
166  (ATTR_LAST_UPDATE, self._feed_last_update_feed_last_update),
167  (ATTR_LAST_UPDATE_SUCCESSFUL, self._feed_last_update_successful_feed_last_update_successful),
168  )
169  if value or isinstance(value, bool)
170  }
def __init__(self, config_entry_id, feed_manager, external_id, unit_system)
Definition: sensor.py:62
def _update_from_feed(self, feed_entry, last_update, last_update_successful)
Definition: sensor.py:108
None async_schedule_update_ha_state(self, bool force_refresh=False)
Definition: entity.py:1265
config_entries.ConfigEntry|None get_entry(HomeAssistant hass, websocket_api.ActiveConnection connection, str entry_id, int msg_id)
None async_setup_entry(HomeAssistant hass, ConfigEntry entry, AddEntitiesCallback async_add_entities)
Definition: sensor.py:35
Callable[[], None] async_dispatcher_connect(HomeAssistant hass, str signal, Callable[..., Any] target)
Definition: dispatcher.py:103