From 1b8696ef34d4e7c5fdf1da0904d808decf6e9623 Mon Sep 17 00:00:00 2001 From: David Pace Date: Sun, 18 Aug 2024 15:41:17 +0200 Subject: [PATCH 1/3] [boschshc] Update location properties when initializing things When a device is initialized, an attempt is made to look up the room name for the room id specified for the device and to store the room name in a thing property. This property is also updated if a room change is detected. The legacy property `Location` is removed if present. From now on the property `location` (with proper lower case spelling) is used. * add constants for location properties * implement location updates in abstract device handler * extend bridge handler to provide a cached list of rooms * add unit tests Signed-off-by: David Pace --- .../devices/BoschSHCBindingConstants.java | 4 ++ .../devices/BoschSHCDeviceHandler.java | 39 +++++++++++ .../devices/bridge/BridgeHandler.java | 33 ++++++++- .../discovery/ThingDiscoveryService.java | 2 +- .../AbstractBoschSHCDeviceHandlerTest.java | 69 +++++++++++++++++++ .../devices/relay/RelayHandlerTest.java | 54 +++++++++++++++ .../discovery/ThingDiscoveryServiceTest.java | 3 +- 7 files changed, 201 insertions(+), 3 deletions(-) diff --git a/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/devices/BoschSHCBindingConstants.java b/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/devices/BoschSHCBindingConstants.java index f48a18bb906..4b803683fde 100644 --- a/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/devices/BoschSHCBindingConstants.java +++ b/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/devices/BoschSHCBindingConstants.java @@ -128,4 +128,8 @@ public class BoschSHCBindingConstants { // static device/service names public static final String SERVICE_INTRUSION_DETECTION = "intrusionDetectionSystem"; + + // thing properties + public static final String PROPERTY_LOCATION_LEGACY = "Location"; + public static final String PROPERTY_LOCATION = "location"; } diff --git a/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/devices/BoschSHCDeviceHandler.java b/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/devices/BoschSHCDeviceHandler.java index 6791b19c927..912543900f2 100644 --- a/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/devices/BoschSHCDeviceHandler.java +++ b/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/devices/BoschSHCDeviceHandler.java @@ -12,6 +12,10 @@ */ package org.openhab.binding.boschshc.internal.devices; +import static org.openhab.binding.boschshc.internal.devices.BoschSHCBindingConstants.PROPERTY_LOCATION; +import static org.openhab.binding.boschshc.internal.devices.BoschSHCBindingConstants.PROPERTY_LOCATION_LEGACY; + +import java.util.Map; import java.util.concurrent.ExecutionException; import java.util.concurrent.TimeoutException; @@ -83,9 +87,44 @@ public abstract class BoschSHCDeviceHandler extends BoschSHCHandler { * otherwise */ protected boolean processDeviceInfo(Device deviceInfo) { + try { + updateLocationPropertiesIfApplicable(deviceInfo); + } catch (Exception e) { + logger.warn("Error while updating location properties for thing {}.", getThing().getUID(), e); + } + // do not cancel thing initialization if location properties cannot be updated return true; } + private void updateLocationPropertiesIfApplicable(Device deviceInfo) throws BoschSHCException { + Map thingProperties = getThing().getProperties(); + removeLegacyLocationPropertyIfApplicable(thingProperties); + updateLocationPropertyIfApplicable(thingProperties, deviceInfo); + } + + private void updateLocationPropertyIfApplicable(Map thingProperties, Device deviceInfo) + throws BoschSHCException { + @Nullable + String roomName = getBridgeHandler().resolveRoomId(deviceInfo.roomId); + if (roomName != null) { + @Nullable + String currentLocation = thingProperties.get(PROPERTY_LOCATION); + if (currentLocation == null || !currentLocation.equals(roomName)) { + logger.debug("Updating property '{}' of thing {} to '{}'.", PROPERTY_LOCATION, getThing().getUID(), + roomName); + updateProperty(PROPERTY_LOCATION, roomName); + } + } + } + + private void removeLegacyLocationPropertyIfApplicable(Map thingProperties) { + if (thingProperties.containsKey(PROPERTY_LOCATION_LEGACY)) { + logger.debug("Removing legacy property '{}' from thing {}.", PROPERTY_LOCATION_LEGACY, getThing().getUID()); + // null value indicates that the property should be removed + updateProperty(PROPERTY_LOCATION_LEGACY, null); + } + } + /** * Attempts to obtain information about the device with the specified ID via a REST call. *

diff --git a/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/devices/bridge/BridgeHandler.java b/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/devices/bridge/BridgeHandler.java index 9575de3c955..56e0246e504 100644 --- a/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/devices/bridge/BridgeHandler.java +++ b/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/devices/bridge/BridgeHandler.java @@ -17,6 +17,7 @@ import static org.eclipse.jetty.http.HttpMethod.POST; import static org.eclipse.jetty.http.HttpMethod.PUT; import java.lang.reflect.Type; +import java.time.Duration; import java.util.ArrayList; import java.util.Collection; import java.util.Collections; @@ -54,6 +55,7 @@ import org.openhab.binding.boschshc.internal.exceptions.PairingFailedException; import org.openhab.binding.boschshc.internal.serialization.GsonUtils; import org.openhab.binding.boschshc.internal.services.dto.BoschSHCServiceState; import org.openhab.binding.boschshc.internal.services.dto.JsonRestExceptionResponse; +import org.openhab.core.cache.ExpiringCache; import org.openhab.core.library.types.StringType; import org.openhab.core.thing.Bridge; import org.openhab.core.thing.Channel; @@ -88,6 +90,8 @@ public class BridgeHandler extends BaseBridgeHandler { private static final String HTTP_CLIENT_NOT_INITIALIZED = "HttpClient not initialized"; + private static final Duration ROOM_CACHE_DURATION = Duration.ofMinutes(2); + private final Logger logger = LoggerFactory.getLogger(BridgeHandler.class); /** @@ -107,13 +111,22 @@ public class BridgeHandler extends BaseBridgeHandler { /** * SHC thing/device discovery service instance. - * Registered and unregistered if service is actived/deactived. + * Registered and unregistered if service is activated/deactivated. * Used to scan for things after bridge is paired with SHC. */ private @Nullable ThingDiscoveryService thingDiscoveryService; private final ScenarioHandler scenarioHandler; + private ExpiringCache> roomCache = new ExpiringCache<>(ROOM_CACHE_DURATION, () -> { + try { + return getRooms(); + } catch (InterruptedException e) { + Thread.currentThread().interrupt(); + return null; + } + }); + public BridgeHandler(Bridge bridge) { super(bridge); scenarioHandler = new ScenarioHandler(); @@ -437,6 +450,24 @@ public class BridgeHandler extends BaseBridgeHandler { } } + public @Nullable List getRoomsWithCache() { + return roomCache.getValue(); + } + + public @Nullable String resolveRoomId(@Nullable String roomId) { + if (roomId == null) { + return null; + } + + @Nullable + List rooms = getRoomsWithCache(); + if (rooms != null) { + return rooms.stream().filter(r -> r.id.equals(roomId)).map(r -> r.name).findAny().orElse(null); + } + + return null; + } + /** * Get public information from Bosch SHC. */ diff --git a/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/discovery/ThingDiscoveryService.java b/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/discovery/ThingDiscoveryService.java index 6f5bb912cca..42ad3e2ff3d 100644 --- a/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/discovery/ThingDiscoveryService.java +++ b/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/discovery/ThingDiscoveryService.java @@ -242,7 +242,7 @@ public class ThingDiscoveryService extends AbstractThingHandlerDiscoveryService< discoveryResult.withBridge(thingHandler.getThing().getUID()); if (!roomName.isEmpty()) { - discoveryResult.withProperty("Location", roomName); + discoveryResult.withProperty(BoschSHCBindingConstants.PROPERTY_LOCATION, roomName); } thingDiscovered(discoveryResult.build()); diff --git a/bundles/org.openhab.binding.boschshc/src/test/java/org/openhab/binding/boschshc/internal/devices/AbstractBoschSHCDeviceHandlerTest.java b/bundles/org.openhab.binding.boschshc/src/test/java/org/openhab/binding/boschshc/internal/devices/AbstractBoschSHCDeviceHandlerTest.java index 70a178c3578..70918ab8863 100644 --- a/bundles/org.openhab.binding.boschshc/src/test/java/org/openhab/binding/boschshc/internal/devices/AbstractBoschSHCDeviceHandlerTest.java +++ b/bundles/org.openhab.binding.boschshc/src/test/java/org/openhab/binding/boschshc/internal/devices/AbstractBoschSHCDeviceHandlerTest.java @@ -14,14 +14,20 @@ package org.openhab.binding.boschshc.internal.devices; import static org.mockito.ArgumentMatchers.any; import static org.mockito.ArgumentMatchers.argThat; +import static org.mockito.Mockito.times; import static org.mockito.Mockito.verify; import static org.mockito.Mockito.when; +import java.util.HashMap; +import java.util.Map; +import java.util.Set; import java.util.concurrent.ExecutionException; import java.util.concurrent.TimeoutException; import org.eclipse.jdt.annotation.NonNullByDefault; +import org.junit.jupiter.api.Tag; import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.TestInfo; import org.junit.jupiter.params.ParameterizedTest; import org.junit.jupiter.params.provider.MethodSource; import org.openhab.binding.boschshc.internal.devices.bridge.dto.Device; @@ -42,11 +48,34 @@ import org.openhab.core.thing.ThingStatusDetail; public abstract class AbstractBoschSHCDeviceHandlerTest extends AbstractBoschSHCHandlerTest { + protected static final String TAG_LEGACY_LOCATION_PROPERTY = "LegacyLocationProperty"; + protected static final String TAG_LOCATION_PROPERTY = "LocationProperty"; + protected static final String DEFAULT_ROOM_ID = "hz_1"; + @Override protected void configureDevice(Device device) { super.configureDevice(device); device.id = getDeviceID(); + device.roomId = DEFAULT_ROOM_ID; + } + + @Override + protected void beforeHandlerInitialization(TestInfo testInfo) { + super.beforeHandlerInitialization(testInfo); + Set tags = testInfo.getTags(); + if (tags.contains(TAG_LEGACY_LOCATION_PROPERTY) || tags.contains(TAG_LOCATION_PROPERTY)) { + Map properties = new HashMap<>(); + when(getThing().getProperties()).thenReturn(properties); + + if (tags.contains(TAG_LEGACY_LOCATION_PROPERTY)) { + properties.put(BoschSHCBindingConstants.PROPERTY_LOCATION_LEGACY, "Living Room"); + } + + if (tags.contains(TAG_LOCATION_PROPERTY)) { + when(getBridgeHandler().resolveRoomId(DEFAULT_ROOM_ID)).thenReturn("Kitchen"); + } + } } @Override @@ -80,4 +109,44 @@ public abstract class AbstractBoschSHCDeviceHandlerTest status.getStatus().equals(ThingStatus.OFFLINE) && status.getStatusDetail().equals(ThingStatusDetail.CONFIGURATION_ERROR))); } + + @Tag(TAG_LEGACY_LOCATION_PROPERTY) + @Test + protected void deleteLegacyLocationProperty() { + verify(getThing()).setProperty(BoschSHCBindingConstants.PROPERTY_LOCATION_LEGACY, null); + verify(getCallback()).thingUpdated(getThing()); + } + + @Tag(TAG_LOCATION_PROPERTY) + @Test + protected void locationPropertyDidNotChange() { + verify(getThing()).setProperty(BoschSHCBindingConstants.PROPERTY_LOCATION, "Kitchen"); + verify(getCallback()).thingUpdated(getThing()); + + getThing().getProperties().put(BoschSHCBindingConstants.PROPERTY_LOCATION, "Kitchen"); + + // re-initialize + getFixture().initialize(); + + verify(getThing()).setProperty(BoschSHCBindingConstants.PROPERTY_LOCATION, "Kitchen"); + verify(getCallback()).thingUpdated(getThing()); + } + + @Tag(TAG_LOCATION_PROPERTY) + @Test + protected void locationPropertyDidChange() { + verify(getThing()).setProperty(BoschSHCBindingConstants.PROPERTY_LOCATION, "Kitchen"); + verify(getCallback()).thingUpdated(getThing()); + + getThing().getProperties().put(BoschSHCBindingConstants.PROPERTY_LOCATION, "Kitchen"); + + getDevice().roomId = "hz_2"; + when(getBridgeHandler().resolveRoomId("hz_2")).thenReturn("Dining Room"); + + // re-initialize + getFixture().initialize(); + + verify(getThing()).setProperty(BoschSHCBindingConstants.PROPERTY_LOCATION, "Dining Room"); + verify(getCallback(), times(2)).thingUpdated(getThing()); + } } diff --git a/bundles/org.openhab.binding.boschshc/src/test/java/org/openhab/binding/boschshc/internal/devices/relay/RelayHandlerTest.java b/bundles/org.openhab.binding.boschshc/src/test/java/org/openhab/binding/boschshc/internal/devices/relay/RelayHandlerTest.java index 8524a104223..92f88b77510 100644 --- a/bundles/org.openhab.binding.boschshc/src/test/java/org/openhab/binding/boschshc/internal/devices/relay/RelayHandlerTest.java +++ b/bundles/org.openhab.binding.boschshc/src/test/java/org/openhab/binding/boschshc/internal/devices/relay/RelayHandlerTest.java @@ -12,8 +12,11 @@ */ package org.openhab.binding.boschshc.internal.devices.relay; +import static org.hamcrest.CoreMatchers.not; import static org.hamcrest.MatcherAssert.assertThat; import static org.hamcrest.Matchers.is; +import static org.hamcrest.collection.IsCollectionWithSize.hasSize; +import static org.hamcrest.collection.IsMapContaining.hasKey; import static org.mockito.ArgumentMatchers.any; import static org.mockito.ArgumentMatchers.argThat; import static org.mockito.ArgumentMatchers.eq; @@ -376,4 +379,55 @@ class RelayHandlerTest extends AbstractPowerSwitchHandlerTest { verify(getCallback(), times(2)).thingUpdated(argThat(t -> ImpulseSwitchService.IMPULSE_SWITCH_SERVICE_NAME .equals(t.getProperties().get(RelayHandler.PROPERTY_MODE)))); } + + /** + * This has to be tested differently for the RelayHandler because the thing mock + * will be replaced by a real thing during the first initialization, which + * modifies the channels. + */ + @Test + @Tag(TAG_LEGACY_LOCATION_PROPERTY) + @Override + protected void deleteLegacyLocationProperty() { + ArgumentCaptor thingCaptor = ArgumentCaptor.forClass(Thing.class); + verify(getCallback(), times(3)).thingUpdated(thingCaptor.capture()); + List allValues = thingCaptor.getAllValues(); + assertThat(allValues, hasSize(3)); + assertThat(allValues.get(2).getProperties(), not(hasKey(BoschSHCBindingConstants.PROPERTY_LOCATION_LEGACY))); + } + + /** + * This has to be tested differently for the RelayHandler because the thing mock + * will be replaced by a real thing during the first initialization, which + * modifies the channels. + */ + @Test + @Tag(TAG_LOCATION_PROPERTY) + @Override + protected void locationPropertyDidNotChange() { + // re-initialize + getFixture().initialize(); + + verify(getCallback(), times(3)).thingUpdated( + argThat(t -> t.getProperties().get(BoschSHCBindingConstants.PROPERTY_LOCATION).equals("Kitchen"))); + } + + /** + * This has to be tested differently for the RelayHandler because the thing mock + * will be replaced by a real thing during the first initialization, which + * modifies the channels. + */ + @Test + @Tag(TAG_LOCATION_PROPERTY) + @Override + protected void locationPropertyDidChange() { + getDevice().roomId = "hz_2"; + when(getBridgeHandler().resolveRoomId("hz_2")).thenReturn("Dining Room"); + + // re-initialize + getFixture().initialize(); + + verify(getCallback(), times(4)).thingUpdated( + argThat(t -> t.getProperties().get(BoschSHCBindingConstants.PROPERTY_LOCATION).equals("Dining Room"))); + } } diff --git a/bundles/org.openhab.binding.boschshc/src/test/java/org/openhab/binding/boschshc/internal/discovery/ThingDiscoveryServiceTest.java b/bundles/org.openhab.binding.boschshc/src/test/java/org/openhab/binding/boschshc/internal/discovery/ThingDiscoveryServiceTest.java index cc5f5d58b4e..2ec461306f5 100644 --- a/bundles/org.openhab.binding.boschshc/src/test/java/org/openhab/binding/boschshc/internal/discovery/ThingDiscoveryServiceTest.java +++ b/bundles/org.openhab.binding.boschshc/src/test/java/org/openhab/binding/boschshc/internal/discovery/ThingDiscoveryServiceTest.java @@ -194,7 +194,8 @@ class ThingDiscoveryServiceTest { assertThat(result.getThingUID().getId(), is("testDevice_ID")); assertThat(result.getBridgeUID().getId(), is("testSHC")); assertThat(result.getLabel(), is("Test Name")); - assertThat(String.valueOf(result.getProperties().get("Location")), is("TestRoom")); + assertThat(String.valueOf(result.getProperties().get(BoschSHCBindingConstants.PROPERTY_LOCATION)), + is("TestRoom")); } @Test From 9aa64a2972e267e3d58a781c595bb1f890731d50 Mon Sep 17 00:00:00 2001 From: David Pace Date: Mon, 6 Jan 2025 21:01:16 +0100 Subject: [PATCH 2/3] Apply suggestions from code review Co-authored-by: Jacob Laursen Signed-off-by: David Pace --- .../boschshc/internal/devices/BoschSHCDeviceHandler.java | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/devices/BoschSHCDeviceHandler.java b/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/devices/BoschSHCDeviceHandler.java index 912543900f2..e3c8f1f0cb2 100644 --- a/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/devices/BoschSHCDeviceHandler.java +++ b/bundles/org.openhab.binding.boschshc/src/main/java/org/openhab/binding/boschshc/internal/devices/BoschSHCDeviceHandler.java @@ -89,7 +89,7 @@ public abstract class BoschSHCDeviceHandler extends BoschSHCHandler { protected boolean processDeviceInfo(Device deviceInfo) { try { updateLocationPropertiesIfApplicable(deviceInfo); - } catch (Exception e) { + } catch (BoschSHCException e) { logger.warn("Error while updating location properties for thing {}.", getThing().getUID(), e); } // do not cancel thing initialization if location properties cannot be updated @@ -104,12 +104,10 @@ public abstract class BoschSHCDeviceHandler extends BoschSHCHandler { private void updateLocationPropertyIfApplicable(Map thingProperties, Device deviceInfo) throws BoschSHCException { - @Nullable String roomName = getBridgeHandler().resolveRoomId(deviceInfo.roomId); if (roomName != null) { - @Nullable String currentLocation = thingProperties.get(PROPERTY_LOCATION); - if (currentLocation == null || !currentLocation.equals(roomName)) { + if (!roomName.equals(currentLocation)) { logger.debug("Updating property '{}' of thing {} to '{}'.", PROPERTY_LOCATION, getThing().getUID(), roomName); updateProperty(PROPERTY_LOCATION, roomName); From 9a5b8f9d7c64bd8190d7642fe4c4955a0db8a6cc Mon Sep 17 00:00:00 2001 From: David Pace Date: Mon, 6 Jan 2025 21:28:18 +0100 Subject: [PATCH 3/3] add unit test for resolveRoomId() Signed-off-by: David Pace --- .../devices/bridge/BridgeHandlerTest.java | 31 +++++++++++++++++++ 1 file changed, 31 insertions(+) diff --git a/bundles/org.openhab.binding.boschshc/src/test/java/org/openhab/binding/boschshc/internal/devices/bridge/BridgeHandlerTest.java b/bundles/org.openhab.binding.boschshc/src/test/java/org/openhab/binding/boschshc/internal/devices/bridge/BridgeHandlerTest.java index 83bc12ff84d..deb1f211c9a 100644 --- a/bundles/org.openhab.binding.boschshc/src/test/java/org/openhab/binding/boschshc/internal/devices/bridge/BridgeHandlerTest.java +++ b/bundles/org.openhab.binding.boschshc/src/test/java/org/openhab/binding/boschshc/internal/devices/bridge/BridgeHandlerTest.java @@ -12,6 +12,8 @@ */ package org.openhab.binding.boschshc.internal.devices.bridge; +import static org.hamcrest.CoreMatchers.is; +import static org.hamcrest.CoreMatchers.nullValue; import static org.hamcrest.MatcherAssert.assertThat; import static org.hamcrest.collection.IsCollectionWithSize.hasSize; import static org.junit.jupiter.api.Assertions.assertEquals; @@ -1126,4 +1128,33 @@ class BridgeHandlerTest { verify(httpClient).createRequest(any(), same(HttpMethod.GET)); verify(httpClient).sendRequest(any(), same(PublicInformation.class), any(), isNull()); } + + @Test + void resolveRoomId() throws InterruptedException, TimeoutException, ExecutionException { + Request request = mock(Request.class); + when(httpClient.createRequest(any(), eq(HttpMethod.GET))).thenReturn(request); + ContentResponse contentResponse = mock(ContentResponse.class); + when(request.send()).thenReturn(contentResponse); + when(contentResponse.getStatus()).thenReturn(200); + String roomsJson = """ + [ + { + "@type": "room", + "id": "hz_1", + "iconId": "icon_room_living_room", + "name": "Living Room" + }, + { + "@type": "room", + "id": "hz_2", + "iconId": "icon_room_dining_room", + "name": "Dining Room" + } + ] + """; + when(contentResponse.getContentAsString()).thenReturn(roomsJson); + assertThat(fixture.resolveRoomId("hz_1"), is("Living Room")); + assertThat(fixture.resolveRoomId("hz_2"), is("Dining Room")); + assertThat(fixture.resolveRoomId(null), is(nullValue())); + } }