2 * Copyright (c) 2010-2024 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.freeboxos.internal.handler;
15 import java.math.BigDecimal;
16 import java.time.ZonedDateTime;
17 import java.util.HashMap;
18 import java.util.Hashtable;
20 import java.util.concurrent.ScheduledFuture;
21 import java.util.concurrent.TimeUnit;
23 import javax.measure.Unit;
25 import org.eclipse.jdt.annotation.NonNullByDefault;
26 import org.eclipse.jdt.annotation.Nullable;
27 import org.openhab.binding.freeboxos.internal.api.FreeboxException;
28 import org.openhab.binding.freeboxos.internal.api.rest.LanBrowserManager.Source;
29 import org.openhab.binding.freeboxos.internal.api.rest.MediaReceiverManager;
30 import org.openhab.binding.freeboxos.internal.api.rest.MediaReceiverManager.MediaType;
31 import org.openhab.binding.freeboxos.internal.api.rest.MediaReceiverManager.Receiver;
32 import org.openhab.binding.freeboxos.internal.api.rest.RestManager;
33 import org.openhab.binding.freeboxos.internal.config.ApiConsumerConfiguration;
34 import org.openhab.binding.freeboxos.internal.config.ClientConfiguration;
35 import org.openhab.core.audio.AudioSink;
36 import org.openhab.core.config.core.Configuration;
37 import org.openhab.core.library.types.DateTimeType;
38 import org.openhab.core.library.types.DecimalType;
39 import org.openhab.core.library.types.OnOffType;
40 import org.openhab.core.library.types.QuantityType;
41 import org.openhab.core.library.types.StringType;
42 import org.openhab.core.thing.Bridge;
43 import org.openhab.core.thing.ChannelUID;
44 import org.openhab.core.thing.Thing;
45 import org.openhab.core.thing.ThingStatus;
46 import org.openhab.core.thing.ThingStatusDetail;
47 import org.openhab.core.thing.ThingStatusInfo;
48 import org.openhab.core.thing.binding.BaseThingHandler;
49 import org.openhab.core.thing.binding.BridgeHandler;
50 import org.openhab.core.types.Command;
51 import org.openhab.core.types.RefreshType;
52 import org.openhab.core.types.State;
53 import org.openhab.core.types.UnDefType;
54 import org.osgi.framework.ServiceRegistration;
55 import org.slf4j.Logger;
56 import org.slf4j.LoggerFactory;
58 import inet.ipaddr.IPAddress;
61 * The {@link ApiConsumerHandler} is a base abstract class for all devices made available by the FreeboxOs bridge
63 * @author Gaƫl L'hopital - Initial contribution
66 public abstract class ApiConsumerHandler extends BaseThingHandler implements ApiConsumerIntf {
67 private final Logger logger = LoggerFactory.getLogger(ApiConsumerHandler.class);
68 private final Map<String, ScheduledFuture<?>> jobs = new HashMap<>();
70 private @Nullable ServiceRegistration<?> reg;
71 protected boolean statusDrivenByBridge = true;
73 ApiConsumerHandler(Thing thing) {
78 public void initialize() {
79 FreeboxOsHandler bridgeHandler = checkBridgeHandler();
80 if (bridgeHandler == null) {
83 initializeOnceBridgeOnline(bridgeHandler);
86 private void initializeOnceBridgeOnline(FreeboxOsHandler bridgeHandler) {
87 Map<String, String> properties = editProperties();
89 initializeProperties(properties);
90 updateProperties(properties);
91 } catch (FreeboxException e) {
92 logger.warn("Error getting thing {} properties: {}", thing.getUID(), e.getMessage());
98 protected void configureMediaSink() {
100 String upnpName = editProperties().getOrDefault(Source.UPNP.name(), "");
101 Receiver receiver = getManager(MediaReceiverManager.class).getReceiver(upnpName);
102 if (receiver != null) {
103 Map<String, String> properties = editProperties();
104 receiver.capabilities().entrySet()
105 .forEach(entry -> properties.put(entry.getKey().name(), entry.getValue().toString()));
106 updateProperties(properties);
108 startAudioSink(receiver);
112 } catch (FreeboxException e) {
113 logger.warn("Unable to retrieve Media Receivers: {}", e.getMessage());
117 private void startAudioSink(Receiver receiver) {
118 FreeboxOsHandler bridgeHandler = checkBridgeHandler();
119 // Only video and photo is supported by the API so use VIDEO capability for audio
120 Boolean isAudioReceiver = receiver.capabilities().get(MediaType.VIDEO);
121 if (reg == null && bridgeHandler != null && isAudioReceiver != null && isAudioReceiver.booleanValue()) {
122 ApiConsumerConfiguration config = getConfig().as(ApiConsumerConfiguration.class);
123 String callbackURL = bridgeHandler.getCallbackURL();
124 if (!config.password.isEmpty() || !receiver.passwordProtected()) {
125 reg = bridgeHandler.getBundleContext()
127 AudioSink.class.getName(), new AirMediaSink(this, bridgeHandler.getAudioHTTPServer(),
128 callbackURL, receiver.name(), config.password, config.acceptAllMp3),
130 logger.debug("Audio sink registered for {}.", receiver.name());
132 logger.warn("A password needs to be configured to enable Air Media capability.");
137 private void stopAudioSink() {
138 ServiceRegistration<?> localReg = reg;
139 if (localReg != null) {
140 localReg.unregister();
141 logger.debug("Audio sink unregistered");
146 public <T extends RestManager> T getManager(Class<T> clazz) throws FreeboxException {
147 FreeboxOsHandler handler = checkBridgeHandler();
148 if (handler != null) {
149 return handler.getManager(clazz);
151 throw new FreeboxException("Bridge handler not yet defined");
154 abstract void initializeProperties(Map<String, String> properties) throws FreeboxException;
157 public void bridgeStatusChanged(ThingStatusInfo bridgeStatusInfo) {
158 FreeboxOsHandler bridgeHandler = checkBridgeHandler();
159 if (bridgeHandler != null) {
160 initializeOnceBridgeOnline(bridgeHandler);
167 public void handleCommand(ChannelUID channelUID, Command command) {
168 if (getThing().getStatus() != ThingStatus.ONLINE) {
172 if (checkBridgeHandler() != null) {
173 if (command instanceof RefreshType) {
175 } else if (!internalHandleCommand(channelUID.getIdWithoutGroup(), command)) {
176 logger.debug("Unexpected command {} on channel {}", command, channelUID.getId());
179 } catch (FreeboxException e) {
180 logger.warn("Error handling command: {}", e.getMessage());
184 private @Nullable FreeboxOsHandler checkBridgeHandler() {
185 Bridge bridge = getBridge();
186 if (bridge != null) {
187 BridgeHandler handler = bridge.getHandler();
188 if (handler instanceof FreeboxOsHandler fbOsHandler) {
189 if (bridge.getStatus() == ThingStatus.ONLINE) {
190 if (statusDrivenByBridge) {
191 updateStatus(ThingStatus.ONLINE);
195 updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.BRIDGE_OFFLINE);
197 updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.HANDLER_MISSING_ERROR);
200 updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.BRIDGE_UNINITIALIZED);
206 public void dispose() {
212 private void startRefreshJob() {
213 removeJob("GlobalJob");
215 int refreshInterval = getConfigAs(ApiConsumerConfiguration.class).refreshInterval;
216 logger.debug("Scheduling state update every {} seconds for thing {}...", refreshInterval, getThing().getUID());
218 ThingStatusDetail detail = thing.getStatusInfo().getStatusDetail();
219 if (ThingStatusDetail.DUTY_CYCLE.equals(detail)) {
222 } catch (FreeboxException ignore) {
223 // An exception is normal if the box is rebooting then let's try again later...
224 addJob("Initialize", this::initialize, 10, TimeUnit.SECONDS);
229 addJob("GlobalJob", () -> {
232 } catch (FreeboxException e) {
233 updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR, e.getMessage());
235 }, 0, refreshInterval, TimeUnit.SECONDS);
238 private void removeJob(String name) {
239 ScheduledFuture<?> existing = jobs.get(name);
240 if (existing != null && !existing.isCancelled()) {
241 existing.cancel(true);
246 public void addJob(String name, Runnable command, long initialDelay, long delay, TimeUnit unit) {
248 jobs.put(name, scheduler.scheduleWithFixedDelay(command, initialDelay, delay, unit));
252 public void addJob(String name, Runnable command, long delay, TimeUnit unit) {
254 jobs.put(name, scheduler.schedule(command, delay, unit));
258 public void stopJobs() {
259 jobs.keySet().forEach(name -> removeJob(name));
263 protected boolean internalHandleCommand(String channelId, Command command) throws FreeboxException {
267 protected abstract void internalPoll() throws FreeboxException;
269 protected void internalForcePoll() throws FreeboxException {
273 private void updateIfActive(String group, String channelId, State state) {
274 ChannelUID id = new ChannelUID(getThing().getUID(), group, channelId);
276 updateState(id, state);
280 protected void updateIfActive(String channelId, State state) {
281 ChannelUID id = new ChannelUID(getThing().getUID(), channelId);
283 updateState(id, state);
287 protected void updateChannelDateTimeState(String channelId, @Nullable ZonedDateTime timestamp) {
288 updateIfActive(channelId, timestamp == null ? UnDefType.NULL : new DateTimeType(timestamp));
291 protected void updateChannelDateTimeState(String group, String channelId, @Nullable ZonedDateTime timestamp) {
292 updateIfActive(group, channelId, timestamp == null ? UnDefType.NULL : new DateTimeType(timestamp));
295 protected void updateChannelOnOff(String group, String channelId, boolean value) {
296 updateIfActive(group, channelId, OnOffType.from(value));
299 protected void updateChannelOnOff(String channelId, boolean value) {
300 updateIfActive(channelId, OnOffType.from(value));
303 protected void updateChannelString(String group, String channelId, @Nullable String value) {
304 updateIfActive(group, channelId, value != null ? new StringType(value) : UnDefType.NULL);
307 protected void updateChannelString(String group, String channelId, @Nullable IPAddress value) {
308 updateIfActive(group, channelId, value != null ? new StringType(value.toCanonicalString()) : UnDefType.NULL);
311 protected void updateChannelString(String channelId, @Nullable String value) {
312 updateIfActive(channelId, value != null ? new StringType(value) : UnDefType.NULL);
315 protected void updateChannelString(String channelId, Enum<?> value) {
316 updateIfActive(channelId, new StringType(value.name()));
319 protected void updateChannelString(String group, String channelId, Enum<?> value) {
320 updateIfActive(group, channelId, new StringType(value.name()));
323 protected void updateChannelQuantity(String group, String channelId, double d, Unit<?> unit) {
324 updateChannelQuantity(group, channelId, new QuantityType<>(d, unit));
327 protected void updateChannelQuantity(String channelId, @Nullable QuantityType<?> quantity) {
328 updateIfActive(channelId, quantity != null ? quantity : UnDefType.NULL);
331 protected void updateChannelQuantity(String group, String channelId, @Nullable QuantityType<?> quantity) {
332 updateIfActive(group, channelId, quantity != null ? quantity : UnDefType.NULL);
335 protected void updateChannelDecimal(String group, String channelId, @Nullable Integer value) {
336 updateIfActive(group, channelId, value != null ? new DecimalType(value) : UnDefType.NULL);
339 protected void updateChannelQuantity(String group, String channelId, QuantityType<?> qtty, Unit<?> unit) {
340 updateChannelQuantity(group, channelId, qtty.toUnit(unit));
344 public void updateStatus(ThingStatus status, ThingStatusDetail statusDetail, @Nullable String description) {
345 super.updateStatus(status, statusDetail, description);
349 public Map<String, String> editProperties() {
350 return super.editProperties();
354 public void updateProperties(@Nullable Map<String, String> properties) {
355 super.updateProperties(properties);
359 public Configuration getConfig() {
360 return super.getConfig();
364 public int getClientId() {
365 return ((BigDecimal) getConfig().get(ClientConfiguration.ID)).intValue();