"""Demo platform for the geolocation component.""" from __future__ import annotations from datetime import timedelta import logging from math import cos, pi, radians, sin import random from homeassistant.components.geo_location import GeolocationEvent from homeassistant.const import LENGTH_KILOMETERS from homeassistant.helpers.event import track_time_interval _LOGGER = logging.getLogger(__name__) AVG_KM_PER_DEGREE = 111.0 DEFAULT_UPDATE_INTERVAL = timedelta(minutes=1) MAX_RADIUS_IN_KM = 50 NUMBER_OF_DEMO_DEVICES = 5 EVENT_NAMES = [ "Bushfire", "Hazard Reduction", "Grass Fire", "Burn off", "Structure Fire", "Fire Alarm", "Thunderstorm", "Tornado", "Cyclone", "Waterspout", "Dust Storm", "Blizzard", "Ice Storm", "Earthquake", "Tsunami", ] SOURCE = "demo" def setup_platform(hass, config, add_entities, discovery_info=None): """Set up the Demo geolocations.""" DemoManager(hass, add_entities) class DemoManager: """Device manager for demo geolocation events.""" def __init__(self, hass, add_entities): """Initialise the demo geolocation event manager.""" self._hass = hass self._add_entities = add_entities self._managed_devices = [] self._update(count=NUMBER_OF_DEMO_DEVICES) self._init_regular_updates() def _generate_random_event(self): """Generate a random event in vicinity of this HA instance.""" home_latitude = self._hass.config.latitude home_longitude = self._hass.config.longitude # Approx. 111km per degree (north-south). radius_in_degrees = random.random() * MAX_RADIUS_IN_KM / AVG_KM_PER_DEGREE radius_in_km = radius_in_degrees * AVG_KM_PER_DEGREE angle = random.random() * 2 * pi # Compute coordinates based on radius and angle. Adjust longitude value # based on HA's latitude. latitude = home_latitude + radius_in_degrees * sin(angle) longitude = home_longitude + radius_in_degrees * cos(angle) / cos( radians(home_latitude) ) event_name = random.choice(EVENT_NAMES) return DemoGeolocationEvent( event_name, radius_in_km, latitude, longitude, LENGTH_KILOMETERS ) def _init_regular_updates(self): """Schedule regular updates based on configured time interval.""" track_time_interval( self._hass, lambda now: self._update(), DEFAULT_UPDATE_INTERVAL ) def _update(self, count=1): """Remove events and add new random events.""" # Remove devices. for _ in range(1, count + 1): if self._managed_devices: device = random.choice(self._managed_devices) if device: _LOGGER.debug("Removing %s", device) self._managed_devices.remove(device) self._hass.add_job(device.async_remove()) # Generate new devices from events. new_devices = [] for _ in range(1, count + 1): new_device = self._generate_random_event() _LOGGER.debug("Adding %s", new_device) new_devices.append(new_device) self._managed_devices.append(new_device) self._add_entities(new_devices) class DemoGeolocationEvent(GeolocationEvent): """This represents a demo geolocation event.""" def __init__(self, name, distance, latitude, longitude, unit_of_measurement): """Initialize entity with data provided.""" self._name = name self._distance = distance self._latitude = latitude self._longitude = longitude self._unit_of_measurement = unit_of_measurement @property def source(self) -> str: """Return source value of this external event.""" return SOURCE @property def name(self) -> str | None: """Return the name of the event.""" return self._name @property def should_poll(self): """No polling needed for a demo geolocation event.""" return False @property def distance(self) -> float | None: """Return distance value of this external event.""" return self._distance @property def latitude(self) -> float | None: """Return latitude value of this external event.""" return self._latitude @property def longitude(self) -> float | None: """Return longitude value of this external event.""" return self._longitude @property def unit_of_measurement(self): """Return the unit of measurement.""" return self._unit_of_measurement