2 * Copyright (c) 2010-2022 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.pushover.internal.handler;
15 import static org.openhab.binding.pushover.internal.PushoverBindingConstants.DEFAULT_SOUND;
17 import java.util.Collection;
18 import java.util.List;
21 import org.eclipse.jdt.annotation.NonNullByDefault;
22 import org.eclipse.jdt.annotation.Nullable;
23 import org.eclipse.jetty.client.HttpClient;
24 import org.openhab.binding.pushover.internal.actions.PushoverActions;
25 import org.openhab.binding.pushover.internal.config.PushoverAccountConfiguration;
26 import org.openhab.binding.pushover.internal.config.PushoverConfigOptionProvider;
27 import org.openhab.binding.pushover.internal.connection.PushoverAPIConnection;
28 import org.openhab.binding.pushover.internal.connection.PushoverCommunicationException;
29 import org.openhab.binding.pushover.internal.connection.PushoverConfigurationException;
30 import org.openhab.binding.pushover.internal.connection.PushoverMessageBuilder;
31 import org.openhab.binding.pushover.internal.dto.Sound;
32 import org.openhab.core.thing.ChannelUID;
33 import org.openhab.core.thing.Thing;
34 import org.openhab.core.thing.ThingStatus;
35 import org.openhab.core.thing.ThingStatusDetail;
36 import org.openhab.core.thing.binding.BaseThingHandler;
37 import org.openhab.core.thing.binding.ThingHandlerService;
38 import org.openhab.core.types.Command;
41 * The {@link PushoverAccountHandler} is responsible for handling commands, which are sent to one of the channels.
43 * @author Christoph Weitkamp - Initial contribution
46 public class PushoverAccountHandler extends BaseThingHandler {
48 private static final Collection<Class<? extends ThingHandlerService>> SUPPORTED_THING_ACTIONS = Set
49 .of(PushoverActions.class, PushoverConfigOptionProvider.class);
51 private final HttpClient httpClient;
53 private PushoverAccountConfiguration config = new PushoverAccountConfiguration();
54 private @Nullable PushoverAPIConnection connection;
56 public PushoverAccountHandler(Thing thing, HttpClient httpClient) {
58 this.httpClient = httpClient;
62 public void handleCommand(ChannelUID channelUID, Command command) {
67 public void initialize() {
68 config = getConfigAs(PushoverAccountConfiguration.class);
70 boolean configValid = true;
71 final String apikey = config.apikey;
72 if (apikey == null || apikey.isEmpty()) {
73 updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
74 "@text/offline.conf-error-missing-apikey");
77 final String user = config.user;
78 if (user == null || user.isEmpty()) {
79 updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
80 "@text/offline.conf-error-missing-user");
85 updateStatus(ThingStatus.UNKNOWN);
87 connection = new PushoverAPIConnection(httpClient, config);
88 scheduler.submit(this::asyncValidateUser);
93 public Collection<Class<? extends ThingHandlerService>> getServices() {
94 return SUPPORTED_THING_ACTIONS;
98 * Retrieves the list of current sounds and their descriptions from the Pushover API.
100 * @return a list of {@link Sound}s
102 public List<Sound> getSounds() {
104 return connection != null ? connection.getSounds() : PushoverAccountConfiguration.DEFAULT_SOUNDS;
105 } catch (PushoverCommunicationException e) {
106 // do nothing, causing exception is already logged
107 } catch (PushoverConfigurationException e) {
108 updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR, e.getMessage());
110 return PushoverAccountConfiguration.DEFAULT_SOUNDS;
114 * Returns a preconfigured {@link PushoverMessageBuilder}.
116 * @param message the message
117 * @return a {@link PushoverMessageBuilder} instance
119 public PushoverMessageBuilder getDefaultPushoverMessageBuilder(String message) {
120 PushoverMessageBuilder builder = PushoverMessageBuilder.getInstance(config.apikey, config.user)
121 .withMessage(message) //
122 .withTitle(config.title) //
123 .withRetry(config.retry) //
124 .withExpire(config.expire);
125 // specify format if defined
126 switch (config.format) {
127 case PushoverMessageBuilder.MESSAGE_KEY_HTML:
128 builder.withHtmlFormatting();
130 case PushoverMessageBuilder.MESSAGE_KEY_MONOSPACE:
131 builder.withMonospaceFormatting();
135 // add sound, if defined
136 if (!DEFAULT_SOUND.equals(config.sound)) {
137 builder.withSound(config.sound);
142 public boolean sendMessage(PushoverMessageBuilder messageBuilder) {
143 if (connection != null) {
145 return connection.sendMessage(messageBuilder);
146 } catch (PushoverCommunicationException e) {
147 // do nothing, causing exception is already logged
148 } catch (PushoverConfigurationException e) {
149 updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR, e.getMessage());
153 throw new IllegalArgumentException("PushoverAPIConnection is null!");
157 public String sendPriorityMessage(PushoverMessageBuilder messageBuilder) {
158 if (connection != null) {
160 return connection.sendPriorityMessage(messageBuilder);
161 } catch (PushoverCommunicationException e) {
162 // do nothing, causing exception is already logged
163 } catch (PushoverConfigurationException e) {
164 updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR, e.getMessage());
168 throw new IllegalArgumentException("PushoverAPIConnection is null!");
172 public boolean cancelPriorityMessage(String receipt) {
173 if (connection != null) {
175 return connection.cancelPriorityMessage(receipt);
176 } catch (PushoverCommunicationException e) {
177 // do nothing, causing exception is already logged
178 } catch (PushoverConfigurationException e) {
179 updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR, e.getMessage());
183 throw new IllegalArgumentException("PushoverAPIConnection is null!");
187 @SuppressWarnings("null")
188 private void asyncValidateUser() {
190 connection.validateUser();
191 updateStatus(ThingStatus.ONLINE);
192 } catch (PushoverCommunicationException | PushoverConfigurationException e) {
193 updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR, e.getMessage());