2019-01-28 23:52:42 +00:00
|
|
|
"""Provide a way to connect devices to one physical location."""
|
|
|
|
from collections import OrderedDict
|
2020-12-01 11:45:56 +00:00
|
|
|
from typing import Container, Dict, Iterable, List, MutableMapping, Optional, cast
|
2019-01-28 23:52:42 +00:00
|
|
|
|
|
|
|
import attr
|
|
|
|
|
|
|
|
from homeassistant.core import callback
|
2021-02-12 16:00:35 +00:00
|
|
|
from homeassistant.helpers import device_registry as dr, entity_registry as er
|
2019-01-28 23:52:42 +00:00
|
|
|
from homeassistant.loader import bind_hass
|
2020-12-01 11:45:56 +00:00
|
|
|
from homeassistant.util import slugify
|
2019-03-27 14:06:20 +00:00
|
|
|
|
2019-02-07 21:34:14 +00:00
|
|
|
from .typing import HomeAssistantType
|
2019-01-28 23:52:42 +00:00
|
|
|
|
2021-02-08 10:59:46 +00:00
|
|
|
# mypy: disallow-any-generics
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
DATA_REGISTRY = "area_registry"
|
|
|
|
EVENT_AREA_REGISTRY_UPDATED = "area_registry_updated"
|
|
|
|
STORAGE_KEY = "core.area_registry"
|
2019-01-28 23:52:42 +00:00
|
|
|
STORAGE_VERSION = 1
|
|
|
|
SAVE_DELAY = 10
|
|
|
|
|
|
|
|
|
|
|
|
@attr.s(slots=True, frozen=True)
|
|
|
|
class AreaEntry:
|
|
|
|
"""Area Registry Entry."""
|
|
|
|
|
2020-12-01 11:45:56 +00:00
|
|
|
name: str = attr.ib()
|
2021-02-20 05:34:33 +00:00
|
|
|
normalized_name: str = attr.ib()
|
2020-12-01 11:45:56 +00:00
|
|
|
id: Optional[str] = attr.ib(default=None)
|
|
|
|
|
2021-02-08 10:59:46 +00:00
|
|
|
def generate_id(self, existing_ids: Container[str]) -> None:
|
2020-12-01 11:45:56 +00:00
|
|
|
"""Initialize ID."""
|
|
|
|
suggestion = suggestion_base = slugify(self.name)
|
|
|
|
tries = 1
|
|
|
|
while suggestion in existing_ids:
|
|
|
|
tries += 1
|
|
|
|
suggestion = f"{suggestion_base}_{tries}"
|
|
|
|
object.__setattr__(self, "id", suggestion)
|
2019-01-28 23:52:42 +00:00
|
|
|
|
|
|
|
|
|
|
|
class AreaRegistry:
|
|
|
|
"""Class to hold a registry of areas."""
|
|
|
|
|
2019-02-07 21:34:14 +00:00
|
|
|
def __init__(self, hass: HomeAssistantType) -> None:
|
2019-01-28 23:52:42 +00:00
|
|
|
"""Initialize the area registry."""
|
|
|
|
self.hass = hass
|
2019-09-04 03:36:04 +00:00
|
|
|
self.areas: MutableMapping[str, AreaEntry] = {}
|
2019-01-28 23:52:42 +00:00
|
|
|
self._store = hass.helpers.storage.Store(STORAGE_VERSION, STORAGE_KEY)
|
2021-02-20 05:34:33 +00:00
|
|
|
self._normalized_name_area_idx: Dict[str, str] = {}
|
2019-01-28 23:52:42 +00:00
|
|
|
|
2019-03-04 17:51:12 +00:00
|
|
|
@callback
|
|
|
|
def async_get_area(self, area_id: str) -> Optional[AreaEntry]:
|
2021-02-20 05:34:33 +00:00
|
|
|
"""Get area by id."""
|
2019-03-04 17:51:12 +00:00
|
|
|
return self.areas.get(area_id)
|
|
|
|
|
2021-02-20 05:34:33 +00:00
|
|
|
@callback
|
|
|
|
def async_get_area_by_name(self, name: str) -> Optional[AreaEntry]:
|
|
|
|
"""Get area by name."""
|
|
|
|
normalized_name = normalize_area_name(name)
|
|
|
|
if normalized_name not in self._normalized_name_area_idx:
|
|
|
|
return None
|
|
|
|
return self.areas[self._normalized_name_area_idx[normalized_name]]
|
|
|
|
|
2019-01-28 23:52:42 +00:00
|
|
|
@callback
|
2019-02-07 21:34:14 +00:00
|
|
|
def async_list_areas(self) -> Iterable[AreaEntry]:
|
2019-01-28 23:52:42 +00:00
|
|
|
"""Get all areas."""
|
|
|
|
return self.areas.values()
|
|
|
|
|
2021-02-20 05:34:33 +00:00
|
|
|
@callback
|
|
|
|
def async_get_or_create(self, name: str) -> AreaEntry:
|
|
|
|
"""Get or create an area."""
|
|
|
|
area = self.async_get_area_by_name(name)
|
|
|
|
if area:
|
|
|
|
return area
|
|
|
|
return self.async_create(name)
|
|
|
|
|
2019-01-28 23:52:42 +00:00
|
|
|
@callback
|
|
|
|
def async_create(self, name: str) -> AreaEntry:
|
|
|
|
"""Create a new area."""
|
2021-02-20 05:34:33 +00:00
|
|
|
normalized_name = normalize_area_name(name)
|
|
|
|
|
|
|
|
if self.async_get_area_by_name(name):
|
|
|
|
raise ValueError(f"The name {name} ({normalized_name}) is already in use")
|
2019-01-28 23:52:42 +00:00
|
|
|
|
2021-02-20 05:34:33 +00:00
|
|
|
area = AreaEntry(name=name, normalized_name=normalized_name)
|
2020-12-01 11:45:56 +00:00
|
|
|
area.generate_id(self.areas)
|
|
|
|
assert area.id is not None
|
2019-01-28 23:52:42 +00:00
|
|
|
self.areas[area.id] = area
|
2021-02-20 05:34:33 +00:00
|
|
|
self._normalized_name_area_idx[normalized_name] = area.id
|
2020-12-01 11:45:56 +00:00
|
|
|
self.async_schedule_save()
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass.bus.async_fire(
|
2020-12-01 11:45:56 +00:00
|
|
|
EVENT_AREA_REGISTRY_UPDATED, {"action": "create", "area_id": area.id}
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2020-12-01 11:45:56 +00:00
|
|
|
return area
|
2019-01-28 23:52:42 +00:00
|
|
|
|
2021-02-12 16:00:35 +00:00
|
|
|
@callback
|
|
|
|
def async_delete(self, area_id: str) -> None:
|
2019-01-28 23:52:42 +00:00
|
|
|
"""Delete area."""
|
2021-02-20 05:34:33 +00:00
|
|
|
area = self.areas[area_id]
|
2021-02-12 16:00:35 +00:00
|
|
|
device_registry = dr.async_get(self.hass)
|
|
|
|
entity_registry = er.async_get(self.hass)
|
2019-01-28 23:52:42 +00:00
|
|
|
device_registry.async_clear_area_id(area_id)
|
2020-10-24 19:25:28 +00:00
|
|
|
entity_registry.async_clear_area_id(area_id)
|
2019-01-28 23:52:42 +00:00
|
|
|
|
|
|
|
del self.areas[area_id]
|
2021-02-20 05:34:33 +00:00
|
|
|
del self._normalized_name_area_idx[area.normalized_name]
|
2019-01-28 23:52:42 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass.bus.async_fire(
|
|
|
|
EVENT_AREA_REGISTRY_UPDATED, {"action": "remove", "area_id": area_id}
|
|
|
|
)
|
2019-05-08 03:04:57 +00:00
|
|
|
|
2019-01-28 23:52:42 +00:00
|
|
|
self.async_schedule_save()
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_update(self, area_id: str, name: str) -> AreaEntry:
|
2019-05-08 03:04:57 +00:00
|
|
|
"""Update name of area."""
|
|
|
|
updated = self._async_update(area_id, name)
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass.bus.async_fire(
|
|
|
|
EVENT_AREA_REGISTRY_UPDATED, {"action": "update", "area_id": area_id}
|
|
|
|
)
|
2019-05-08 03:04:57 +00:00
|
|
|
return updated
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def _async_update(self, area_id: str, name: str) -> AreaEntry:
|
2019-01-28 23:52:42 +00:00
|
|
|
"""Update name of area."""
|
|
|
|
old = self.areas[area_id]
|
|
|
|
|
|
|
|
changes = {}
|
|
|
|
|
|
|
|
if name == old.name:
|
|
|
|
return old
|
|
|
|
|
2021-02-20 05:34:33 +00:00
|
|
|
normalized_name = normalize_area_name(name)
|
|
|
|
|
|
|
|
if normalized_name != old.normalized_name:
|
|
|
|
if self.async_get_area_by_name(name):
|
|
|
|
raise ValueError(
|
|
|
|
f"The name {name} ({normalized_name}) is already in use"
|
|
|
|
)
|
2019-02-27 21:10:40 +00:00
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
changes["name"] = name
|
2021-02-20 05:34:33 +00:00
|
|
|
changes["normalized_name"] = normalized_name
|
2019-01-28 23:52:42 +00:00
|
|
|
|
|
|
|
new = self.areas[area_id] = attr.evolve(old, **changes)
|
2021-02-20 05:34:33 +00:00
|
|
|
self._normalized_name_area_idx[
|
|
|
|
normalized_name
|
|
|
|
] = self._normalized_name_area_idx.pop(old.normalized_name)
|
|
|
|
|
2019-01-28 23:52:42 +00:00
|
|
|
self.async_schedule_save()
|
|
|
|
return new
|
|
|
|
|
|
|
|
async def async_load(self) -> None:
|
|
|
|
"""Load the area registry."""
|
|
|
|
data = await self._store.async_load()
|
|
|
|
|
2019-09-04 03:36:04 +00:00
|
|
|
areas: MutableMapping[str, AreaEntry] = OrderedDict()
|
2019-01-28 23:52:42 +00:00
|
|
|
|
|
|
|
if data is not None:
|
2019-07-31 19:25:30 +00:00
|
|
|
for area in data["areas"]:
|
2021-02-20 05:34:33 +00:00
|
|
|
normalized_name = normalize_area_name(area["name"])
|
|
|
|
areas[area["id"]] = AreaEntry(
|
|
|
|
name=area["name"], id=area["id"], normalized_name=normalized_name
|
|
|
|
)
|
|
|
|
self._normalized_name_area_idx[normalized_name] = area["id"]
|
2019-01-28 23:52:42 +00:00
|
|
|
|
|
|
|
self.areas = areas
|
|
|
|
|
|
|
|
@callback
|
|
|
|
def async_schedule_save(self) -> None:
|
|
|
|
"""Schedule saving the area registry."""
|
|
|
|
self._store.async_delay_save(self._data_to_save, SAVE_DELAY)
|
|
|
|
|
|
|
|
@callback
|
2020-07-22 15:06:37 +00:00
|
|
|
def _data_to_save(self) -> Dict[str, List[Dict[str, Optional[str]]]]:
|
2019-01-28 23:52:42 +00:00
|
|
|
"""Return data of area registry to store in a file."""
|
|
|
|
data = {}
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
data["areas"] = [
|
2021-02-20 05:34:33 +00:00
|
|
|
{
|
|
|
|
"name": entry.name,
|
|
|
|
"id": entry.id,
|
|
|
|
}
|
|
|
|
for entry in self.areas.values()
|
2019-01-28 23:52:42 +00:00
|
|
|
]
|
|
|
|
|
|
|
|
return data
|
|
|
|
|
|
|
|
|
2021-02-11 16:36:19 +00:00
|
|
|
@callback
|
|
|
|
def async_get(hass: HomeAssistantType) -> AreaRegistry:
|
|
|
|
"""Get area registry."""
|
|
|
|
return cast(AreaRegistry, hass.data[DATA_REGISTRY])
|
2019-03-27 14:06:20 +00:00
|
|
|
|
|
|
|
|
2021-02-11 16:36:19 +00:00
|
|
|
async def async_load(hass: HomeAssistantType) -> None:
|
|
|
|
"""Load area registry."""
|
|
|
|
assert DATA_REGISTRY not in hass.data
|
|
|
|
hass.data[DATA_REGISTRY] = AreaRegistry(hass)
|
|
|
|
await hass.data[DATA_REGISTRY].async_load()
|
2019-01-28 23:52:42 +00:00
|
|
|
|
|
|
|
|
2021-02-11 16:36:19 +00:00
|
|
|
@bind_hass
|
|
|
|
async def async_get_registry(hass: HomeAssistantType) -> AreaRegistry:
|
|
|
|
"""Get area registry.
|
2019-01-28 23:52:42 +00:00
|
|
|
|
2021-02-11 16:36:19 +00:00
|
|
|
This is deprecated and will be removed in the future. Use async_get instead.
|
|
|
|
"""
|
|
|
|
return async_get(hass)
|
2021-02-20 05:34:33 +00:00
|
|
|
|
|
|
|
|
|
|
|
def normalize_area_name(area_name: str) -> str:
|
|
|
|
"""Normalize an area name by removing whitespace and case folding."""
|
|
|
|
return area_name.casefold().replace(" ", "")
|