mirror of
https://github.com/home-assistant/core.git
synced 2025-07-19 19:27:45 +00:00
Allow storing picture in area registry (#58539)
This commit is contained in:
parent
f228537458
commit
6cd83d1f66
@ -2,124 +2,102 @@
|
|||||||
import voluptuous as vol
|
import voluptuous as vol
|
||||||
|
|
||||||
from homeassistant.components import websocket_api
|
from homeassistant.components import websocket_api
|
||||||
from homeassistant.components.websocket_api.decorators import (
|
|
||||||
async_response,
|
|
||||||
require_admin,
|
|
||||||
)
|
|
||||||
from homeassistant.core import callback
|
from homeassistant.core import callback
|
||||||
from homeassistant.helpers.area_registry import async_get_registry
|
from homeassistant.helpers.area_registry import async_get
|
||||||
|
|
||||||
WS_TYPE_LIST = "config/area_registry/list"
|
|
||||||
SCHEMA_WS_LIST = websocket_api.BASE_COMMAND_MESSAGE_SCHEMA.extend(
|
|
||||||
{vol.Required("type"): WS_TYPE_LIST}
|
|
||||||
)
|
|
||||||
|
|
||||||
WS_TYPE_CREATE = "config/area_registry/create"
|
|
||||||
SCHEMA_WS_CREATE = websocket_api.BASE_COMMAND_MESSAGE_SCHEMA.extend(
|
|
||||||
{vol.Required("type"): WS_TYPE_CREATE, vol.Required("name"): str}
|
|
||||||
)
|
|
||||||
|
|
||||||
WS_TYPE_DELETE = "config/area_registry/delete"
|
|
||||||
SCHEMA_WS_DELETE = websocket_api.BASE_COMMAND_MESSAGE_SCHEMA.extend(
|
|
||||||
{vol.Required("type"): WS_TYPE_DELETE, vol.Required("area_id"): str}
|
|
||||||
)
|
|
||||||
|
|
||||||
WS_TYPE_UPDATE = "config/area_registry/update"
|
|
||||||
SCHEMA_WS_UPDATE = websocket_api.BASE_COMMAND_MESSAGE_SCHEMA.extend(
|
|
||||||
{
|
|
||||||
vol.Required("type"): WS_TYPE_UPDATE,
|
|
||||||
vol.Required("area_id"): str,
|
|
||||||
vol.Required("name"): str,
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
async def async_setup(hass):
|
async def async_setup(hass):
|
||||||
"""Enable the Area Registry views."""
|
"""Enable the Area Registry views."""
|
||||||
hass.components.websocket_api.async_register_command(
|
hass.components.websocket_api.async_register_command(websocket_list_areas)
|
||||||
WS_TYPE_LIST, websocket_list_areas, SCHEMA_WS_LIST
|
hass.components.websocket_api.async_register_command(websocket_create_area)
|
||||||
)
|
hass.components.websocket_api.async_register_command(websocket_delete_area)
|
||||||
hass.components.websocket_api.async_register_command(
|
hass.components.websocket_api.async_register_command(websocket_update_area)
|
||||||
WS_TYPE_CREATE, websocket_create_area, SCHEMA_WS_CREATE
|
|
||||||
)
|
|
||||||
hass.components.websocket_api.async_register_command(
|
|
||||||
WS_TYPE_DELETE, websocket_delete_area, SCHEMA_WS_DELETE
|
|
||||||
)
|
|
||||||
hass.components.websocket_api.async_register_command(
|
|
||||||
WS_TYPE_UPDATE, websocket_update_area, SCHEMA_WS_UPDATE
|
|
||||||
)
|
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
@async_response
|
@websocket_api.websocket_command({vol.Required("type"): "config/area_registry/list"})
|
||||||
async def websocket_list_areas(hass, connection, msg):
|
@callback
|
||||||
|
def websocket_list_areas(hass, connection, msg):
|
||||||
"""Handle list areas command."""
|
"""Handle list areas command."""
|
||||||
registry = await async_get_registry(hass)
|
registry = async_get(hass)
|
||||||
connection.send_message(
|
connection.send_result(
|
||||||
websocket_api.result_message(
|
msg["id"],
|
||||||
msg["id"],
|
[_entry_dict(entry) for entry in registry.async_list_areas()],
|
||||||
[
|
|
||||||
{"name": entry.name, "area_id": entry.id}
|
|
||||||
for entry in registry.async_list_areas()
|
|
||||||
],
|
|
||||||
)
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@require_admin
|
@websocket_api.websocket_command(
|
||||||
@async_response
|
{
|
||||||
async def websocket_create_area(hass, connection, msg):
|
vol.Required("type"): "config/area_registry/create",
|
||||||
|
vol.Required("name"): str,
|
||||||
|
vol.Optional("picture"): vol.Any(str, None),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
@websocket_api.require_admin
|
||||||
|
@callback
|
||||||
|
def websocket_create_area(hass, connection, msg):
|
||||||
"""Create area command."""
|
"""Create area command."""
|
||||||
registry = await async_get_registry(hass)
|
registry = async_get(hass)
|
||||||
|
|
||||||
|
data = dict(msg)
|
||||||
|
data.pop("type")
|
||||||
|
data.pop("id")
|
||||||
|
|
||||||
try:
|
try:
|
||||||
entry = registry.async_create(msg["name"])
|
entry = registry.async_create(**data)
|
||||||
except ValueError as err:
|
except ValueError as err:
|
||||||
connection.send_message(
|
connection.send_error(msg["id"], "invalid_info", str(err))
|
||||||
websocket_api.error_message(msg["id"], "invalid_info", str(err))
|
|
||||||
)
|
|
||||||
else:
|
else:
|
||||||
connection.send_message(
|
connection.send_result(msg["id"], _entry_dict(entry))
|
||||||
websocket_api.result_message(msg["id"], _entry_dict(entry))
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
@require_admin
|
@websocket_api.websocket_command(
|
||||||
@async_response
|
{
|
||||||
async def websocket_delete_area(hass, connection, msg):
|
vol.Required("type"): "config/area_registry/delete",
|
||||||
|
vol.Required("area_id"): str,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
@websocket_api.require_admin
|
||||||
|
@callback
|
||||||
|
def websocket_delete_area(hass, connection, msg):
|
||||||
"""Delete area command."""
|
"""Delete area command."""
|
||||||
registry = await async_get_registry(hass)
|
registry = async_get(hass)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
registry.async_delete(msg["area_id"])
|
registry.async_delete(msg["area_id"])
|
||||||
except KeyError:
|
except KeyError:
|
||||||
connection.send_message(
|
connection.send_error(msg["id"], "invalid_info", "Area ID doesn't exist")
|
||||||
websocket_api.error_message(
|
|
||||||
msg["id"], "invalid_info", "Area ID doesn't exist"
|
|
||||||
)
|
|
||||||
)
|
|
||||||
else:
|
else:
|
||||||
connection.send_message(websocket_api.result_message(msg["id"], "success"))
|
connection.send_message(websocket_api.result_message(msg["id"], "success"))
|
||||||
|
|
||||||
|
|
||||||
@require_admin
|
@websocket_api.websocket_command(
|
||||||
@async_response
|
{
|
||||||
async def websocket_update_area(hass, connection, msg):
|
vol.Required("type"): "config/area_registry/update",
|
||||||
|
vol.Required("area_id"): str,
|
||||||
|
vol.Optional("name"): str,
|
||||||
|
vol.Optional("picture"): vol.Any(str, None),
|
||||||
|
}
|
||||||
|
)
|
||||||
|
@websocket_api.require_admin
|
||||||
|
@callback
|
||||||
|
def websocket_update_area(hass, connection, msg):
|
||||||
"""Handle update area websocket command."""
|
"""Handle update area websocket command."""
|
||||||
registry = await async_get_registry(hass)
|
registry = async_get(hass)
|
||||||
|
|
||||||
|
data = dict(msg)
|
||||||
|
data.pop("type")
|
||||||
|
data.pop("id")
|
||||||
|
|
||||||
try:
|
try:
|
||||||
entry = registry.async_update(msg["area_id"], msg["name"])
|
entry = registry.async_update(**data)
|
||||||
except ValueError as err:
|
except ValueError as err:
|
||||||
connection.send_message(
|
connection.send_error(msg["id"], "invalid_info", str(err))
|
||||||
websocket_api.error_message(msg["id"], "invalid_info", str(err))
|
|
||||||
)
|
|
||||||
else:
|
else:
|
||||||
connection.send_message(
|
connection.send_result(msg["id"], _entry_dict(entry))
|
||||||
websocket_api.result_message(msg["id"], _entry_dict(entry))
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def _entry_dict(entry):
|
def _entry_dict(entry):
|
||||||
"""Convert entry to API format."""
|
"""Convert entry to API format."""
|
||||||
return {"area_id": entry.id, "name": entry.name}
|
return {"area_id": entry.id, "name": entry.name, "picture": entry.picture}
|
||||||
|
@ -12,6 +12,8 @@ from homeassistant.helpers import device_registry as dr, entity_registry as er
|
|||||||
from homeassistant.loader import bind_hass
|
from homeassistant.loader import bind_hass
|
||||||
from homeassistant.util import slugify
|
from homeassistant.util import slugify
|
||||||
|
|
||||||
|
from .typing import UNDEFINED, UndefinedType
|
||||||
|
|
||||||
# mypy: disallow-any-generics
|
# mypy: disallow-any-generics
|
||||||
|
|
||||||
DATA_REGISTRY = "area_registry"
|
DATA_REGISTRY = "area_registry"
|
||||||
@ -27,6 +29,7 @@ class AreaEntry:
|
|||||||
|
|
||||||
name: str = attr.ib()
|
name: str = attr.ib()
|
||||||
normalized_name: str = attr.ib()
|
normalized_name: str = attr.ib()
|
||||||
|
picture: str | None = attr.ib(default=None)
|
||||||
id: str | None = attr.ib(default=None)
|
id: str | None = attr.ib(default=None)
|
||||||
|
|
||||||
def generate_id(self, existing_ids: Container[str]) -> None:
|
def generate_id(self, existing_ids: Container[str]) -> None:
|
||||||
@ -76,14 +79,14 @@ class AreaRegistry:
|
|||||||
return self.async_create(name)
|
return self.async_create(name)
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def async_create(self, name: str) -> AreaEntry:
|
def async_create(self, name: str, picture: str | None = None) -> AreaEntry:
|
||||||
"""Create a new area."""
|
"""Create a new area."""
|
||||||
normalized_name = normalize_area_name(name)
|
normalized_name = normalize_area_name(name)
|
||||||
|
|
||||||
if self.async_get_area_by_name(name):
|
if self.async_get_area_by_name(name):
|
||||||
raise ValueError(f"The name {name} ({normalized_name}) is already in use")
|
raise ValueError(f"The name {name} ({normalized_name}) is already in use")
|
||||||
|
|
||||||
area = AreaEntry(name=name, normalized_name=normalized_name)
|
area = AreaEntry(name=name, normalized_name=normalized_name, picture=picture)
|
||||||
area.generate_id(self.areas)
|
area.generate_id(self.areas)
|
||||||
assert area.id is not None
|
assert area.id is not None
|
||||||
self.areas[area.id] = area
|
self.areas[area.id] = area
|
||||||
@ -113,36 +116,57 @@ class AreaRegistry:
|
|||||||
self.async_schedule_save()
|
self.async_schedule_save()
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def async_update(self, area_id: str, name: str) -> AreaEntry:
|
def async_update(
|
||||||
|
self,
|
||||||
|
area_id: str,
|
||||||
|
name: str | UndefinedType = UNDEFINED,
|
||||||
|
picture: str | None | UndefinedType = UNDEFINED,
|
||||||
|
) -> AreaEntry:
|
||||||
"""Update name of area."""
|
"""Update name of area."""
|
||||||
updated = self._async_update(area_id, name)
|
updated = self._async_update(area_id, name=name, picture=picture)
|
||||||
self.hass.bus.async_fire(
|
self.hass.bus.async_fire(
|
||||||
EVENT_AREA_REGISTRY_UPDATED, {"action": "update", "area_id": area_id}
|
EVENT_AREA_REGISTRY_UPDATED, {"action": "update", "area_id": area_id}
|
||||||
)
|
)
|
||||||
return updated
|
return updated
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def _async_update(self, area_id: str, name: str) -> AreaEntry:
|
def _async_update(
|
||||||
|
self,
|
||||||
|
area_id: str,
|
||||||
|
name: str | UndefinedType = UNDEFINED,
|
||||||
|
picture: str | None | UndefinedType = UNDEFINED,
|
||||||
|
) -> AreaEntry:
|
||||||
"""Update name of area."""
|
"""Update name of area."""
|
||||||
old = self.areas[area_id]
|
old = self.areas[area_id]
|
||||||
|
|
||||||
changes = {}
|
changes = {}
|
||||||
|
|
||||||
if name == old.name:
|
if picture is not UNDEFINED:
|
||||||
|
changes["picture"] = picture
|
||||||
|
|
||||||
|
normalized_name = None
|
||||||
|
|
||||||
|
if name is not UNDEFINED:
|
||||||
|
normalized_name = normalize_area_name(name)
|
||||||
|
|
||||||
|
if normalized_name != old.normalized_name and self.async_get_area_by_name(
|
||||||
|
name
|
||||||
|
):
|
||||||
|
raise ValueError(
|
||||||
|
f"The name {name} ({normalized_name}) is already in use"
|
||||||
|
)
|
||||||
|
|
||||||
|
changes["name"] = name
|
||||||
|
changes["normalized_name"] = normalized_name
|
||||||
|
|
||||||
|
if not changes:
|
||||||
return old
|
return old
|
||||||
|
|
||||||
normalized_name = normalize_area_name(name)
|
|
||||||
|
|
||||||
if normalized_name != old.normalized_name and self.async_get_area_by_name(name):
|
|
||||||
raise ValueError(f"The name {name} ({normalized_name}) is already in use")
|
|
||||||
|
|
||||||
changes["name"] = name
|
|
||||||
changes["normalized_name"] = normalized_name
|
|
||||||
|
|
||||||
new = self.areas[area_id] = attr.evolve(old, **changes)
|
new = self.areas[area_id] = attr.evolve(old, **changes)
|
||||||
self._normalized_name_area_idx[
|
if normalized_name is not None:
|
||||||
normalized_name
|
self._normalized_name_area_idx[
|
||||||
] = self._normalized_name_area_idx.pop(old.normalized_name)
|
normalized_name
|
||||||
|
] = self._normalized_name_area_idx.pop(old.normalized_name)
|
||||||
|
|
||||||
self.async_schedule_save()
|
self.async_schedule_save()
|
||||||
return new
|
return new
|
||||||
@ -157,7 +181,11 @@ class AreaRegistry:
|
|||||||
for area in data["areas"]:
|
for area in data["areas"]:
|
||||||
normalized_name = normalize_area_name(area["name"])
|
normalized_name = normalize_area_name(area["name"])
|
||||||
areas[area["id"]] = AreaEntry(
|
areas[area["id"]] = AreaEntry(
|
||||||
name=area["name"], id=area["id"], normalized_name=normalized_name
|
name=area["name"],
|
||||||
|
id=area["id"],
|
||||||
|
# New in 2021.11
|
||||||
|
picture=area.get("picture"),
|
||||||
|
normalized_name=normalized_name,
|
||||||
)
|
)
|
||||||
self._normalized_name_area_idx[normalized_name] = area["id"]
|
self._normalized_name_area_idx[normalized_name] = area["id"]
|
||||||
|
|
||||||
@ -174,10 +202,7 @@ class AreaRegistry:
|
|||||||
data = {}
|
data = {}
|
||||||
|
|
||||||
data["areas"] = [
|
data["areas"] = [
|
||||||
{
|
{"name": entry.name, "id": entry.id, "picture": entry.picture}
|
||||||
"name": entry.name,
|
|
||||||
"id": entry.id,
|
|
||||||
}
|
|
||||||
for entry in self.areas.values()
|
for entry in self.areas.values()
|
||||||
]
|
]
|
||||||
|
|
||||||
|
@ -22,13 +22,17 @@ def registry(hass):
|
|||||||
async def test_list_areas(hass, client, registry):
|
async def test_list_areas(hass, client, registry):
|
||||||
"""Test list entries."""
|
"""Test list entries."""
|
||||||
registry.async_create("mock 1")
|
registry.async_create("mock 1")
|
||||||
registry.async_create("mock 2")
|
registry.async_create("mock 2", "/image/example.png")
|
||||||
|
|
||||||
await client.send_json({"id": 1, "type": "config/area_registry/list"})
|
await client.send_json({"id": 1, "type": "config/area_registry/list"})
|
||||||
|
|
||||||
msg = await client.receive_json()
|
msg = await client.receive_json()
|
||||||
|
|
||||||
assert len(msg["result"]) == len(registry.areas)
|
assert len(msg["result"]) == len(registry.areas)
|
||||||
|
assert msg["result"][0]["name"] == "mock 1"
|
||||||
|
assert msg["result"][0]["picture"] is None
|
||||||
|
assert msg["result"][1]["name"] == "mock 2"
|
||||||
|
assert msg["result"][1]["picture"] == "/image/example.png"
|
||||||
|
|
||||||
|
|
||||||
async def test_create_area(hass, client, registry):
|
async def test_create_area(hass, client, registry):
|
||||||
@ -98,6 +102,7 @@ async def test_update_area(hass, client, registry):
|
|||||||
"id": 1,
|
"id": 1,
|
||||||
"area_id": area.id,
|
"area_id": area.id,
|
||||||
"name": "mock 2",
|
"name": "mock 2",
|
||||||
|
"picture": "/image/example.png",
|
||||||
"type": "config/area_registry/update",
|
"type": "config/area_registry/update",
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
@ -106,6 +111,23 @@ async def test_update_area(hass, client, registry):
|
|||||||
|
|
||||||
assert msg["result"]["area_id"] == area.id
|
assert msg["result"]["area_id"] == area.id
|
||||||
assert msg["result"]["name"] == "mock 2"
|
assert msg["result"]["name"] == "mock 2"
|
||||||
|
assert msg["result"]["picture"] == "/image/example.png"
|
||||||
|
assert len(registry.areas) == 1
|
||||||
|
|
||||||
|
await client.send_json(
|
||||||
|
{
|
||||||
|
"id": 2,
|
||||||
|
"area_id": area.id,
|
||||||
|
"picture": None,
|
||||||
|
"type": "config/area_registry/update",
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
msg = await client.receive_json()
|
||||||
|
|
||||||
|
assert msg["result"]["area_id"] == area.id
|
||||||
|
assert msg["result"]["name"] == "mock 2"
|
||||||
|
assert msg["result"]["picture"] is None
|
||||||
assert len(registry.areas) == 1
|
assert len(registry.areas) == 1
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user