2019-02-13 20:21:14 +00:00
|
|
|
"""Support for the definition of zones."""
|
2018-04-26 21:59:22 +00:00
|
|
|
import logging
|
2019-09-29 17:07:49 +00:00
|
|
|
from typing import Set, cast
|
2018-04-26 21:59:22 +00:00
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.const import (
|
2019-12-09 08:38:38 +00:00
|
|
|
ATTR_LATITUDE,
|
|
|
|
ATTR_LONGITUDE,
|
|
|
|
CONF_ICON,
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_LATITUDE,
|
|
|
|
CONF_LONGITUDE,
|
2019-12-09 08:38:38 +00:00
|
|
|
CONF_NAME,
|
2019-07-31 19:25:30 +00:00
|
|
|
CONF_RADIUS,
|
|
|
|
EVENT_CORE_CONFIG_UPDATE,
|
|
|
|
)
|
2019-12-09 08:38:38 +00:00
|
|
|
from homeassistant.core import State, callback
|
2018-04-26 21:59:22 +00:00
|
|
|
from homeassistant.helpers import config_per_platform
|
2019-12-09 08:38:38 +00:00
|
|
|
import homeassistant.helpers.config_validation as cv
|
2018-04-26 21:59:22 +00:00
|
|
|
from homeassistant.helpers.entity import async_generate_entity_id
|
2019-12-09 08:38:38 +00:00
|
|
|
from homeassistant.loader import bind_hass
|
2018-04-26 21:59:22 +00:00
|
|
|
from homeassistant.util import slugify
|
2019-05-25 20:34:53 +00:00
|
|
|
from homeassistant.util.location import distance
|
|
|
|
|
2018-04-26 21:59:22 +00:00
|
|
|
from .config_flow import configured_zones
|
2019-12-09 08:38:38 +00:00
|
|
|
from .const import ATTR_PASSIVE, ATTR_RADIUS, CONF_PASSIVE, DOMAIN, HOME_ZONE
|
2018-04-26 21:59:22 +00:00
|
|
|
from .zone import Zone
|
|
|
|
|
2019-09-29 17:07:49 +00:00
|
|
|
# mypy: allow-untyped-calls, allow-untyped-defs
|
|
|
|
|
2018-04-26 21:59:22 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DEFAULT_NAME = "Unnamed zone"
|
2018-04-26 21:59:22 +00:00
|
|
|
DEFAULT_PASSIVE = False
|
|
|
|
DEFAULT_RADIUS = 100
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ENTITY_ID_FORMAT = "zone.{}"
|
2018-04-26 21:59:22 +00:00
|
|
|
ENTITY_ID_HOME = ENTITY_ID_FORMAT.format(HOME_ZONE)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ICON_HOME = "mdi:home"
|
|
|
|
ICON_IMPORT = "mdi:import"
|
2018-04-26 21:59:22 +00:00
|
|
|
|
|
|
|
# The config that zone accepts is the same as if it has platforms.
|
2019-07-31 19:25:30 +00:00
|
|
|
PLATFORM_SCHEMA = vol.Schema(
|
|
|
|
{
|
|
|
|
vol.Optional(CONF_NAME, default=DEFAULT_NAME): cv.string,
|
|
|
|
vol.Required(CONF_LATITUDE): cv.latitude,
|
|
|
|
vol.Required(CONF_LONGITUDE): cv.longitude,
|
|
|
|
vol.Optional(CONF_RADIUS, default=DEFAULT_RADIUS): vol.Coerce(float),
|
|
|
|
vol.Optional(CONF_PASSIVE, default=DEFAULT_PASSIVE): cv.boolean,
|
|
|
|
vol.Optional(CONF_ICON): cv.icon,
|
|
|
|
},
|
|
|
|
extra=vol.ALLOW_EXTRA,
|
|
|
|
)
|
2018-04-26 21:59:22 +00:00
|
|
|
|
|
|
|
|
2019-05-25 20:34:53 +00:00
|
|
|
@bind_hass
|
|
|
|
def async_active_zone(hass, latitude, longitude, radius=0):
|
|
|
|
"""Find the active zone for given latitude, longitude.
|
|
|
|
|
|
|
|
This method must be run in the event loop.
|
|
|
|
"""
|
|
|
|
# Sort entity IDs so that we are deterministic if equal distance to 2 zones
|
2019-07-31 19:25:30 +00:00
|
|
|
zones = (
|
|
|
|
hass.states.get(entity_id)
|
|
|
|
for entity_id in sorted(hass.states.async_entity_ids(DOMAIN))
|
|
|
|
)
|
2019-05-25 20:34:53 +00:00
|
|
|
|
|
|
|
min_dist = None
|
|
|
|
closest = None
|
|
|
|
|
|
|
|
for zone in zones:
|
|
|
|
if zone.attributes.get(ATTR_PASSIVE):
|
|
|
|
continue
|
|
|
|
|
|
|
|
zone_dist = distance(
|
2019-07-31 19:25:30 +00:00
|
|
|
latitude,
|
|
|
|
longitude,
|
|
|
|
zone.attributes[ATTR_LATITUDE],
|
|
|
|
zone.attributes[ATTR_LONGITUDE],
|
|
|
|
)
|
2019-05-25 20:34:53 +00:00
|
|
|
|
|
|
|
within_zone = zone_dist - radius < zone.attributes[ATTR_RADIUS]
|
2019-09-29 17:07:49 +00:00
|
|
|
closer_zone = closest is None or zone_dist < min_dist # type: ignore
|
2019-07-31 19:25:30 +00:00
|
|
|
smaller_zone = (
|
|
|
|
zone_dist == min_dist
|
2019-09-29 17:07:49 +00:00
|
|
|
and zone.attributes[ATTR_RADIUS]
|
|
|
|
< cast(State, closest).attributes[ATTR_RADIUS]
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2019-05-25 20:34:53 +00:00
|
|
|
|
|
|
|
if within_zone and (closer_zone or smaller_zone):
|
|
|
|
min_dist = zone_dist
|
|
|
|
closest = zone
|
|
|
|
|
|
|
|
return closest
|
|
|
|
|
|
|
|
|
2018-04-26 21:59:22 +00:00
|
|
|
async def async_setup(hass, config):
|
2018-08-19 20:29:08 +00:00
|
|
|
"""Set up configured zones as well as home assistant zone if necessary."""
|
2018-06-07 21:06:13 +00:00
|
|
|
hass.data[DOMAIN] = {}
|
2019-09-29 17:07:49 +00:00
|
|
|
entities: Set[str] = set()
|
2018-04-26 21:59:22 +00:00
|
|
|
zone_entries = configured_zones(hass)
|
|
|
|
for _, entry in config_per_platform(config, DOMAIN):
|
2018-06-07 21:06:13 +00:00
|
|
|
if slugify(entry[CONF_NAME]) not in zone_entries:
|
2019-07-31 19:25:30 +00:00
|
|
|
zone = Zone(
|
|
|
|
hass,
|
|
|
|
entry[CONF_NAME],
|
|
|
|
entry[CONF_LATITUDE],
|
|
|
|
entry[CONF_LONGITUDE],
|
|
|
|
entry.get(CONF_RADIUS),
|
|
|
|
entry.get(CONF_ICON),
|
|
|
|
entry.get(CONF_PASSIVE),
|
|
|
|
)
|
2018-04-26 21:59:22 +00:00
|
|
|
zone.entity_id = async_generate_entity_id(
|
2019-07-31 19:25:30 +00:00
|
|
|
ENTITY_ID_FORMAT, entry[CONF_NAME], entities
|
|
|
|
)
|
2018-10-02 09:03:09 +00:00
|
|
|
hass.async_create_task(zone.async_update_ha_state())
|
2018-06-07 21:06:13 +00:00
|
|
|
entities.add(zone.entity_id)
|
2018-04-26 21:59:22 +00:00
|
|
|
|
2019-06-01 06:03:45 +00:00
|
|
|
if ENTITY_ID_HOME in entities or HOME_ZONE in zone_entries:
|
|
|
|
return True
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
zone = Zone(
|
|
|
|
hass,
|
|
|
|
hass.config.location_name,
|
|
|
|
hass.config.latitude,
|
|
|
|
hass.config.longitude,
|
|
|
|
DEFAULT_RADIUS,
|
|
|
|
ICON_HOME,
|
|
|
|
False,
|
|
|
|
)
|
2019-06-01 06:03:45 +00:00
|
|
|
zone.entity_id = ENTITY_ID_HOME
|
|
|
|
hass.async_create_task(zone.async_update_ha_state())
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def core_config_updated(_):
|
|
|
|
"""Handle core config updated."""
|
|
|
|
zone.name = hass.config.location_name
|
|
|
|
zone.latitude = hass.config.latitude
|
|
|
|
zone.longitude = hass.config.longitude
|
|
|
|
zone.async_write_ha_state()
|
|
|
|
|
|
|
|
hass.bus.async_listen(EVENT_CORE_CONFIG_UPDATE, core_config_updated)
|
2018-04-26 21:59:22 +00:00
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
async def async_setup_entry(hass, config_entry):
|
|
|
|
"""Set up zone as config entry."""
|
|
|
|
entry = config_entry.data
|
|
|
|
name = entry[CONF_NAME]
|
2019-07-31 19:25:30 +00:00
|
|
|
zone = Zone(
|
|
|
|
hass,
|
|
|
|
name,
|
|
|
|
entry[CONF_LATITUDE],
|
|
|
|
entry[CONF_LONGITUDE],
|
|
|
|
entry.get(CONF_RADIUS, DEFAULT_RADIUS),
|
|
|
|
entry.get(CONF_ICON),
|
|
|
|
entry.get(CONF_PASSIVE, DEFAULT_PASSIVE),
|
|
|
|
)
|
|
|
|
zone.entity_id = async_generate_entity_id(ENTITY_ID_FORMAT, name, None, hass)
|
2018-10-02 09:03:09 +00:00
|
|
|
hass.async_create_task(zone.async_update_ha_state())
|
2018-04-26 21:59:22 +00:00
|
|
|
hass.data[DOMAIN][slugify(name)] = zone
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
async def async_unload_entry(hass, config_entry):
|
|
|
|
"""Unload a config entry."""
|
|
|
|
zones = hass.data[DOMAIN]
|
|
|
|
name = slugify(config_entry.data[CONF_NAME])
|
|
|
|
zone = zones.pop(name)
|
|
|
|
await zone.async_remove()
|
|
|
|
return True
|