2 * Copyright (c) 2010-2022 Contributors to the openHAB project
4 * See the NOTICE file(s) distributed with this work for additional
7 * This program and the accompanying materials are made available under the
8 * terms of the Eclipse Public License 2.0 which is available at
9 * http://www.eclipse.org/legal/epl-2.0
11 * SPDX-License-Identifier: EPL-2.0
13 package org.openhab.binding.touchwand.internal.discovery;
15 import static org.openhab.binding.touchwand.internal.TouchWandBindingConstants.*;
17 import java.util.Arrays;
18 import java.util.Date;
19 import java.util.HashMap;
21 import java.util.concurrent.CopyOnWriteArraySet;
22 import java.util.concurrent.ScheduledFuture;
23 import java.util.concurrent.TimeUnit;
25 import org.eclipse.jdt.annotation.NonNull;
26 import org.eclipse.jdt.annotation.NonNullByDefault;
27 import org.eclipse.jdt.annotation.Nullable;
28 import org.openhab.binding.touchwand.internal.TouchWandBridgeHandler;
29 import org.openhab.binding.touchwand.internal.TouchWandUnitStatusUpdateListener;
30 import org.openhab.binding.touchwand.internal.dto.TouchWandUnitData;
31 import org.openhab.binding.touchwand.internal.dto.TouchWandUnitFromJson;
32 import org.openhab.core.config.discovery.AbstractDiscoveryService;
33 import org.openhab.core.config.discovery.DiscoveryResultBuilder;
34 import org.openhab.core.thing.ThingStatus;
35 import org.openhab.core.thing.ThingTypeUID;
36 import org.openhab.core.thing.ThingUID;
37 import org.openhab.core.thing.binding.ThingHandler;
38 import org.openhab.core.thing.binding.ThingHandlerService;
39 import org.slf4j.Logger;
40 import org.slf4j.LoggerFactory;
42 import com.google.gson.JsonArray;
43 import com.google.gson.JsonElement;
44 import com.google.gson.JsonParser;
45 import com.google.gson.JsonSyntaxException;
48 * The {@link TouchWandUnitDiscoveryService} Discovery service for TouchWand units.
50 * @author Roie Geron - Initial contribution
53 public class TouchWandUnitDiscoveryService extends AbstractDiscoveryService implements ThingHandlerService {
55 private static final int SEARCH_TIME_SEC = 10;
56 private static final int SCAN_INTERVAL_SEC = 60;
57 private static final int LINK_DISCOVERY_SERVICE_INITIAL_DELAY_SEC = 5;
58 private static final String[] CONNECTIVITY_OPTIONS = { CONNECTIVITY_KNX, CONNECTIVITY_ZWAVE, CONNECTIVITY_RISCO,
59 CONNECTIVITY_PIMA, CONNECTIVITY_ACWAND };
60 private @NonNullByDefault({}) TouchWandBridgeHandler touchWandBridgeHandler;
61 private final Logger logger = LoggerFactory.getLogger(TouchWandUnitDiscoveryService.class);
63 private @Nullable ScheduledFuture<?> scanningJob;
64 private CopyOnWriteArraySet<TouchWandUnitStatusUpdateListener> listeners = new CopyOnWriteArraySet<>();
66 public TouchWandUnitDiscoveryService() {
67 super(SUPPORTED_THING_TYPES_UIDS, SEARCH_TIME_SEC, true);
71 protected void startScan() {
72 if (touchWandBridgeHandler.getThing().getStatus() != ThingStatus.ONLINE) {
73 logger.debug("Could not scan units while bridge offline");
77 logger.debug("Starting TouchWand discovery on bridge {}", touchWandBridgeHandler.getThing().getUID());
78 String response = touchWandBridgeHandler.touchWandClient.cmdListUnits();
79 if (response.isEmpty()) {
84 JsonArray jsonArray = JsonParser.parseString(response).getAsJsonArray();
85 if (jsonArray.isJsonArray()) {
87 for (JsonElement unit : jsonArray) {
88 TouchWandUnitData touchWandUnit;
89 touchWandUnit = TouchWandUnitFromJson.parseResponse(unit.getAsJsonObject());
91 if (!touchWandBridgeHandler.isAddSecondaryControllerUnits()) {
92 if (!Arrays.asList(CONNECTIVITY_OPTIONS).contains(touchWandUnit.getConnectivity())) {
96 String type = touchWandUnit.getType();
97 if (!Arrays.asList(SUPPORTED_TOUCHWAND_TYPES).contains(type)) {
98 logger.debug("Unit discovery skipping unsupported unit type : {} ", type);
102 case TYPE_WALLCONTROLLER:
103 addDeviceDiscoveryResult(touchWandUnit, THING_TYPE_WALLCONTROLLER);
106 addDeviceDiscoveryResult(touchWandUnit, THING_TYPE_SWITCH);
109 addDeviceDiscoveryResult(touchWandUnit, THING_TYPE_DIMMER);
112 addDeviceDiscoveryResult(touchWandUnit, THING_TYPE_SHUTTER);
114 case TYPE_ALARMSENSOR:
115 addDeviceDiscoveryResult(touchWandUnit, THING_TYPE_ALARMSENSOR);
118 addDeviceDiscoveryResult(touchWandUnit, THING_TYPE_BSENSOR);
120 case TYPE_THERMOSTAT:
121 addDeviceDiscoveryResult(touchWandUnit, THING_TYPE_THERMOSTAT);
126 notifyListeners(touchWandUnit);
128 } catch (JsonSyntaxException e) {
129 logger.warn("Could not parse unit {}", e.getMessage());
132 } catch (JsonSyntaxException msg) {
133 logger.warn("Could not parse list units response error:{} ", msg.getMessage());
137 private void notifyListeners(TouchWandUnitData touchWandUnit) {
138 for (TouchWandUnitStatusUpdateListener listener : listeners) {
139 listener.onDataReceived(touchWandUnit);
144 protected void stopScan() {
145 removeOlderResults(getTimestampOfLastScan());
150 public void activate() {
151 super.activate(null);
152 removeOlderResults(new Date().getTime(), touchWandBridgeHandler.getThing().getUID());
156 public void deactivate() {
157 removeOlderResults(new Date().getTime(), touchWandBridgeHandler.getThing().getUID());
162 protected void startBackgroundDiscovery() {
163 ScheduledFuture<?> localScanningJob = scanningJob;
164 if (localScanningJob == null || localScanningJob.isCancelled()) {
165 scanningJob = scheduler.scheduleWithFixedDelay(this::startScan, LINK_DISCOVERY_SERVICE_INITIAL_DELAY_SEC,
166 SCAN_INTERVAL_SEC, TimeUnit.SECONDS);
171 protected void stopBackgroundDiscovery() {
172 ScheduledFuture<?> myScanningJob = scanningJob;
173 if (myScanningJob != null) {
174 myScanningJob.cancel(true);
179 public void registerListener(TouchWandUnitStatusUpdateListener listener) {
180 if (!listeners.contains(listener)) {
181 logger.debug("Adding TouchWandWebSocket listener {}", listener);
182 listeners.add(listener);
186 public void unregisterListener(TouchWandUnitStatusUpdateListener listener) {
187 logger.debug("Removing TouchWandWebSocket listener {}", listener);
188 listeners.remove(listener);
192 public int getScanTimeout() {
193 return SEARCH_TIME_SEC;
196 private void addDeviceDiscoveryResult(TouchWandUnitData unit, ThingTypeUID typeUID) {
197 ThingUID bridgeUID = touchWandBridgeHandler.getThing().getUID();
198 ThingUID thingUID = new ThingUID(typeUID, bridgeUID, unit.getId().toString());
199 Map<String, Object> properties = new HashMap<>();
200 properties.put(HANDLER_PROPERTIES_ID, unit.getId().toString());
201 properties.put(HANDLER_PROPERTIES_NAME, unit.getName());
203 thingDiscovered(DiscoveryResultBuilder.create(thingUID)
204 .withThingType(typeUID)
205 .withLabel(unit.getName())
206 .withBridge(bridgeUID)
207 .withProperties(properties)
208 .withRepresentationProperty(HANDLER_PROPERTIES_ID)
215 public void setThingHandler(@NonNullByDefault({}) ThingHandler handler) {
216 if (handler instanceof TouchWandBridgeHandler) {
217 touchWandBridgeHandler = (TouchWandBridgeHandler) handler;
218 registerListener(touchWandBridgeHandler);
223 public @NonNull ThingHandler getThingHandler() {
224 return touchWandBridgeHandler;