2 * Copyright (c) 2010-2023 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.homeconnect.internal.client;
15 import static org.openhab.binding.homeconnect.internal.HomeConnectBindingConstants.*;
17 import java.util.Collection;
18 import java.util.HashMap;
19 import java.util.List;
21 import java.util.concurrent.ScheduledExecutorService;
22 import java.util.concurrent.TimeUnit;
23 import java.util.stream.Collectors;
25 import javax.ws.rs.client.Client;
26 import javax.ws.rs.client.ClientBuilder;
27 import javax.ws.rs.sse.SseEventSource;
29 import org.eclipse.jdt.annotation.NonNullByDefault;
30 import org.eclipse.jdt.annotation.Nullable;
31 import org.openhab.binding.homeconnect.internal.client.exception.AuthorizationException;
32 import org.openhab.binding.homeconnect.internal.client.exception.CommunicationException;
33 import org.openhab.binding.homeconnect.internal.client.listener.HomeConnectEventListener;
34 import org.openhab.binding.homeconnect.internal.client.model.Event;
35 import org.openhab.core.auth.client.oauth2.OAuthClientService;
36 import org.osgi.service.jaxrs.client.SseEventSourceFactory;
37 import org.slf4j.Logger;
38 import org.slf4j.LoggerFactory;
41 * Server-Sent-Events client for Home Connect API.
43 * @author Jonas BrĂ¼stel - Initial contribution
44 * @author Laurent Garnier - Replace okhttp SSE by JAX-RS SSE
48 public class HomeConnectEventSourceClient {
50 private static final int SSE_REQUEST_READ_TIMEOUT = 90;
51 private static final int EVENT_QUEUE_SIZE = 300;
53 private final String apiUrl;
54 private final ClientBuilder clientBuilder;
55 private final SseEventSourceFactory eventSourceFactory;
56 private final OAuthClientService oAuthClientService;
57 private final Map<HomeConnectEventListener, SseEventSource> eventSourceConnections;
58 private final Map<SseEventSource, HomeConnectEventSourceListener> eventSourceListeners;
59 private final ScheduledExecutorService scheduler;
60 private final CircularQueue<Event> eventQueue;
62 private final Logger logger = LoggerFactory.getLogger(HomeConnectEventSourceClient.class);
64 public HomeConnectEventSourceClient(ClientBuilder clientBuilder, SseEventSourceFactory eventSourceFactory,
65 OAuthClientService oAuthClientService, boolean simulated, ScheduledExecutorService scheduler,
66 @Nullable List<Event> eventHistory) {
67 this.scheduler = scheduler;
68 this.clientBuilder = clientBuilder;
69 this.eventSourceFactory = eventSourceFactory;
70 this.oAuthClientService = oAuthClientService;
72 apiUrl = simulated ? API_SIMULATOR_BASE_URL : API_BASE_URL;
73 eventSourceConnections = new HashMap<>();
74 eventSourceListeners = new HashMap<>();
75 eventQueue = new CircularQueue<>(EVENT_QUEUE_SIZE);
76 if (eventHistory != null) {
77 eventQueue.addAll(eventHistory);
82 * Register {@link HomeConnectEventListener} to receive events by Home Connect API. This helps to reduce the
83 * amount of request you would usually need to update all channels.
85 * Checkout rate limits of the API at. https://developer.home-connect.com/docs/general/ratelimiting
87 * @param haId appliance id
88 * @param eventListener appliance event listener
89 * @throws CommunicationException API communication exception
90 * @throws AuthorizationException oAuth authorization exception
92 public synchronized void registerEventListener(final String haId, final HomeConnectEventListener eventListener)
93 throws CommunicationException, AuthorizationException {
94 logger.debug("Register event listener for '{}': {}", haId, eventListener);
96 if (!eventSourceConnections.containsKey(eventListener)) {
97 logger.debug("Create new event source listener for '{}'.", haId);
98 String target = apiUrl + "/api/homeappliances/" + haId + "/events";
100 Client client = createClient(target);
102 SseEventSource eventSource = eventSourceFactory.newSource(client.target(target));
103 HomeConnectEventSourceListener eventSourceListener = new HomeConnectEventSourceListener(haId, eventListener,
104 this, scheduler, eventQueue);
105 eventSource.register(eventSourceListener::onEvent, eventSourceListener::onError,
106 eventSourceListener::onComplete);
107 eventSourceListeners.put(eventSource, eventSourceListener);
108 eventSourceConnections.put(eventListener, eventSource);
114 * Unregister {@link HomeConnectEventListener}.
116 * @param eventListener appliance event listener
118 public synchronized void unregisterEventListener(HomeConnectEventListener eventListener) {
119 unregisterEventListener(eventListener, false, false);
123 * Unregister {@link HomeConnectEventListener}.
125 * @param eventListener appliance event listener
126 * @param completed true when the event source is known as already completed by the server
128 public synchronized void unregisterEventListener(HomeConnectEventListener eventListener, boolean completed) {
129 unregisterEventListener(eventListener, false, completed);
133 * Unregister {@link HomeConnectEventListener}.
135 * @param eventListener appliance event listener
136 * @param immediate true when the unregistering of the event source has to be fast
137 * @param completed true when the event source is known as already completed by the server
139 public synchronized void unregisterEventListener(HomeConnectEventListener eventListener, boolean immediate,
141 if (eventSourceConnections.containsKey(eventListener)) {
142 SseEventSource eventSource = eventSourceConnections.get(eventListener);
143 if (eventSource != null) {
144 closeEventSource(eventSource, immediate, completed);
145 eventSourceListeners.remove(eventSource);
147 eventSourceConnections.remove(eventListener);
151 private void closeEventSource(SseEventSource eventSource, boolean immediate, boolean completed) {
152 var open = eventSource.isOpen();
153 logger.debug("Closing event source. open={}, completed={}, immediate={}", open, completed, immediate);
154 if (open && !completed) {
155 eventSource.close(immediate ? 0 : 5, TimeUnit.SECONDS);
156 logger.debug("Event source closed.");
158 HomeConnectEventSourceListener eventSourceListener = eventSourceListeners.get(eventSource);
159 if (eventSourceListener != null) {
160 eventSourceListener.stopMonitor();
164 private Client createClient(String target) throws CommunicationException, AuthorizationException {
165 boolean filterRegistered = clientBuilder.getConfiguration()
166 .isRegistered(HomeConnectStreamingRequestFilter.class);
169 HomeConnectStreamingRequestFilter filter;
170 if (filterRegistered) {
171 filter = clientBuilder.getConfiguration().getInstances().stream()
172 .filter(instance -> instance instanceof HomeConnectStreamingRequestFilter)
173 .map(instance -> (HomeConnectStreamingRequestFilter) instance).findAny().orElseThrow();
174 client = clientBuilder.readTimeout(SSE_REQUEST_READ_TIMEOUT, TimeUnit.SECONDS).build();
176 filter = new HomeConnectStreamingRequestFilter();
177 client = clientBuilder.readTimeout(SSE_REQUEST_READ_TIMEOUT, TimeUnit.SECONDS).register(filter).build();
179 filter.setAuthorizationHeader(target, HttpHelper.getAuthorizationHeader(oAuthClientService));
187 * @return connection count
189 public synchronized int connectionSize() {
190 return eventSourceConnections.size();
194 * Dispose event source client
196 * @param immediate true to request a fast execution
198 public synchronized void dispose(boolean immediate) {
199 eventSourceConnections.forEach((key, eventSource) -> closeEventSource(eventSource, immediate, false));
200 eventSourceListeners.clear();
201 eventSourceConnections.clear();
207 * @return event queue
209 public Collection<Event> getLatestEvents() {
210 return eventQueue.getAll();
214 * Get latest events by haId
216 * @param haId appliance id
217 * @return event queue
219 public List<Event> getLatestEvents(String haId) {
220 return eventQueue.getAll().stream().filter(event -> haId.equals(event.getHaId())).collect(Collectors.toList());