Allow storing picture in area registry (#58539)

This commit is contained in:
Paulus Schoutsen 2021-10-27 10:11:05 -07:00 committed by GitHub
parent f228537458
commit 6cd83d1f66
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 132 additions and 107 deletions

View File

@ -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}

View File

@ -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()
] ]

View File

@ -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