463 lines
15 KiB
Python
463 lines
15 KiB
Python
"""Provide a way to connect devices to one physical location."""
|
|
|
|
from __future__ import annotations
|
|
|
|
from collections import defaultdict
|
|
from collections.abc import Iterable
|
|
import dataclasses
|
|
from functools import cached_property
|
|
from typing import Any, Literal, TypedDict
|
|
|
|
from homeassistant.core import HomeAssistant, callback
|
|
from homeassistant.util import slugify
|
|
from homeassistant.util.event_type import EventType
|
|
from homeassistant.util.hass_dict import HassKey
|
|
|
|
from . import device_registry as dr, entity_registry as er
|
|
from .json import json_bytes, json_fragment
|
|
from .normalized_name_base_registry import (
|
|
NormalizedNameBaseRegistryEntry,
|
|
NormalizedNameBaseRegistryItems,
|
|
normalize_name,
|
|
)
|
|
from .registry import BaseRegistry, RegistryIndexType
|
|
from .singleton import singleton
|
|
from .storage import Store
|
|
from .typing import UNDEFINED, UndefinedType
|
|
|
|
DATA_REGISTRY: HassKey[AreaRegistry] = HassKey("area_registry")
|
|
EVENT_AREA_REGISTRY_UPDATED: EventType[EventAreaRegistryUpdatedData] = EventType(
|
|
"area_registry_updated"
|
|
)
|
|
STORAGE_KEY = "core.area_registry"
|
|
STORAGE_VERSION_MAJOR = 1
|
|
STORAGE_VERSION_MINOR = 6
|
|
|
|
|
|
class _AreaStoreData(TypedDict):
|
|
"""Data type for individual area. Used in AreasRegistryStoreData."""
|
|
|
|
aliases: list[str]
|
|
floor_id: str | None
|
|
icon: str | None
|
|
id: str
|
|
labels: list[str]
|
|
name: str
|
|
picture: str | None
|
|
|
|
|
|
class AreasRegistryStoreData(TypedDict):
|
|
"""Store data type for AreaRegistry."""
|
|
|
|
areas: list[_AreaStoreData]
|
|
|
|
|
|
class EventAreaRegistryUpdatedData(TypedDict):
|
|
"""EventAreaRegistryUpdated data."""
|
|
|
|
action: Literal["create", "remove", "update"]
|
|
area_id: str
|
|
|
|
|
|
@dataclasses.dataclass(frozen=True, kw_only=True)
|
|
class AreaEntry(NormalizedNameBaseRegistryEntry):
|
|
"""Area Registry Entry."""
|
|
|
|
aliases: set[str]
|
|
floor_id: str | None
|
|
icon: str | None
|
|
id: str
|
|
labels: set[str] = dataclasses.field(default_factory=set)
|
|
picture: str | None
|
|
|
|
@cached_property
|
|
def json_fragment(self) -> json_fragment:
|
|
"""Return a JSON representation of this AreaEntry."""
|
|
return json_fragment(
|
|
json_bytes(
|
|
{
|
|
"aliases": list(self.aliases),
|
|
"area_id": self.id,
|
|
"floor_id": self.floor_id,
|
|
"icon": self.icon,
|
|
"labels": list(self.labels),
|
|
"name": self.name,
|
|
"picture": self.picture,
|
|
}
|
|
)
|
|
)
|
|
|
|
|
|
class AreaRegistryStore(Store[AreasRegistryStoreData]):
|
|
"""Store area registry data."""
|
|
|
|
async def _async_migrate_func(
|
|
self,
|
|
old_major_version: int,
|
|
old_minor_version: int,
|
|
old_data: dict[str, list[dict[str, Any]]],
|
|
) -> AreasRegistryStoreData:
|
|
"""Migrate to the new version."""
|
|
if old_major_version < 2:
|
|
if old_minor_version < 2:
|
|
# Version 1.2 implements migration and freezes the available keys
|
|
for area in old_data["areas"]:
|
|
# Populate keys which were introduced before version 1.2
|
|
area.setdefault("picture", None)
|
|
|
|
if old_minor_version < 3:
|
|
# Version 1.3 adds aliases
|
|
for area in old_data["areas"]:
|
|
area["aliases"] = []
|
|
|
|
if old_minor_version < 4:
|
|
# Version 1.4 adds icon
|
|
for area in old_data["areas"]:
|
|
area["icon"] = None
|
|
|
|
if old_minor_version < 5:
|
|
# Version 1.5 adds floor_id
|
|
for area in old_data["areas"]:
|
|
area["floor_id"] = None
|
|
|
|
if old_minor_version < 6:
|
|
# Version 1.6 adds labels
|
|
for area in old_data["areas"]:
|
|
area["labels"] = []
|
|
|
|
if old_major_version > 1:
|
|
raise NotImplementedError
|
|
return old_data # type: ignore[return-value]
|
|
|
|
|
|
class AreaRegistryItems(NormalizedNameBaseRegistryItems[AreaEntry]):
|
|
"""Class to hold area registry items."""
|
|
|
|
def __init__(self) -> None:
|
|
"""Initialize the area registry items."""
|
|
super().__init__()
|
|
self._labels_index: RegistryIndexType = defaultdict(dict)
|
|
self._floors_index: RegistryIndexType = defaultdict(dict)
|
|
|
|
def _index_entry(self, key: str, entry: AreaEntry) -> None:
|
|
"""Index an entry."""
|
|
if entry.floor_id is not None:
|
|
self._floors_index[entry.floor_id][key] = True
|
|
for label in entry.labels:
|
|
self._labels_index[label][key] = True
|
|
super()._index_entry(key, entry)
|
|
|
|
def _unindex_entry(
|
|
self, key: str, replacement_entry: AreaEntry | None = None
|
|
) -> None:
|
|
entry = self.data[key]
|
|
if labels := entry.labels:
|
|
for label in labels:
|
|
self._unindex_entry_value(key, label, self._labels_index)
|
|
if floor_id := entry.floor_id:
|
|
self._unindex_entry_value(key, floor_id, self._floors_index)
|
|
return super()._unindex_entry(key, replacement_entry)
|
|
|
|
def get_areas_for_label(self, label: str) -> list[AreaEntry]:
|
|
"""Get areas for label."""
|
|
data = self.data
|
|
return [data[key] for key in self._labels_index.get(label, ())]
|
|
|
|
def get_areas_for_floor(self, floor: str) -> list[AreaEntry]:
|
|
"""Get areas for floor."""
|
|
data = self.data
|
|
return [data[key] for key in self._floors_index.get(floor, ())]
|
|
|
|
|
|
class AreaRegistry(BaseRegistry[AreasRegistryStoreData]):
|
|
"""Class to hold a registry of areas."""
|
|
|
|
areas: AreaRegistryItems
|
|
_area_data: dict[str, AreaEntry]
|
|
|
|
def __init__(self, hass: HomeAssistant) -> None:
|
|
"""Initialize the area registry."""
|
|
self.hass = hass
|
|
self._store = AreaRegistryStore(
|
|
hass,
|
|
STORAGE_VERSION_MAJOR,
|
|
STORAGE_KEY,
|
|
atomic_writes=True,
|
|
minor_version=STORAGE_VERSION_MINOR,
|
|
)
|
|
|
|
@callback
|
|
def async_get_area(self, area_id: str) -> AreaEntry | None:
|
|
"""Get area by id.
|
|
|
|
We retrieve the DeviceEntry from the underlying dict to avoid
|
|
the overhead of the UserDict __getitem__.
|
|
"""
|
|
return self._area_data.get(area_id)
|
|
|
|
@callback
|
|
def async_get_area_by_name(self, name: str) -> AreaEntry | None:
|
|
"""Get area by name."""
|
|
return self.areas.get_by_name(name)
|
|
|
|
@callback
|
|
def async_list_areas(self) -> Iterable[AreaEntry]:
|
|
"""Get all areas."""
|
|
return self.areas.values()
|
|
|
|
@callback
|
|
def async_get_or_create(self, name: str) -> AreaEntry:
|
|
"""Get or create an area."""
|
|
if area := self.async_get_area_by_name(name):
|
|
return area
|
|
return self.async_create(name)
|
|
|
|
@callback
|
|
def async_create(
|
|
self,
|
|
name: str,
|
|
*,
|
|
aliases: set[str] | None = None,
|
|
floor_id: str | None = None,
|
|
icon: str | None = None,
|
|
labels: set[str] | None = None,
|
|
picture: str | None = None,
|
|
) -> AreaEntry:
|
|
"""Create a new area."""
|
|
self.hass.verify_event_loop_thread("area_registry.async_create")
|
|
normalized_name = normalize_name(name)
|
|
|
|
if self.async_get_area_by_name(name):
|
|
raise ValueError(f"The name {name} ({normalized_name}) is already in use")
|
|
|
|
area_id = self._generate_area_id(name)
|
|
area = AreaEntry(
|
|
aliases=aliases or set(),
|
|
floor_id=floor_id,
|
|
icon=icon,
|
|
id=area_id,
|
|
labels=labels or set(),
|
|
name=name,
|
|
normalized_name=normalized_name,
|
|
picture=picture,
|
|
)
|
|
assert area.id is not None
|
|
self.areas[area.id] = area
|
|
self.async_schedule_save()
|
|
self.hass.bus.async_fire_internal(
|
|
EVENT_AREA_REGISTRY_UPDATED,
|
|
EventAreaRegistryUpdatedData(action="create", area_id=area.id),
|
|
)
|
|
return area
|
|
|
|
@callback
|
|
def async_delete(self, area_id: str) -> None:
|
|
"""Delete area."""
|
|
self.hass.verify_event_loop_thread("area_registry.async_delete")
|
|
device_registry = dr.async_get(self.hass)
|
|
entity_registry = er.async_get(self.hass)
|
|
device_registry.async_clear_area_id(area_id)
|
|
entity_registry.async_clear_area_id(area_id)
|
|
|
|
del self.areas[area_id]
|
|
|
|
self.hass.bus.async_fire_internal(
|
|
EVENT_AREA_REGISTRY_UPDATED,
|
|
EventAreaRegistryUpdatedData(action="remove", area_id=area_id),
|
|
)
|
|
|
|
self.async_schedule_save()
|
|
|
|
@callback
|
|
def async_update(
|
|
self,
|
|
area_id: str,
|
|
*,
|
|
aliases: set[str] | UndefinedType = UNDEFINED,
|
|
floor_id: str | None | UndefinedType = UNDEFINED,
|
|
icon: str | None | UndefinedType = UNDEFINED,
|
|
labels: set[str] | UndefinedType = UNDEFINED,
|
|
name: str | UndefinedType = UNDEFINED,
|
|
picture: str | None | UndefinedType = UNDEFINED,
|
|
) -> AreaEntry:
|
|
"""Update name of area."""
|
|
updated = self._async_update(
|
|
area_id,
|
|
aliases=aliases,
|
|
floor_id=floor_id,
|
|
icon=icon,
|
|
labels=labels,
|
|
name=name,
|
|
picture=picture,
|
|
)
|
|
# Since updated may be the old or the new and we always fire
|
|
# an event even if nothing has changed we cannot use async_fire_internal
|
|
# here because we do not know if the thread safety check already
|
|
# happened or not in _async_update.
|
|
self.hass.bus.async_fire(
|
|
EVENT_AREA_REGISTRY_UPDATED,
|
|
EventAreaRegistryUpdatedData(action="update", area_id=area_id),
|
|
)
|
|
return updated
|
|
|
|
@callback
|
|
def _async_update(
|
|
self,
|
|
area_id: str,
|
|
*,
|
|
aliases: set[str] | UndefinedType = UNDEFINED,
|
|
floor_id: str | None | UndefinedType = UNDEFINED,
|
|
icon: str | None | UndefinedType = UNDEFINED,
|
|
labels: set[str] | UndefinedType = UNDEFINED,
|
|
name: str | UndefinedType = UNDEFINED,
|
|
picture: str | None | UndefinedType = UNDEFINED,
|
|
) -> AreaEntry:
|
|
"""Update name of area."""
|
|
old = self.areas[area_id]
|
|
|
|
new_values = {}
|
|
|
|
for attr_name, value in (
|
|
("aliases", aliases),
|
|
("icon", icon),
|
|
("labels", labels),
|
|
("picture", picture),
|
|
("floor_id", floor_id),
|
|
):
|
|
if value is not UNDEFINED and value != getattr(old, attr_name):
|
|
new_values[attr_name] = value
|
|
|
|
if name is not UNDEFINED and name != old.name:
|
|
new_values["name"] = name
|
|
new_values["normalized_name"] = normalize_name(name)
|
|
|
|
if not new_values:
|
|
return old
|
|
|
|
self.hass.verify_event_loop_thread("area_registry.async_update")
|
|
new = self.areas[area_id] = dataclasses.replace(old, **new_values) # type: ignore[arg-type]
|
|
|
|
self.async_schedule_save()
|
|
return new
|
|
|
|
async def async_load(self) -> None:
|
|
"""Load the area registry."""
|
|
self._async_setup_cleanup()
|
|
|
|
data = await self._store.async_load()
|
|
|
|
areas = AreaRegistryItems()
|
|
|
|
if data is not None:
|
|
for area in data["areas"]:
|
|
assert area["name"] is not None and area["id"] is not None
|
|
normalized_name = normalize_name(area["name"])
|
|
areas[area["id"]] = AreaEntry(
|
|
aliases=set(area["aliases"]),
|
|
floor_id=area["floor_id"],
|
|
icon=area["icon"],
|
|
id=area["id"],
|
|
labels=set(area["labels"]),
|
|
name=area["name"],
|
|
normalized_name=normalized_name,
|
|
picture=area["picture"],
|
|
)
|
|
|
|
self.areas = areas
|
|
self._area_data = areas.data
|
|
|
|
@callback
|
|
def _data_to_save(self) -> AreasRegistryStoreData:
|
|
"""Return data of area registry to store in a file."""
|
|
return {
|
|
"areas": [
|
|
{
|
|
"aliases": list(entry.aliases),
|
|
"floor_id": entry.floor_id,
|
|
"icon": entry.icon,
|
|
"id": entry.id,
|
|
"labels": list(entry.labels),
|
|
"name": entry.name,
|
|
"picture": entry.picture,
|
|
}
|
|
for entry in self.areas.values()
|
|
]
|
|
}
|
|
|
|
def _generate_area_id(self, name: str) -> str:
|
|
"""Generate area ID."""
|
|
suggestion = suggestion_base = slugify(name)
|
|
tries = 1
|
|
while suggestion in self.areas:
|
|
tries += 1
|
|
suggestion = f"{suggestion_base}_{tries}"
|
|
return suggestion
|
|
|
|
@callback
|
|
def _async_setup_cleanup(self) -> None:
|
|
"""Set up the area registry cleanup."""
|
|
# pylint: disable-next=import-outside-toplevel
|
|
from . import ( # Circular dependencies
|
|
floor_registry as fr,
|
|
label_registry as lr,
|
|
)
|
|
|
|
@callback
|
|
def _removed_from_registry_filter(
|
|
event_data: fr.EventFloorRegistryUpdatedData
|
|
| lr.EventLabelRegistryUpdatedData,
|
|
) -> bool:
|
|
"""Filter all except for the item removed from registry events."""
|
|
return event_data["action"] == "remove"
|
|
|
|
@callback
|
|
def _handle_floor_registry_update(event: fr.EventFloorRegistryUpdated) -> None:
|
|
"""Update areas that are associated with a floor that has been removed."""
|
|
floor_id = event.data["floor_id"]
|
|
for area in self.areas.get_areas_for_floor(floor_id):
|
|
self.async_update(area.id, floor_id=None)
|
|
|
|
self.hass.bus.async_listen(
|
|
event_type=fr.EVENT_FLOOR_REGISTRY_UPDATED,
|
|
event_filter=_removed_from_registry_filter,
|
|
listener=_handle_floor_registry_update,
|
|
)
|
|
|
|
@callback
|
|
def _handle_label_registry_update(event: lr.EventLabelRegistryUpdated) -> None:
|
|
"""Update areas that have a label that has been removed."""
|
|
label_id = event.data["label_id"]
|
|
for area in self.areas.get_areas_for_label(label_id):
|
|
self.async_update(area.id, labels=area.labels - {label_id})
|
|
|
|
self.hass.bus.async_listen(
|
|
event_type=lr.EVENT_LABEL_REGISTRY_UPDATED,
|
|
event_filter=_removed_from_registry_filter,
|
|
listener=_handle_label_registry_update,
|
|
)
|
|
|
|
|
|
@callback
|
|
@singleton(DATA_REGISTRY)
|
|
def async_get(hass: HomeAssistant) -> AreaRegistry:
|
|
"""Get area registry."""
|
|
return AreaRegistry(hass)
|
|
|
|
|
|
async def async_load(hass: HomeAssistant) -> None:
|
|
"""Load area registry."""
|
|
assert DATA_REGISTRY not in hass.data
|
|
await async_get(hass).async_load()
|
|
|
|
|
|
@callback
|
|
def async_entries_for_floor(registry: AreaRegistry, floor_id: str) -> list[AreaEntry]:
|
|
"""Return entries that match a floor."""
|
|
return registry.areas.get_areas_for_floor(floor_id)
|
|
|
|
|
|
@callback
|
|
def async_entries_for_label(registry: AreaRegistry, label_id: str) -> list[AreaEntry]:
|
|
"""Return entries that match a label."""
|
|
return registry.areas.get_areas_for_label(label_id)
|