Skip to content

Commit 2546228

Browse files
Lash-Lallenporter
andauthored
Add Roborock entity with the name of the current room (#140895)
* Add current room entity * Update homeassistant/components/roborock/models.py Co-authored-by: Allen Porter <allen.porter@gmail.com> * Update homeassistant/components/roborock/models.py Co-authored-by: Allen Porter <allen.porter@gmail.com> * use current_room property * remove select changes --------- Co-authored-by: Allen Porter <allen.porter@gmail.com>
1 parent c41d5f2 commit 2546228

File tree

6 files changed

+77
-13
lines changed

6 files changed

+77
-13
lines changed

homeassistant/components/roborock/coordinator.py

+13-8
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,7 @@
2929
from vacuum_map_parser_base.config.color import ColorsPalette
3030
from vacuum_map_parser_base.config.image_config import ImageConfig
3131
from vacuum_map_parser_base.config.size import Sizes
32+
from vacuum_map_parser_base.map_data import MapData
3233
from vacuum_map_parser_roborock.map_data_parser import RoborockMapDataParser
3334

3435
from homeassistant.config_entries import ConfigEntry
@@ -168,18 +169,20 @@ def dock_device_info(self) -> DeviceInfo:
168169
sw_version=self.roborock_device_info.device.fv,
169170
)
170171

171-
def parse_image(self, map_bytes: bytes) -> bytes | None:
172-
"""Parse map_bytes and store it as image bytes."""
172+
def parse_map_data_v1(
173+
self, map_bytes: bytes
174+
) -> tuple[bytes | None, MapData | None]:
175+
"""Parse map_bytes and return MapData and the image."""
173176
try:
174177
parsed_map = self.map_parser.parse(map_bytes)
175178
except (IndexError, ValueError) as err:
176179
_LOGGER.debug("Exception when parsing map contents: %s", err)
177-
return None
180+
return None, None
178181
if parsed_map.image is None:
179-
return None
182+
return None, None
180183
img_byte_arr = io.BytesIO()
181184
parsed_map.image.data.save(img_byte_arr, format=MAP_FILE_FORMAT)
182-
return img_byte_arr.getvalue()
185+
return img_byte_arr.getvalue(), parsed_map
183186

184187
async def _async_setup(self) -> None:
185188
"""Set up the coordinator."""
@@ -206,6 +209,7 @@ async def _async_setup(self) -> None:
206209
rooms={},
207210
image=image,
208211
last_updated=dt_util.utcnow() - IMAGE_CACHE_INTERVAL,
212+
map_data=None,
209213
)
210214
for image, roborock_map in zip(stored_images, roborock_maps, strict=False)
211215
}
@@ -230,20 +234,21 @@ async def update_map(self) -> None:
230234
translation_domain=DOMAIN,
231235
translation_key="map_failure",
232236
)
233-
parsed_image = self.parse_image(response)
234-
if parsed_image is None:
237+
parsed_image, parsed_map = self.parse_map_data_v1(response)
238+
if parsed_image is None or parsed_map is None:
235239
raise HomeAssistantError(
236240
translation_domain=DOMAIN,
237241
translation_key="map_failure",
238242
)
243+
current_roborock_map_info = self.maps[self.current_map]
239244
if parsed_image != self.maps[self.current_map].image:
240245
await self.map_storage.async_save_map(
241246
self.current_map,
242247
parsed_image,
243248
)
244-
current_roborock_map_info = self.maps[self.current_map]
245249
current_roborock_map_info.image = parsed_image
246250
current_roborock_map_info.last_updated = dt_util.utcnow()
251+
current_roborock_map_info.map_data = parsed_map
247252

248253
async def _verify_api(self) -> None:
249254
"""Verify that the api is reachable. If it is not, switch clients."""

homeassistant/components/roborock/models.py

+9
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@
66

77
from roborock.containers import HomeDataDevice, HomeDataProduct, NetworkInfo
88
from roborock.roborock_typing import DeviceProp
9+
from vacuum_map_parser_base.map_data import MapData
910

1011

1112
@dataclass
@@ -51,3 +52,11 @@ class RoborockMapInfo:
5152
rooms: dict[int, str]
5253
image: bytes | None
5354
last_updated: datetime
55+
map_data: MapData | None
56+
57+
@property
58+
def current_room(self) -> str | None:
59+
"""Get the currently active room for this map if any."""
60+
if self.map_data is None or self.map_data.vacuum_room is None:
61+
return None
62+
return self.rooms.get(self.map_data.vacuum_room)

homeassistant/components/roborock/sensor.py

+46-4
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,11 @@
3636
RoborockDataUpdateCoordinator,
3737
RoborockDataUpdateCoordinatorA01,
3838
)
39-
from .entity import RoborockCoordinatedEntityA01, RoborockCoordinatedEntityV1
39+
from .entity import (
40+
RoborockCoordinatedEntityA01,
41+
RoborockCoordinatedEntityV1,
42+
RoborockEntity,
43+
)
4044

4145
PARALLEL_UPDATES = 0
4246

@@ -306,16 +310,17 @@ async def async_setup_entry(
306310
) -> None:
307311
"""Set up the Roborock vacuum sensors."""
308312
coordinators = config_entry.runtime_data
309-
async_add_entities(
313+
entities: list[RoborockEntity] = [
310314
RoborockSensorEntity(
311315
coordinator,
312316
description,
313317
)
314318
for coordinator in coordinators.v1
315319
for description in SENSOR_DESCRIPTIONS
316320
if description.value_fn(coordinator.roborock_device_info.props) is not None
317-
)
318-
async_add_entities(
321+
]
322+
entities.extend(RoborockCurrentRoom(coordinator) for coordinator in coordinators.v1)
323+
entities.extend(
319324
RoborockSensorEntityA01(
320325
coordinator,
321326
description,
@@ -324,6 +329,7 @@ async def async_setup_entry(
324329
for description in A01_SENSOR_DESCRIPTIONS
325330
if description.data_protocol in coordinator.data
326331
)
332+
async_add_entities(entities)
327333

328334

329335
class RoborockSensorEntity(RoborockCoordinatedEntityV1, SensorEntity):
@@ -353,6 +359,42 @@ def native_value(self) -> StateType | datetime.datetime:
353359
)
354360

355361

362+
class RoborockCurrentRoom(RoborockCoordinatedEntityV1, SensorEntity):
363+
"""Representation of a Current Room Sensor."""
364+
365+
_attr_device_class = SensorDeviceClass.ENUM
366+
_attr_translation_key = "current_room"
367+
_attr_entity_category = EntityCategory.DIAGNOSTIC
368+
369+
def __init__(
370+
self,
371+
coordinator: RoborockDataUpdateCoordinator,
372+
) -> None:
373+
"""Initialize the entity."""
374+
super().__init__(
375+
f"current_room_{coordinator.duid_slug}",
376+
coordinator,
377+
None,
378+
is_dock_entity=False,
379+
)
380+
381+
@property
382+
def options(self) -> list[str]:
383+
"""Return the currently valid rooms."""
384+
if self.coordinator.current_map is not None:
385+
return list(
386+
self.coordinator.maps[self.coordinator.current_map].rooms.values()
387+
)
388+
return []
389+
390+
@property
391+
def native_value(self) -> str | None:
392+
"""Return the value reported by the sensor."""
393+
if self.coordinator.current_map is not None:
394+
return self.coordinator.maps[self.coordinator.current_map].current_room
395+
return None
396+
397+
356398
class RoborockSensorEntityA01(RoborockCoordinatedEntityA01, SensorEntity):
357399
"""Representation of a A01 Roborock sensor."""
358400

homeassistant/components/roborock/strings.json

+3
Original file line numberDiff line numberDiff line change
@@ -181,6 +181,9 @@
181181
"countdown": {
182182
"name": "Countdown"
183183
},
184+
"current_room": {
185+
"name": "Current room"
186+
},
184187
"dock_error": {
185188
"name": "Dock error",
186189
"state": {

tests/components/roborock/mock_data.py

+1
Original file line numberDiff line numberDiff line change
@@ -1151,6 +1151,7 @@
11511151
MAP_DATA.image = ImageData(
11521152
100, 10, 10, 10, 10, ImageConfig(), Image.new("RGB", (1, 1)), lambda p: p
11531153
)
1154+
MAP_DATA.vacuum_room = 17
11541155

11551156

11561157
SCENES = [

tests/components/roborock/test_sensor.py

+5-1
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,7 @@ def platforms() -> list[Platform]:
2929

3030
async def test_sensors(hass: HomeAssistant, setup_entry: MockConfigEntry) -> None:
3131
"""Test sensors and check test values are correctly set."""
32-
assert len(hass.states.async_all("sensor")) == 40
32+
assert len(hass.states.async_all("sensor")) == 42
3333
assert hass.states.get("sensor.roborock_s7_maxv_main_brush_time_left").state == str(
3434
MAIN_BRUSH_REPLACE_TIME - 74382
3535
)
@@ -63,6 +63,10 @@ async def test_sensors(hass: HomeAssistant, setup_entry: MockConfigEntry) -> Non
6363
hass.states.get("sensor.roborock_s7_maxv_last_clean_end").state
6464
== "2023-01-01T03:43:58+00:00"
6565
)
66+
assert (
67+
hass.states.get("sensor.roborock_s7_maxv_current_room").state
68+
== "Example room 2"
69+
)
6670
assert hass.states.get("sensor.dyad_pro_status").state == "drying"
6771
assert hass.states.get("sensor.dyad_pro_battery").state == "100"
6872
assert hass.states.get("sensor.dyad_pro_filter_time_left").state == "111"

0 commit comments

Comments
 (0)