|
39 | 39 | from homeassistant.helpers.device_registry import DeviceInfo
|
40 | 40 | from homeassistant.helpers.typing import StateType
|
41 | 41 | from homeassistant.helpers.update_coordinator import DataUpdateCoordinator, UpdateFailed
|
42 |
| -from homeassistant.util import slugify |
| 42 | +from homeassistant.util import dt as dt_util, slugify |
43 | 43 |
|
44 | 44 | from .const import (
|
45 | 45 | A01_UPDATE_INTERVAL,
|
46 | 46 | DEFAULT_DRAWABLES,
|
47 | 47 | DOMAIN,
|
48 | 48 | DRAWABLES,
|
| 49 | + IMAGE_CACHE_INTERVAL, |
49 | 50 | MAP_FILE_FORMAT,
|
50 | 51 | MAP_SCALE,
|
51 | 52 | MAP_SLEEP,
|
@@ -191,15 +192,59 @@ async def _async_setup(self) -> None:
|
191 | 192 | except RoborockException as err:
|
192 | 193 | raise UpdateFailed("Failed to get map data: {err}") from err
|
193 | 194 | # Rooms names populated later with calls to `set_current_map_rooms` for each map
|
| 195 | + roborock_maps = maps.map_info if (maps and maps.map_info) else () |
| 196 | + stored_images = await asyncio.gather( |
| 197 | + *[ |
| 198 | + self.map_storage.async_load_map(roborock_map.mapFlag) |
| 199 | + for roborock_map in roborock_maps |
| 200 | + ] |
| 201 | + ) |
194 | 202 | self.maps = {
|
195 | 203 | roborock_map.mapFlag: RoborockMapInfo(
|
196 | 204 | flag=roborock_map.mapFlag,
|
197 | 205 | name=roborock_map.name or f"Map {roborock_map.mapFlag}",
|
198 | 206 | rooms={},
|
| 207 | + image=image, |
| 208 | + last_updated=dt_util.utcnow() - IMAGE_CACHE_INTERVAL, |
199 | 209 | )
|
200 |
| - for roborock_map in (maps.map_info if (maps and maps.map_info) else ()) |
| 210 | + for image, roborock_map in zip(stored_images, roborock_maps, strict=False) |
201 | 211 | }
|
202 | 212 |
|
| 213 | + async def update_map(self) -> None: |
| 214 | + """Update the currently selected map.""" |
| 215 | + # The current map was set in the props update, so these can be done without |
| 216 | + # worry of applying them to the wrong map. |
| 217 | + if self.current_map is None: |
| 218 | + # This exists as a safeguard/ to keep mypy happy. |
| 219 | + return |
| 220 | + try: |
| 221 | + response = await self.cloud_api.get_map_v1() |
| 222 | + except RoborockException as ex: |
| 223 | + raise HomeAssistantError( |
| 224 | + translation_domain=DOMAIN, |
| 225 | + translation_key="map_failure", |
| 226 | + ) from ex |
| 227 | + if not isinstance(response, bytes): |
| 228 | + _LOGGER.debug("Failed to parse map contents: %s", response) |
| 229 | + raise HomeAssistantError( |
| 230 | + translation_domain=DOMAIN, |
| 231 | + translation_key="map_failure", |
| 232 | + ) |
| 233 | + parsed_image = self.parse_image(response) |
| 234 | + if parsed_image is None: |
| 235 | + raise HomeAssistantError( |
| 236 | + translation_domain=DOMAIN, |
| 237 | + translation_key="map_failure", |
| 238 | + ) |
| 239 | + if parsed_image != self.maps[self.current_map].image: |
| 240 | + await self.map_storage.async_save_map( |
| 241 | + self.current_map, |
| 242 | + parsed_image, |
| 243 | + ) |
| 244 | + current_roborock_map_info = self.maps[self.current_map] |
| 245 | + current_roborock_map_info.image = parsed_image |
| 246 | + current_roborock_map_info.last_updated = dt_util.utcnow() |
| 247 | + |
203 | 248 | async def _verify_api(self) -> None:
|
204 | 249 | """Verify that the api is reachable. If it is not, switch clients."""
|
205 | 250 | if isinstance(self.api, RoborockLocalClientV1):
|
@@ -240,6 +285,19 @@ async def _async_update_data(self) -> DeviceProp:
|
240 | 285 | # Set the new map id from the updated device props
|
241 | 286 | self._set_current_map()
|
242 | 287 | # Get the rooms for that map id.
|
| 288 | + |
| 289 | + # If the vacuum is currently cleaning and it has been IMAGE_CACHE_INTERVAL |
| 290 | + # since the last map update, you can update the map. |
| 291 | + if ( |
| 292 | + self.current_map is not None |
| 293 | + and self.roborock_device_info.props.status.in_cleaning |
| 294 | + and (dt_util.utcnow() - self.maps[self.current_map].last_updated) |
| 295 | + > IMAGE_CACHE_INTERVAL |
| 296 | + ): |
| 297 | + try: |
| 298 | + await self.update_map() |
| 299 | + except HomeAssistantError as err: |
| 300 | + _LOGGER.debug("Failed to update map: %s", err) |
243 | 301 | await self.set_current_map_rooms()
|
244 | 302 | except RoborockException as ex:
|
245 | 303 | _LOGGER.debug("Failed to update data: %s", ex)
|
@@ -338,7 +396,14 @@ async def refresh_coordinator_map(self) -> None:
|
338 | 396 | # We cannot get the map until the roborock servers fully process the
|
339 | 397 | # map change.
|
340 | 398 | await asyncio.sleep(MAP_SLEEP)
|
341 |
| - await self.set_current_map_rooms() |
| 399 | + tasks = [self.set_current_map_rooms()] |
| 400 | + # The image is set within async_setup, so if it exists, we have it here. |
| 401 | + if self.maps[map_flag].image is None: |
| 402 | + # If we don't have a cached map, let's update it here so that it can be |
| 403 | + # cached in the future. |
| 404 | + tasks.append(self.update_map()) |
| 405 | + # If either of these fail, we don't care, and we want to continue. |
| 406 | + await asyncio.gather(*tasks, return_exceptions=True) |
342 | 407 |
|
343 | 408 | if len(self.maps) != 1:
|
344 | 409 | # Set the map back to the map the user previously had selected so that it
|
|
0 commit comments