2 * Copyright (c) 2010-2021 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.revogi.internal.api;
15 import java.util.List;
16 import java.util.Objects;
17 import java.util.concurrent.CompletableFuture;
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;
26 import com.google.gson.Gson;
27 import com.google.gson.GsonBuilder;
28 import com.google.gson.JsonSyntaxException;
31 * The {@link SwitchService} enables the binding to actually switch plugs on and of
33 * @author Andi Bräu - Initial contribution
36 public class SwitchService {
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);
42 private final Gson gson = new GsonBuilder().create();
43 private final UdpSenderService udpSenderService;
45 public SwitchService(UdpSenderService udpSenderService) {
46 this.udpSenderService = udpSenderService;
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");
54 throw new IllegalArgumentException("Given port doesn't exist");
57 CompletableFuture<List<UdpResponseDTO>> responses;
58 if (ipAddress.trim().isEmpty()) {
59 responses = udpSenderService
60 .broadcastUdpDatagram(String.format(UDP_DISCOVERY_QUERY, serialNumber, port, state));
62 responses = udpSenderService.sendMessage(String.format(UDP_DISCOVERY_QUERY, serialNumber, port, state),
66 return responses.thenApply(this::getSwitchResponse);
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));
77 private SwitchResponseDTO deserializeString(String response) {
78 String extractedJsonResponse = response.substring(response.lastIndexOf(VERSION_STRING) + 2);
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);