]> git.basschouten.com Git - openhab-addons.git/blob
1d7f1d80a9a8b7acc36685e3f8c853fbbec8e330
[openhab-addons.git] /
1 /**
2  * Copyright (c) 2010-2021 Contributors to the openHAB project
3  *
4  * See the NOTICE file(s) distributed with this work for additional
5  * information.
6  *
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
10  *
11  * SPDX-License-Identifier: EPL-2.0
12  */
13 package org.openhab.binding.touchwand.internal.discovery;
14
15 import static org.openhab.binding.touchwand.internal.TouchWandBindingConstants.*;
16
17 import java.util.Arrays;
18 import java.util.Date;
19 import java.util.HashMap;
20 import java.util.Map;
21 import java.util.concurrent.CopyOnWriteArraySet;
22 import java.util.concurrent.ScheduledFuture;
23 import java.util.concurrent.TimeUnit;
24
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.config.discovery.DiscoveryService;
35 import org.openhab.core.thing.ThingStatus;
36 import org.openhab.core.thing.ThingTypeUID;
37 import org.openhab.core.thing.ThingUID;
38 import org.openhab.core.thing.binding.ThingHandler;
39 import org.openhab.core.thing.binding.ThingHandlerService;
40 import org.slf4j.Logger;
41 import org.slf4j.LoggerFactory;
42
43 import com.google.gson.JsonArray;
44 import com.google.gson.JsonElement;
45 import com.google.gson.JsonParser;
46 import com.google.gson.JsonSyntaxException;
47
48 /**
49  * The {@link TouchWandUnitDiscoveryService} Discovery service for TouchWand units.
50  *
51  * @author Roie Geron - Initial contribution
52  */
53 @NonNullByDefault
54 public class TouchWandUnitDiscoveryService extends AbstractDiscoveryService
55         implements DiscoveryService, ThingHandlerService {
56
57     private static final int SEARCH_TIME_SEC = 10;
58     private static final int SCAN_INTERVAL_SEC = 60;
59     private static final int LINK_DISCOVERY_SERVICE_INITIAL_DELAY_SEC = 5;
60     private static final String[] CONNECTIVITY_OPTIONS = { CONNECTIVITY_KNX, CONNECTIVITY_ZWAVE };
61     private @NonNullByDefault({}) TouchWandBridgeHandler touchWandBridgeHandler;
62     private final Logger logger = LoggerFactory.getLogger(TouchWandUnitDiscoveryService.class);
63
64     private @Nullable ScheduledFuture<?> scanningJob;
65     private CopyOnWriteArraySet<TouchWandUnitStatusUpdateListener> listeners = new CopyOnWriteArraySet<>();
66
67     public TouchWandUnitDiscoveryService() {
68         super(SUPPORTED_THING_TYPES_UIDS, SEARCH_TIME_SEC, true);
69     }
70
71     @Override
72     protected void startScan() {
73         if (touchWandBridgeHandler.getThing().getStatus() != ThingStatus.ONLINE) {
74             logger.warn("Could not scan units while bridge offline");
75             return;
76         }
77
78         logger.debug("Starting TouchWand discovery on bridge {}", touchWandBridgeHandler.getThing().getUID());
79         String response = touchWandBridgeHandler.touchWandClient.cmdListUnits();
80         if (response.isEmpty()) {
81             return;
82         }
83
84         try {
85             JsonArray jsonArray = JsonParser.parseString(response).getAsJsonArray();
86             if (jsonArray.isJsonArray()) {
87                 try {
88                     for (JsonElement unit : jsonArray) {
89                         TouchWandUnitData touchWandUnit;
90                         touchWandUnit = TouchWandUnitFromJson.parseResponse(unit.getAsJsonObject());
91
92                         if (!touchWandBridgeHandler.isAddSecondaryControllerUnits()) {
93                             if (!Arrays.asList(CONNECTIVITY_OPTIONS).contains(touchWandUnit.getConnectivity())) {
94                                 continue;
95                             }
96                         }
97                         String type = touchWandUnit.getType();
98                         if (!Arrays.asList(SUPPORTED_TOUCHWAND_TYPES).contains(type)) {
99                             logger.debug("Unit discovery skipping unsupported unit type : {} ", type);
100                             continue;
101                         }
102                         switch (type) {
103                             case TYPE_WALLCONTROLLER:
104                                 addDeviceDiscoveryResult(touchWandUnit, THING_TYPE_WALLCONTROLLER);
105                                 break;
106                             case TYPE_SWITCH:
107                                 addDeviceDiscoveryResult(touchWandUnit, THING_TYPE_SWITCH);
108                                 notifyListeners(touchWandUnit);
109                                 break;
110                             case TYPE_DIMMER:
111                                 addDeviceDiscoveryResult(touchWandUnit, THING_TYPE_DIMMER);
112                                 notifyListeners(touchWandUnit);
113                                 break;
114                             case TYPE_SHUTTER:
115                                 addDeviceDiscoveryResult(touchWandUnit, THING_TYPE_SHUTTER);
116                                 break;
117                             case TYPE_ALARMSENSOR:
118                                 addDeviceDiscoveryResult(touchWandUnit, THING_TYPE_ALARMSENSOR);
119                                 break;
120                             default:
121                                 continue;
122                         }
123                     }
124                 } catch (JsonSyntaxException e) {
125                     logger.warn("Could not parse unit {}", e.getMessage());
126                 }
127             }
128         } catch (JsonSyntaxException msg) {
129             logger.warn("Could not parse list units response {}", msg.getMessage());
130         }
131     }
132
133     private void notifyListeners(TouchWandUnitData touchWandUnit) {
134         for (TouchWandUnitStatusUpdateListener listener : listeners) {
135             listener.onDataReceived(touchWandUnit);
136         }
137     }
138
139     @Override
140     protected void stopScan() {
141         removeOlderResults(getTimestampOfLastScan());
142         super.stopScan();
143     }
144
145     @Override
146     public void activate() {
147         super.activate(null);
148         removeOlderResults(new Date().getTime(), touchWandBridgeHandler.getThing().getUID());
149     }
150
151     @Override
152     public void deactivate() {
153         removeOlderResults(new Date().getTime(), touchWandBridgeHandler.getThing().getUID());
154         super.deactivate();
155     }
156
157     @Override
158     protected void startBackgroundDiscovery() {
159         ScheduledFuture<?> localScanningJob = scanningJob;
160         if (localScanningJob == null || localScanningJob.isCancelled()) {
161             scanningJob = scheduler.scheduleWithFixedDelay(this::startScan, LINK_DISCOVERY_SERVICE_INITIAL_DELAY_SEC,
162                     SCAN_INTERVAL_SEC, TimeUnit.SECONDS);
163         }
164     }
165
166     @Override
167     protected void stopBackgroundDiscovery() {
168         ScheduledFuture<?> myScanningJob = scanningJob;
169         if (myScanningJob != null) {
170             myScanningJob.cancel(true);
171             scanningJob = null;
172         }
173     }
174
175     public void registerListener(TouchWandUnitStatusUpdateListener listener) {
176         if (!listeners.contains(listener)) {
177             logger.debug("Adding TouchWandWebSocket listener {}", listener);
178             listeners.add(listener);
179         }
180     }
181
182     public void unregisterListener(TouchWandUnitStatusUpdateListener listener) {
183         logger.debug("Removing TouchWandWebSocket listener {}", listener);
184         listeners.remove(listener);
185     }
186
187     @Override
188     public int getScanTimeout() {
189         return SEARCH_TIME_SEC;
190     }
191
192     private void addDeviceDiscoveryResult(TouchWandUnitData unit, ThingTypeUID typeUID) {
193         ThingUID bridgeUID = touchWandBridgeHandler.getThing().getUID();
194         ThingUID thingUID = new ThingUID(typeUID, bridgeUID, unit.getId().toString());
195         Map<String, Object> properties = new HashMap<>();
196         properties.put(HANDLER_PROPERTIES_ID, unit.getId().toString());
197         properties.put(HANDLER_PROPERTIES_NAME, unit.getName());
198         // @formatter:off
199         thingDiscovered(DiscoveryResultBuilder.create(thingUID)
200                 .withThingType(typeUID)
201                 .withLabel(unit.getName())
202                 .withBridge(bridgeUID)
203                 .withProperties(properties)
204                 .withRepresentationProperty(HANDLER_PROPERTIES_ID)
205                 .build()
206         );
207         // @formatter:on
208     }
209
210     @Override
211     public void setThingHandler(@NonNullByDefault({}) ThingHandler handler) {
212         if (handler instanceof TouchWandBridgeHandler) {
213             touchWandBridgeHandler = (TouchWandBridgeHandler) handler;
214             registerListener(touchWandBridgeHandler);
215         }
216     }
217
218     @Override
219     public @NonNull ThingHandler getThingHandler() {
220         return touchWandBridgeHandler;
221     }
222 }