]> git.basschouten.com Git - openhab-addons.git/blob
042bfeccba8ebb59b7870ee3e7c5d376ec14ff29
[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.revogi.internal.api;
14
15 import java.util.List;
16 import java.util.Objects;
17 import java.util.concurrent.CompletableFuture;
18
19 import org.eclipse.jdt.annotation.NonNullByDefault;
20 import org.jetbrains.annotations.NotNull;
21 import org.openhab.binding.revogi.internal.udp.UdpResponseDTO;
22 import org.openhab.binding.revogi.internal.udp.UdpSenderService;
23 import org.slf4j.Logger;
24 import org.slf4j.LoggerFactory;
25
26 import com.google.gson.Gson;
27 import com.google.gson.GsonBuilder;
28 import com.google.gson.JsonSyntaxException;
29
30 /**
31  * The {@link SwitchService} enables the binding to actually switch plugs on and of
32  *
33  * @author Andi Bräu - Initial contribution
34  */
35 @NonNullByDefault
36 public class SwitchService {
37
38     private static final String UDP_DISCOVERY_QUERY = "V3{\"sn\":\"%s\", \"cmd\": 20, \"port\": %d, \"state\": %d}";
39     private static final String VERSION_STRING = "V3";
40     private final Logger logger = LoggerFactory.getLogger(SwitchService.class);
41
42     private final Gson gson = new GsonBuilder().create();
43     private final UdpSenderService udpSenderService;
44
45     public SwitchService(UdpSenderService udpSenderService) {
46         this.udpSenderService = udpSenderService;
47     }
48
49     public CompletableFuture<SwitchResponseDTO> switchPort(String serialNumber, String ipAddress, int port, int state) {
50         if (state < 0 || state > 1) {
51             throw new IllegalArgumentException("state has to be 0 or 1");
52         }
53         if (port < 0) {
54             throw new IllegalArgumentException("Given port doesn't exist");
55         }
56
57         CompletableFuture<List<UdpResponseDTO>> responses;
58         if (ipAddress.trim().isEmpty()) {
59             responses = udpSenderService
60                     .broadcastUdpDatagram(String.format(UDP_DISCOVERY_QUERY, serialNumber, port, state));
61         } else {
62             responses = udpSenderService.sendMessage(String.format(UDP_DISCOVERY_QUERY, serialNumber, port, state),
63                     ipAddress);
64         }
65
66         return responses.thenApply(this::getSwitchResponse);
67     }
68
69     @NotNull
70     private SwitchResponseDTO getSwitchResponse(final List<UdpResponseDTO> singleResponse) {
71         return singleResponse.stream().filter(response -> !response.getAnswer().isEmpty())
72                 .map(response -> deserializeString(response.getAnswer()))
73                 .filter(switchResponse -> switchResponse.getCode() == 200 && switchResponse.getResponse() == 20)
74                 .findFirst().orElse(new SwitchResponseDTO(0, 503));
75     }
76
77     private SwitchResponseDTO deserializeString(String response) {
78         String extractedJsonResponse = response.substring(response.lastIndexOf(VERSION_STRING) + 2);
79         try {
80             return Objects.requireNonNull(gson.fromJson(extractedJsonResponse, SwitchResponseDTO.class));
81         } catch (JsonSyntaxException e) {
82             logger.warn("Could not parse string \"{}\" to SwitchResponse", response);
83             return new SwitchResponseDTO(0, 503);
84         }
85     }
86 }