Home Assistant Unofficial Reference 2024.12.1
__init__.py
Go to the documentation of this file.
1 """The GeoNet NZ Quakes integration."""
2 
3 from datetime import timedelta
4 import logging
5 
6 from aio_geojson_geonetnz_quakes import GeonetnzQuakesFeedManager
7 import voluptuous as vol
8 
9 from homeassistant.config_entries import SOURCE_IMPORT, ConfigEntry
10 from homeassistant.const import (
11  CONF_LATITUDE,
12  CONF_LONGITUDE,
13  CONF_RADIUS,
14  CONF_SCAN_INTERVAL,
15  UnitOfLength,
16 )
17 from homeassistant.core import HomeAssistant, callback
18 from homeassistant.helpers import aiohttp_client, config_validation as cv
19 from homeassistant.helpers.dispatcher import async_dispatcher_send
20 from homeassistant.helpers.event import async_track_time_interval
21 from homeassistant.helpers.typing import ConfigType
22 from homeassistant.util.unit_conversion import DistanceConverter
23 from homeassistant.util.unit_system import US_CUSTOMARY_SYSTEM
24 
25 from .const import (
26  CONF_MINIMUM_MAGNITUDE,
27  CONF_MMI,
28  DEFAULT_FILTER_TIME_INTERVAL,
29  DEFAULT_MINIMUM_MAGNITUDE,
30  DEFAULT_MMI,
31  DEFAULT_RADIUS,
32  DEFAULT_SCAN_INTERVAL,
33  DOMAIN,
34  FEED,
35  PLATFORMS,
36 )
37 
38 _LOGGER = logging.getLogger(__name__)
39 
40 CONFIG_SCHEMA = vol.Schema(
41  {
42  DOMAIN: vol.Schema(
43  {
44  vol.Inclusive(CONF_LATITUDE, "coordinates"): cv.latitude,
45  vol.Inclusive(CONF_LONGITUDE, "coordinates"): cv.longitude,
46  vol.Optional(CONF_MMI, default=DEFAULT_MMI): vol.All(
47  vol.Coerce(int), vol.Range(min=-1, max=8)
48  ),
49  vol.Optional(CONF_RADIUS, default=DEFAULT_RADIUS): vol.Coerce(float),
50  vol.Optional(
51  CONF_MINIMUM_MAGNITUDE, default=DEFAULT_MINIMUM_MAGNITUDE
52  ): cv.positive_float,
53  vol.Optional(
54  CONF_SCAN_INTERVAL, default=DEFAULT_SCAN_INTERVAL
55  ): cv.time_period,
56  }
57  )
58  },
59  extra=vol.ALLOW_EXTRA,
60 )
61 
62 
63 async def async_setup(hass: HomeAssistant, config: ConfigType) -> bool:
64  """Set up the GeoNet NZ Quakes component."""
65  if DOMAIN not in config:
66  return True
67 
68  conf = config[DOMAIN]
69  latitude = conf.get(CONF_LATITUDE, hass.config.latitude)
70  longitude = conf.get(CONF_LONGITUDE, hass.config.longitude)
71  mmi = conf[CONF_MMI]
72  scan_interval = conf[CONF_SCAN_INTERVAL]
73 
74  hass.async_create_task(
75  hass.config_entries.flow.async_init(
76  DOMAIN,
77  context={"source": SOURCE_IMPORT},
78  data={
79  CONF_LATITUDE: latitude,
80  CONF_LONGITUDE: longitude,
81  CONF_RADIUS: conf[CONF_RADIUS],
82  CONF_MINIMUM_MAGNITUDE: conf[CONF_MINIMUM_MAGNITUDE],
83  CONF_MMI: mmi,
84  CONF_SCAN_INTERVAL: scan_interval,
85  },
86  )
87  )
88 
89  return True
90 
91 
92 async def async_setup_entry(hass: HomeAssistant, config_entry: ConfigEntry) -> bool:
93  """Set up the GeoNet NZ Quakes component as config entry."""
94  hass.data.setdefault(DOMAIN, {})
95  feeds = hass.data[DOMAIN].setdefault(FEED, {})
96 
97  radius = config_entry.data[CONF_RADIUS]
98  if hass.config.units is US_CUSTOMARY_SYSTEM:
99  radius = DistanceConverter.convert(
100  radius, UnitOfLength.MILES, UnitOfLength.KILOMETERS
101  )
102  # Create feed entity manager for all platforms.
103  manager = GeonetnzQuakesFeedEntityManager(hass, config_entry, radius)
104  feeds[config_entry.entry_id] = manager
105  _LOGGER.debug("Feed entity manager added for %s", config_entry.entry_id)
106  await manager.async_init()
107  return True
108 
109 
110 async def async_unload_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
111  """Unload an GeoNet NZ Quakes component config entry."""
112  manager = hass.data[DOMAIN][FEED].pop(entry.entry_id)
113  await manager.async_stop()
114  return await hass.config_entries.async_unload_platforms(entry, PLATFORMS)
115 
116 
118  """Feed Entity Manager for GeoNet NZ Quakes feed."""
119 
120  def __init__(self, hass, config_entry, radius_in_km):
121  """Initialize the Feed Entity Manager."""
122  self._hass_hass = hass
123  self._config_entry_config_entry = config_entry
124  coordinates = (
125  config_entry.data[CONF_LATITUDE],
126  config_entry.data[CONF_LONGITUDE],
127  )
128  websession = aiohttp_client.async_get_clientsession(hass)
129  self._feed_manager_feed_manager = GeonetnzQuakesFeedManager(
130  websession,
131  self._generate_entity_generate_entity,
132  self._update_entity_update_entity,
133  self._remove_entity_remove_entity,
134  coordinates,
135  mmi=config_entry.data[CONF_MMI],
136  filter_radius=radius_in_km,
137  filter_minimum_magnitude=config_entry.data[CONF_MINIMUM_MAGNITUDE],
138  filter_time=DEFAULT_FILTER_TIME_INTERVAL,
139  status_callback=self._status_update_status_update,
140  )
141  self._config_entry_id_config_entry_id = config_entry.entry_id
142  self._scan_interval_scan_interval = timedelta(seconds=config_entry.data[CONF_SCAN_INTERVAL])
143  self._track_time_remove_callback_track_time_remove_callback = None
144  self._status_info_status_info = None
145  self.listenerslisteners = []
146 
147  async def async_init(self):
148  """Schedule initial and regular updates based on configured time interval."""
149 
150  await self._hass_hass.config_entries.async_forward_entry_setups(
151  self._config_entry_config_entry, PLATFORMS
152  )
153 
154  async def update(event_time):
155  """Update."""
156  await self.async_updateasync_update()
157 
158  # Trigger updates at regular intervals.
159  self._track_time_remove_callback_track_time_remove_callback = async_track_time_interval(
160  self._hass_hass, update, self._scan_interval_scan_interval
161  )
162 
163  _LOGGER.debug("Feed entity manager initialized")
164 
165  async def async_update(self):
166  """Refresh data."""
167  await self._feed_manager_feed_manager.update()
168  _LOGGER.debug("Feed entity manager updated")
169 
170  async def async_stop(self):
171  """Stop this feed entity manager from refreshing."""
172  for unsub_dispatcher in self.listenerslisteners:
173  unsub_dispatcher()
174  self.listenerslisteners = []
175  if self._track_time_remove_callback_track_time_remove_callback:
176  self._track_time_remove_callback_track_time_remove_callback()
177  _LOGGER.debug("Feed entity manager stopped")
178 
179  @callback
181  """Return manager specific event to signal new entity."""
182  return f"geonetnz_quakes_new_geolocation_{self._config_entry_id}"
183 
184  def get_entry(self, external_id):
185  """Get feed entry by external id."""
186  return self._feed_manager_feed_manager.feed_entries.get(external_id)
187 
188  def status_info(self):
189  """Return latest status update info received."""
190  return self._status_info_status_info
191 
192  async def _generate_entity(self, external_id):
193  """Generate new entity."""
195  self._hass_hass,
196  self.async_event_new_entityasync_event_new_entity(),
197  self,
198  self._config_entry_config_entry.unique_id,
199  external_id,
200  )
201 
202  async def _update_entity(self, external_id):
203  """Update entity."""
204  async_dispatcher_send(self._hass_hass, f"geonetnz_quakes_update_{external_id}")
205 
206  async def _remove_entity(self, external_id):
207  """Remove entity."""
208  async_dispatcher_send(self._hass_hass, f"geonetnz_quakes_delete_{external_id}")
209 
210  async def _status_update(self, status_info):
211  """Propagate status update."""
212  _LOGGER.debug("Status update received: %s", status_info)
213  self._status_info_status_info = status_info
215  self._hass_hass, f"geonetnz_quakes_status_{self._config_entry_id}"
216  )
def __init__(self, hass, config_entry, radius_in_km)
Definition: __init__.py:120
bool async_setup_entry(HomeAssistant hass, ConfigEntry config_entry)
Definition: __init__.py:92
bool async_unload_entry(HomeAssistant hass, ConfigEntry entry)
Definition: __init__.py:110
bool async_setup(HomeAssistant hass, ConfigType config)
Definition: __init__.py:63
IssData update(pyiss.ISS iss)
Definition: __init__.py:33
None async_dispatcher_send(HomeAssistant hass, str signal, *Any args)
Definition: dispatcher.py:193
CALLBACK_TYPE async_track_time_interval(HomeAssistant hass, Callable[[datetime], Coroutine[Any, Any, None]|None] action, timedelta interval, *str|None name=None, bool|None cancel_on_shutdown=None)
Definition: event.py:1679