[pushover] Add support for Expiring Messages (#15289)

* Add support for Expiring Messages
* Add action variants with TTL parameter

Resolves #15288

---------

Signed-off-by: Jacob Laursen <jacob-github@vindvejr.dk>
This commit is contained in:
Jacob Laursen 2023-10-21 13:34:25 +02:00 committed by GitHub
parent 7d63504eee
commit 089b4152c0
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
6 changed files with 257 additions and 18 deletions

View File

@ -40,20 +40,34 @@ The parameter `message` is **mandatory**, the `title` parameter defaults to what
Parameters declared as `@Nullable` are not optional. Parameters declared as `@Nullable` are not optional.
One has to pass a `null` value if it should be skipped or the default value for it should be used. One has to pass a `null` value if it should be skipped or the default value for it should be used.
- `sendMessage(String message, @Nullable String title, @Nullable String sound, @Nullable String url, @Nullable String urlTitle, @Nullable String attachment, @Nullable String contentType, @Nullable Integer priority, @Nullable String device)` - This method is used to send a plain text message providing all available parameters. - `sendMessage(String message, @Nullable String title, @Nullable String sound, @Nullable String url, @Nullable String urlTitle, @Nullable String attachment, @Nullable String contentType, @Nullable Integer priority, @Nullable String device, @Nullable Duration ttl)` - This method is used to send a plain text message providing all available parameters.
- `sendMessage(String message)` - This method is used to send a plain text message with default title.
- `sendMessage(String message, @Nullable String title)` - This method is used to send a plain text message. - `sendMessage(String message, @Nullable String title)` - This method is used to send a plain text message.
- `sendMessage(String message, @Nullable String title, @Nullable Duration ttl)` - This method is used to send a plain text message with TTL.
- `sendHtmlMessage(String message, @Nullable String title)` - This method is used to send a HTML message. - `sendHtmlMessage(String message, @Nullable String title)` - This method is used to send a HTML message.
- `sendHtmlMessage(String message, @Nullable String title, @Nullable Duration ttl)` - This method is used to send a HTML message with TTL.
- `sendMonospaceMessage(String message, @Nullable String title)` - This method is used to send a monospace message. - `sendMonospaceMessage(String message, @Nullable String title)` - This method is used to send a monospace message.
- `sendMonospaceMessage(String message, @Nullable String title, @Nullable Duration ttl)` - This method is used to send a monospace message with TTL.
- `sendAttachmentMessage(String message, @Nullable String title, String attachment, @Nullable String contentType)` - This method is used to send a message with an attachment. It takes a local path or URL to the attachment (parameter `attachment` **mandatory**). Additionally you can pass a data URI scheme to this parameter. Optionally pass a `contentType` to define the content-type of the attachment (default: `image/jpeg` or guessed from image data). - `sendAttachmentMessage(String message, @Nullable String title, String attachment, @Nullable String contentType)` - This method is used to send a message with an attachment. It takes a local path or URL to the attachment (parameter `attachment` **mandatory**). Additionally you can pass a data URI scheme to this parameter. Optionally pass a `contentType` to define the content-type of the attachment (default: `image/jpeg` or guessed from image data).
- `sendAttachmentMessage(String message, @Nullable String title, String attachment, @Nullable String contentType, @Nullable Duration ttl)` - This method is used to send a message with an attachment and TTL. See previous method for details.
- `sendURLMessage(String message, @Nullable String title, String url, @Nullable String urlTitle)` - This method is used to send a message with an URL. A supplementary `url` to show with the message and a `urlTitle` for the URL, otherwise just the URL is shown. - `sendURLMessage(String message, @Nullable String title, String url, @Nullable String urlTitle)` - This method is used to send a message with an URL. A supplementary `url` to show with the message and a `urlTitle` for the URL, otherwise just the URL is shown.
- `sendURLMessage(String message, @Nullable String title, String url, @Nullable String urlTitle, @Nullable Duration ttl)` - This method is used to send a message with an URL and TTL. See previous method for details.
- `sendMessageToDevice(String device, String message, @Nullable String title)` - This method is used to send a message to a specific device. Parameter `device` **mandatory** is the name of a specific device (multiple devices may be separated by a comma). - `sendMessageToDevice(String device, String message, @Nullable String title)` - This method is used to send a message to a specific device. Parameter `device` **mandatory** is the name of a specific device (multiple devices may be separated by a comma).
- `sendMessageToDevice(String device, String message, @Nullable String title, @Nullable Duration ttl)` - This method is used to send a message to a specific device with TTL. See previous method for details.
- `sendPriorityMessage(String message, @Nullable String title, @Nullable Integer priority)` - This method is used to send a priority message. - `sendPriorityMessage(String message, @Nullable String title, @Nullable Integer priority)` - This method is used to send a priority message.
Parameter `priority` is the priority to be used (`-2` = lowest priority, `-1` = low priority, `0` = normal priority, `1` = high priority, `2` = emergency priority; default: `2`). Parameter `priority` is the priority to be used (`-2` = lowest priority, `-1` = low priority, `0` = normal priority, `1` = high priority, `2` = emergency priority; default: `2`).
For priority `2` only, the action returns a `String` value (the `receipt`) if the message was sent successfully, otherwise `null`. For priority `2` only, the action returns a `String` value (the `receipt`) if the message was sent successfully, otherwise `null`.
@ -100,3 +114,27 @@ if( receipt !== null ) {
receipt = null receipt = null
} }
``` ```
:::: tabs
::: tab DSL
```java
val actions = getActions("pushover", "pushover:pushover-account:account")
// send expiring message
actions.sendMessage("Boost has been activated", "Recuperator", Duration.ofHours(1))
```
:::
::: tab JavaScript
```javascript
var pushoverActions = actions.thingActions('pushover', 'pushover:pushover-account:account');
// send expiring message
pushoverActions.sendMessage("Boost has been activated", "Recuperator", time.Duration.ofHours(1));
```
:::
::::

View File

@ -15,6 +15,8 @@ package org.openhab.binding.pushover.internal.actions;
import static org.openhab.binding.pushover.internal.PushoverBindingConstants.*; import static org.openhab.binding.pushover.internal.PushoverBindingConstants.*;
import static org.openhab.binding.pushover.internal.connection.PushoverMessageBuilder.*; import static org.openhab.binding.pushover.internal.connection.PushoverMessageBuilder.*;
import java.time.Duration;
import org.eclipse.jdt.annotation.NonNullByDefault; import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable; import org.eclipse.jdt.annotation.Nullable;
import org.openhab.binding.pushover.internal.connection.PushoverMessageBuilder; import org.openhab.binding.pushover.internal.connection.PushoverMessageBuilder;
@ -32,6 +34,7 @@ import org.slf4j.LoggerFactory;
* Some automation actions to be used with a {@link PushoverAccountHandler}. * Some automation actions to be used with a {@link PushoverAccountHandler}.
* *
* @author Christoph Weitkamp - Initial contribution * @author Christoph Weitkamp - Initial contribution
* @author Jacob Laursen - Added support for Expiring Messages
*/ */
@ThingActionsScope(name = "pushover") @ThingActionsScope(name = "pushover")
@NonNullByDefault @NonNullByDefault
@ -53,7 +56,8 @@ public class PushoverActions implements ThingActions {
@ActionInput(name = "attachment", label = "@text/sendMessageActionInputAttachmentLabel", description = "@text/sendMessageActionInputAttachmentDescription", type = "java.lang.String") @Nullable String attachment, @ActionInput(name = "attachment", label = "@text/sendMessageActionInputAttachmentLabel", description = "@text/sendMessageActionInputAttachmentDescription", type = "java.lang.String") @Nullable String attachment,
@ActionInput(name = "contentType", label = "@text/sendMessageActionInputContentTypeLabel", description = "@text/sendMessageActionInputContentTypeDescription", type = "java.lang.String", defaultValue = DEFAULT_CONTENT_TYPE) @Nullable String contentType, @ActionInput(name = "contentType", label = "@text/sendMessageActionInputContentTypeLabel", description = "@text/sendMessageActionInputContentTypeDescription", type = "java.lang.String", defaultValue = DEFAULT_CONTENT_TYPE) @Nullable String contentType,
@ActionInput(name = "priority", label = "@text/sendMessageActionInputPriorityLabel", description = "@text/sendMessageActionInputPriorityDescription", type = "java.lang.Integer", defaultValue = DEFAULT_EMERGENCY_PRIORITY) @Nullable Integer priority, @ActionInput(name = "priority", label = "@text/sendMessageActionInputPriorityLabel", description = "@text/sendMessageActionInputPriorityDescription", type = "java.lang.Integer", defaultValue = DEFAULT_EMERGENCY_PRIORITY) @Nullable Integer priority,
@ActionInput(name = "device", label = "@text/sendMessageActionInputDeviceLabel", description = "@text/sendMessageActionInputDeviceDescription", type = "java.lang.String") @Nullable String device) { @ActionInput(name = "device", label = "@text/sendMessageActionInputDeviceLabel", description = "@text/sendMessageActionInputDeviceDescription", type = "java.lang.String") @Nullable String device,
@ActionInput(name = "ttl", label = "@text/sendMessageActionInputTTLLabel", description = "@text/sendMessageActionInputTTLDescription", type = "java.time.Duration") @Nullable Duration ttl) {
logger.trace( logger.trace(
"ThingAction 'sendMessage' called with value(s): message='{}', title='{}', sound='{}', url='{}', urlTitle='{}', attachment='{}', contentType='{}', priority='{}', device='{}'", "ThingAction 'sendMessage' called with value(s): message='{}', title='{}', sound='{}', url='{}', urlTitle='{}', attachment='{}', contentType='{}', priority='{}', device='{}'",
message, title, sound, url, urlTitle, attachment, contentType, priority, device); message, title, sound, url, urlTitle, attachment, contentType, priority, device);
@ -81,14 +85,28 @@ public class PushoverActions implements ThingActions {
if (device != null) { if (device != null) {
builder.withDevice(device); builder.withDevice(device);
} }
if (ttl != null) {
builder.withTTL(ttl);
}
return send(builder, title); return send(builder, title);
} }
public static Boolean sendMessage(ThingActions actions, String message, @Nullable String title, public static Boolean sendMessage(ThingActions actions, String message, @Nullable String title,
@Nullable String sound, @Nullable String url, @Nullable String urlTitle, @Nullable String attachment, @Nullable String sound, @Nullable String url, @Nullable String urlTitle, @Nullable String attachment,
@Nullable String contentType, @Nullable Integer priority, @Nullable String device) { @Nullable String contentType, @Nullable Integer priority, @Nullable String device, @Nullable Duration ttl) {
return ((PushoverActions) actions).sendMessage(message, title, sound, url, urlTitle, attachment, contentType, return ((PushoverActions) actions).sendMessage(message, title, sound, url, urlTitle, attachment, contentType,
priority, device); priority, device, ttl);
}
@RuleAction(label = "@text/sendMessageActionLabel", description = "@text/sendMessageActionDescription")
public @ActionOutput(name = "sent", label = "@text/sendMessageActionOutputLabel", description = "@text/sendMessageActionOutputDescription", type = "java.lang.Boolean") Boolean sendMessage(
@ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message) {
logger.trace("ThingAction 'sendMessage' called with value(s): message='{}'", message);
return send(getDefaultPushoverMessageBuilder(message), null);
}
public static Boolean sendMessage(ThingActions actions, String message) {
return ((PushoverActions) actions).sendMessage(message);
} }
@RuleAction(label = "@text/sendMessageActionLabel", description = "@text/sendMessageActionDescription") @RuleAction(label = "@text/sendMessageActionLabel", description = "@text/sendMessageActionDescription")
@ -103,11 +121,30 @@ public class PushoverActions implements ThingActions {
return ((PushoverActions) actions).sendMessage(message, title); return ((PushoverActions) actions).sendMessage(message, title);
} }
@RuleAction(label = "@text/sendMessageActionLabel", description = "@text/sendMessageActionDescription")
public @ActionOutput(name = "sent", label = "@text/sendMessageActionOutputLabel", description = "@text/sendMessageActionOutputDescription", type = "java.lang.Boolean") Boolean sendMessage(
@ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message,
@ActionInput(name = "title", label = "@text/sendMessageActionInputTitleLabel", description = "@text/sendMessageActionInputTitleDescription", type = "java.lang.String", defaultValue = DEFAULT_TITLE) @Nullable String title,
@ActionInput(name = "ttl", label = "@text/sendMessageActionInputTTLLabel", description = "@text/sendMessageActionInputTTLDescription", type = "java.time.Duration") @Nullable Duration ttl) {
logger.trace("ThingAction 'sendMessage' called with value(s): message='{}', title='{}', ttl='{}'", message,
title, ttl);
PushoverMessageBuilder builder = getDefaultPushoverMessageBuilder(message);
if (ttl != null) {
builder.withTTL(ttl);
}
return send(builder, title);
}
public static Boolean sendMessage(ThingActions actions, String message, @Nullable String title,
@Nullable Duration ttl) {
return ((PushoverActions) actions).sendMessage(message, title, ttl);
}
@RuleAction(label = "@text/sendURLMessageActionLabel", description = "@text/sendURLMessageActionDescription") @RuleAction(label = "@text/sendURLMessageActionLabel", description = "@text/sendURLMessageActionDescription")
public @ActionOutput(name = "sent", label = "@text/sendMessageActionOutputLabel", description = "@text/sendMessageActionOutputDescription", type = "java.lang.Boolean") Boolean sendURLMessage( public @ActionOutput(name = "sent", label = "@text/sendMessageActionOutputLabel", description = "@text/sendMessageActionOutputDescription", type = "java.lang.Boolean") Boolean sendURLMessage(
@ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message, @ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message,
@ActionInput(name = "title", label = "@text/sendMessageActionInputTitleLabel", description = "@text/sendMessageActionInputTitleDescription", type = "java.lang.String", defaultValue = DEFAULT_TITLE) @Nullable String title, @ActionInput(name = "title", label = "@text/sendMessageActionInputTitleLabel", description = "@text/sendMessageActionInputTitleDescription", type = "java.lang.String", defaultValue = DEFAULT_TITLE) @Nullable String title,
@ActionInput(name = "url", label = "@text/sendMessageActionInputURLLabel", description = "@text/sendMessageActionInputURLDescription", type = "java.lang.String", required = true) String url, @ActionInput(name = "url", label = "@text/sendMessageActionInputURLLabel", description = "@text/sendMessageActionInputURLDescription", type = "java.lang.String", required = true) @Nullable String url,
@ActionInput(name = "urlTitle", label = "@text/sendMessageActionInputURLTitleLabel", description = "@text/sendMessageActionInputURLTitleDescription", type = "java.lang.String") @Nullable String urlTitle) { @ActionInput(name = "urlTitle", label = "@text/sendMessageActionInputURLTitleLabel", description = "@text/sendMessageActionInputURLTitleDescription", type = "java.lang.String") @Nullable String urlTitle) {
logger.trace( logger.trace(
"ThingAction 'sendURLMessage' called with value(s): message='{}', url='{}', title='{}', urlTitle='{}'", "ThingAction 'sendURLMessage' called with value(s): message='{}', url='{}', title='{}', urlTitle='{}'",
@ -128,6 +165,35 @@ public class PushoverActions implements ThingActions {
return ((PushoverActions) actions).sendURLMessage(message, title, url, urlTitle); return ((PushoverActions) actions).sendURLMessage(message, title, url, urlTitle);
} }
@RuleAction(label = "@text/sendURLMessageActionLabel", description = "@text/sendURLMessageActionDescription")
public @ActionOutput(name = "sent", label = "@text/sendMessageActionOutputLabel", description = "@text/sendMessageActionOutputDescription", type = "java.lang.Boolean") Boolean sendURLMessage(
@ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message,
@ActionInput(name = "title", label = "@text/sendMessageActionInputTitleLabel", description = "@text/sendMessageActionInputTitleDescription", type = "java.lang.String", defaultValue = DEFAULT_TITLE) @Nullable String title,
@ActionInput(name = "url", label = "@text/sendMessageActionInputURLLabel", description = "@text/sendMessageActionInputURLDescription", type = "java.lang.String", required = true) @Nullable String url,
@ActionInput(name = "urlTitle", label = "@text/sendMessageActionInputURLTitleLabel", description = "@text/sendMessageActionInputURLTitleDescription", type = "java.lang.String") @Nullable String urlTitle,
@ActionInput(name = "ttl", label = "@text/sendMessageActionInputTTLLabel", description = "@text/sendMessageActionInputTTLDescription", type = "java.time.Duration") @Nullable Duration ttl) {
logger.trace(
"ThingAction 'sendURLMessage' called with value(s): message='{}', url='{}', title='{}', urlTitle='{}', ttl='{}'",
message, url, title, urlTitle, ttl);
if (url == null) {
throw new IllegalArgumentException("Skip sending message as 'url' is null.");
}
PushoverMessageBuilder builder = getDefaultPushoverMessageBuilder(message).withUrl(url);
if (urlTitle != null) {
builder.withUrl(urlTitle);
}
if (ttl != null) {
builder.withTTL(ttl);
}
return send(builder, title);
}
public static Boolean sendURLMessage(ThingActions actions, String message, @Nullable String title, String url,
@Nullable String urlTitle, @Nullable Duration ttl) {
return ((PushoverActions) actions).sendURLMessage(message, title, url, urlTitle, ttl);
}
@RuleAction(label = "@text/sendHTMLMessageActionLabel", description = "@text/sendHTMLMessageActionDescription") @RuleAction(label = "@text/sendHTMLMessageActionLabel", description = "@text/sendHTMLMessageActionDescription")
public @ActionOutput(name = "sent", label = "@text/sendMessageActionOutputLabel", description = "@text/sendMessageActionOutputDescription", type = "java.lang.Boolean") Boolean sendHtmlMessage( public @ActionOutput(name = "sent", label = "@text/sendMessageActionOutputLabel", description = "@text/sendMessageActionOutputDescription", type = "java.lang.Boolean") Boolean sendHtmlMessage(
@ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message, @ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message,
@ -140,6 +206,25 @@ public class PushoverActions implements ThingActions {
return ((PushoverActions) actions).sendHtmlMessage(message, title); return ((PushoverActions) actions).sendHtmlMessage(message, title);
} }
@RuleAction(label = "@text/sendHTMLMessageActionLabel", description = "@text/sendHTMLMessageActionDescription")
public @ActionOutput(name = "sent", label = "@text/sendMessageActionOutputLabel", description = "@text/sendMessageActionOutputDescription", type = "java.lang.Boolean") Boolean sendHtmlMessage(
@ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message,
@ActionInput(name = "title", label = "@text/sendMessageActionInputTitleLabel", description = "@text/sendMessageActionInputTitleDescription", type = "java.lang.String", defaultValue = DEFAULT_TITLE) @Nullable String title,
@ActionInput(name = "ttl", label = "@text/sendMessageActionInputTTLLabel", description = "@text/sendMessageActionInputTTLDescription", type = "java.time.Duration") @Nullable Duration ttl) {
logger.trace("ThingAction 'sendHtmlMessage' called with value(s): message='{}', title='{}', ttl='{}'", message,
title, ttl);
PushoverMessageBuilder builder = getDefaultPushoverMessageBuilder(message).withHtmlFormatting();
if (ttl != null) {
builder.withTTL(ttl);
}
return send(builder, title);
}
public static Boolean sendHtmlMessage(ThingActions actions, String message, @Nullable String title,
@Nullable Duration ttl) {
return ((PushoverActions) actions).sendHtmlMessage(message, title, ttl);
}
@RuleAction(label = "@text/sendMonospaceMessageActionLabel", description = "@text/sendMonospaceMessageActionDescription") @RuleAction(label = "@text/sendMonospaceMessageActionLabel", description = "@text/sendMonospaceMessageActionDescription")
public @ActionOutput(name = "sent", label = "@text/sendMessageActionOutputLabel", description = "@text/sendMessageActionOutputDescription", type = "java.lang.Boolean") Boolean sendMonospaceMessage( public @ActionOutput(name = "sent", label = "@text/sendMessageActionOutputLabel", description = "@text/sendMessageActionOutputDescription", type = "java.lang.Boolean") Boolean sendMonospaceMessage(
@ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message, @ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message,
@ -153,11 +238,30 @@ public class PushoverActions implements ThingActions {
return ((PushoverActions) actions).sendMonospaceMessage(message, title); return ((PushoverActions) actions).sendMonospaceMessage(message, title);
} }
@RuleAction(label = "@text/sendMonospaceMessageActionLabel", description = "@text/sendMonospaceMessageActionDescription")
public @ActionOutput(name = "sent", label = "@text/sendMessageActionOutputLabel", description = "@text/sendMessageActionOutputDescription", type = "java.lang.Boolean") Boolean sendMonospaceMessage(
@ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message,
@ActionInput(name = "title", label = "@text/sendMessageActionInputTitleLabel", description = "@text/sendMessageActionInputTitleDescription", type = "java.lang.String", defaultValue = DEFAULT_TITLE) @Nullable String title,
@ActionInput(name = "ttl", label = "@text/sendMessageActionInputTTLLabel", description = "@text/sendMessageActionInputTTLDescription", type = "java.time.Duration") @Nullable Duration ttl) {
logger.trace("ThingAction 'sendMonospaceMessage' called with value(s): message='{}', title='{}', ttl='{}'",
message, title, ttl);
PushoverMessageBuilder builder = getDefaultPushoverMessageBuilder(message).withMonospaceFormatting();
if (ttl != null) {
builder.withTTL(ttl);
}
return send(builder, title);
}
public static Boolean sendMonospaceMessage(ThingActions actions, String message, @Nullable String title,
@Nullable Duration ttl) {
return ((PushoverActions) actions).sendMonospaceMessage(message, title, ttl);
}
@RuleAction(label = "@text/sendAttachmentMessageActionLabel", description = "@text/sendAttachmentMessageActionDescription") @RuleAction(label = "@text/sendAttachmentMessageActionLabel", description = "@text/sendAttachmentMessageActionDescription")
public @ActionOutput(name = "sent", label = "@text/sendMessageActionOutputLabel", description = "@text/sendMessageActionOutputDescription", type = "java.lang.Boolean") Boolean sendAttachmentMessage( public @ActionOutput(name = "sent", label = "@text/sendMessageActionOutputLabel", description = "@text/sendMessageActionOutputDescription", type = "java.lang.Boolean") Boolean sendAttachmentMessage(
@ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message, @ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message,
@ActionInput(name = "title", label = "@text/sendMessageActionInputTitleLabel", description = "@text/sendMessageActionInputTitleDescription", type = "java.lang.String", defaultValue = DEFAULT_TITLE) @Nullable String title, @ActionInput(name = "title", label = "@text/sendMessageActionInputTitleLabel", description = "@text/sendMessageActionInputTitleDescription", type = "java.lang.String", defaultValue = DEFAULT_TITLE) @Nullable String title,
@ActionInput(name = "attachment", label = "@text/sendMessageActionInputAttachmentLabel", description = "@text/sendMessageActionInputAttachmentDescription", type = "java.lang.String", required = true) String attachment, @ActionInput(name = "attachment", label = "@text/sendMessageActionInputAttachmentLabel", description = "@text/sendMessageActionInputAttachmentDescription", type = "java.lang.String", required = true) @Nullable String attachment,
@ActionInput(name = "contentType", label = "@text/sendMessageActionInputContentTypeLabel", description = "@text/sendMessageActionInputContentTypeDescription", type = "java.lang.String", defaultValue = DEFAULT_CONTENT_TYPE) @Nullable String contentType) { @ActionInput(name = "contentType", label = "@text/sendMessageActionInputContentTypeLabel", description = "@text/sendMessageActionInputContentTypeDescription", type = "java.lang.String", defaultValue = DEFAULT_CONTENT_TYPE) @Nullable String contentType) {
logger.trace( logger.trace(
"ThingAction 'sendAttachmentMessage' called with value(s): message='{}', title='{}', attachment='{}', contentType='{}'", "ThingAction 'sendAttachmentMessage' called with value(s): message='{}', title='{}', attachment='{}', contentType='{}'",
@ -178,6 +282,35 @@ public class PushoverActions implements ThingActions {
return ((PushoverActions) actions).sendAttachmentMessage(message, title, attachment, contentType); return ((PushoverActions) actions).sendAttachmentMessage(message, title, attachment, contentType);
} }
@RuleAction(label = "@text/sendAttachmentMessageActionLabel", description = "@text/sendAttachmentMessageActionDescription")
public @ActionOutput(name = "sent", label = "@text/sendMessageActionOutputLabel", description = "@text/sendMessageActionOutputDescription", type = "java.lang.Boolean") Boolean sendAttachmentMessage(
@ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message,
@ActionInput(name = "title", label = "@text/sendMessageActionInputTitleLabel", description = "@text/sendMessageActionInputTitleDescription", type = "java.lang.String", defaultValue = DEFAULT_TITLE) @Nullable String title,
@ActionInput(name = "attachment", label = "@text/sendMessageActionInputAttachmentLabel", description = "@text/sendMessageActionInputAttachmentDescription", type = "java.lang.String", required = true) @Nullable String attachment,
@ActionInput(name = "contentType", label = "@text/sendMessageActionInputContentTypeLabel", description = "@text/sendMessageActionInputContentTypeDescription", type = "java.lang.String", defaultValue = DEFAULT_CONTENT_TYPE) @Nullable String contentType,
@ActionInput(name = "ttl", label = "@text/sendMessageActionInputTTLLabel", description = "@text/sendMessageActionInputTTLDescription", type = "java.time.Duration") @Nullable Duration ttl) {
logger.trace(
"ThingAction 'sendAttachmentMessage' called with value(s): message='{}', title='{}', attachment='{}', contentType='{}', ttl='{}'",
message, title, attachment, contentType, ttl);
if (attachment == null) {
throw new IllegalArgumentException("Skip sending message as 'attachment' is null.");
}
PushoverMessageBuilder builder = getDefaultPushoverMessageBuilder(message).withAttachment(attachment);
if (contentType != null) {
builder.withContentType(contentType);
}
if (ttl != null) {
builder.withTTL(ttl);
}
return send(builder, title);
}
public static Boolean sendAttachmentMessage(ThingActions actions, String message, @Nullable String title,
String attachment, @Nullable String contentType, @Nullable Duration ttl) {
return ((PushoverActions) actions).sendAttachmentMessage(message, title, attachment, contentType, ttl);
}
@RuleAction(label = "@text/sendPriorityMessageActionLabel", description = "@text/sendPriorityMessageActionDescription") @RuleAction(label = "@text/sendPriorityMessageActionLabel", description = "@text/sendPriorityMessageActionDescription")
public @ActionOutput(name = "receipt", label = "@text/sendPriorityMessageActionOutputLabel", description = "@text/sendPriorityMessageActionOutputDescription", type = "java.lang.String") String sendPriorityMessage( public @ActionOutput(name = "receipt", label = "@text/sendPriorityMessageActionOutputLabel", description = "@text/sendPriorityMessageActionOutputDescription", type = "java.lang.String") String sendPriorityMessage(
@ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message, @ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message,
@ -201,10 +334,10 @@ public class PushoverActions implements ThingActions {
@RuleAction(label = "@text/cancelPriorityMessageActionLabel", description = "@text/cancelPriorityMessageActionDescription") @RuleAction(label = "@text/cancelPriorityMessageActionLabel", description = "@text/cancelPriorityMessageActionDescription")
public @ActionOutput(name = "canceled", label = "@text/cancelPriorityMessageActionOutputLabel", description = "@text/cancelPriorityMessageActionOutputDescription", type = "java.lang.Boolean") Boolean cancelPriorityMessage( public @ActionOutput(name = "canceled", label = "@text/cancelPriorityMessageActionOutputLabel", description = "@text/cancelPriorityMessageActionOutputDescription", type = "java.lang.Boolean") Boolean cancelPriorityMessage(
@ActionInput(name = "receipt", label = "@text/cancelPriorityMessageActionInputReceiptLabel", description = "@text/cancelPriorityMessageActionInputReceiptDescription", type = "java.lang.String", required = true) String receipt) { @ActionInput(name = "receipt", label = "@text/cancelPriorityMessageActionInputReceiptLabel", description = "@text/cancelPriorityMessageActionInputReceiptDescription", type = "java.lang.String", required = true) @Nullable String receipt) {
logger.trace("ThingAction 'cancelPriorityMessage' called with value(s): '{}'", receipt); logger.trace("ThingAction 'cancelPriorityMessage' called with value(s): '{}'", receipt);
if (accountHandler == null) { if (accountHandler == null) {
throw new RuntimeException("PushoverAccountHandler is null!"); throw new IllegalStateException("PushoverAccountHandler is null!");
} }
if (receipt == null) { if (receipt == null) {
@ -220,7 +353,7 @@ public class PushoverActions implements ThingActions {
@RuleAction(label = "@text/sendMessageToDeviceActionLabel", description = "@text/sendMessageToDeviceActionDescription") @RuleAction(label = "@text/sendMessageToDeviceActionLabel", description = "@text/sendMessageToDeviceActionDescription")
public @ActionOutput(name = "sent", label = "@text/sendMessageActionOutputLabel", description = "@text/sendMessageActionOutputDescription", type = "java.lang.Boolean") Boolean sendMessageToDevice( public @ActionOutput(name = "sent", label = "@text/sendMessageActionOutputLabel", description = "@text/sendMessageActionOutputDescription", type = "java.lang.Boolean") Boolean sendMessageToDevice(
@ActionInput(name = "device", label = "@text/sendMessageActionInputDeviceLabel", description = "@text/sendMessageActionInputDeviceDescription", type = "java.lang.String", required = true) String device, @ActionInput(name = "device", label = "@text/sendMessageActionInputDeviceLabel", description = "@text/sendMessageActionInputDeviceDescription", type = "java.lang.String", required = true) @Nullable String device,
@ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message, @ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message,
@ActionInput(name = "title", label = "@text/sendMessageActionInputTitleLabel", description = "@text/sendMessageActionInputTitleDescription", type = "java.lang.String", defaultValue = DEFAULT_TITLE) @Nullable String title) { @ActionInput(name = "title", label = "@text/sendMessageActionInputTitleLabel", description = "@text/sendMessageActionInputTitleDescription", type = "java.lang.String", defaultValue = DEFAULT_TITLE) @Nullable String title) {
logger.trace("ThingAction 'sendMessageToDevice' called with value(s): device='{}', message='{}', title='{}'", logger.trace("ThingAction 'sendMessageToDevice' called with value(s): device='{}', message='{}', title='{}'",
@ -237,9 +370,34 @@ public class PushoverActions implements ThingActions {
return ((PushoverActions) actions).sendMessageToDevice(device, message, title); return ((PushoverActions) actions).sendMessageToDevice(device, message, title);
} }
private PushoverMessageBuilder getDefaultPushoverMessageBuilder(String message) { @RuleAction(label = "@text/sendMessageToDeviceActionLabel", description = "@text/sendMessageToDeviceActionDescription")
public @ActionOutput(name = "sent", label = "@text/sendMessageActionOutputLabel", description = "@text/sendMessageActionOutputDescription", type = "java.lang.Boolean") Boolean sendMessageToDevice(
@ActionInput(name = "device", label = "@text/sendMessageActionInputDeviceLabel", description = "@text/sendMessageActionInputDeviceDescription", type = "java.lang.String", required = true) @Nullable String device,
@ActionInput(name = "message", label = "@text/sendMessageActionInputMessageLabel", description = "@text/sendMessageActionInputMessageDescription", type = "java.lang.String", required = true) String message,
@ActionInput(name = "title", label = "@text/sendMessageActionInputTitleLabel", description = "@text/sendMessageActionInputTitleDescription", type = "java.lang.String", defaultValue = DEFAULT_TITLE) @Nullable String title,
@ActionInput(name = "ttl", label = "@text/sendMessageActionInputTTLLabel", description = "@text/sendMessageActionInputTTLDescription", type = "java.time.Duration") @Nullable Duration ttl) {
logger.trace(
"ThingAction 'sendMessageToDevice' called with value(s): device='{}', message='{}', title='{}', ttl='{}'",
device, message, title, ttl);
if (device == null) {
throw new IllegalArgumentException("Skip sending message as 'device' is null.");
}
PushoverMessageBuilder builder = getDefaultPushoverMessageBuilder(message).withDevice(device);
if (ttl != null) {
builder.withTTL(ttl);
}
return send(builder, title);
}
public static Boolean sendMessageToDevice(ThingActions actions, String device, String message,
@Nullable String title, @Nullable Duration ttl) {
return ((PushoverActions) actions).sendMessageToDevice(device, message, title, ttl);
}
private PushoverMessageBuilder getDefaultPushoverMessageBuilder(@Nullable String message) {
if (accountHandler == null) { if (accountHandler == null) {
throw new RuntimeException("PushoverAccountHandler is null!"); throw new IllegalStateException("PushoverAccountHandler is null!");
} }
if (message == null) { if (message == null) {

View File

@ -18,6 +18,7 @@ import java.io.File;
import java.io.IOException; import java.io.IOException;
import java.nio.file.Files; import java.nio.file.Files;
import java.nio.file.Path; import java.nio.file.Path;
import java.time.Duration;
import java.util.List; import java.util.List;
import java.util.stream.Collectors; import java.util.stream.Collectors;
@ -52,6 +53,7 @@ public class PushoverMessageBuilder {
private static final String MESSAGE_KEY_PRIORITY = "priority"; private static final String MESSAGE_KEY_PRIORITY = "priority";
private static final String MESSAGE_KEY_RETRY = "retry"; private static final String MESSAGE_KEY_RETRY = "retry";
private static final String MESSAGE_KEY_EXPIRE = "expire"; private static final String MESSAGE_KEY_EXPIRE = "expire";
private static final String MESSAGE_KEY_TTL = "ttl";
private static final String MESSAGE_KEY_URL = "url"; private static final String MESSAGE_KEY_URL = "url";
private static final String MESSAGE_KEY_URL_TITLE = "url_title"; private static final String MESSAGE_KEY_URL_TITLE = "url_title";
private static final String MESSAGE_KEY_SOUND = "sound"; private static final String MESSAGE_KEY_SOUND = "sound";
@ -79,6 +81,7 @@ public class PushoverMessageBuilder {
private int priority = DEFAULT_PRIORITY; private int priority = DEFAULT_PRIORITY;
private int retry = 300; private int retry = 300;
private int expire = 3600; private int expire = 3600;
private Duration ttl = Duration.ZERO;
private @Nullable String url; private @Nullable String url;
private @Nullable String urlTitle; private @Nullable String urlTitle;
private @Nullable String sound; private @Nullable String sound;
@ -135,6 +138,11 @@ public class PushoverMessageBuilder {
return this; return this;
} }
public PushoverMessageBuilder withTTL(Duration ttl) {
this.ttl = ttl;
return this;
}
public PushoverMessageBuilder withUrl(String url) { public PushoverMessageBuilder withUrl(String url) {
this.url = url; this.url = url;
return this; return this;
@ -171,6 +179,7 @@ public class PushoverMessageBuilder {
} }
public ContentProvider build() throws CommunicationException { public ContentProvider build() throws CommunicationException {
String message = this.message;
if (message != null) { if (message != null) {
if (message.length() > MAX_MESSAGE_LENGTH) { if (message.length() > MAX_MESSAGE_LENGTH) {
throw new IllegalArgumentException(String.format( throw new IllegalArgumentException(String.format(
@ -179,6 +188,7 @@ public class PushoverMessageBuilder {
body.addFieldPart(MESSAGE_KEY_MESSAGE, new StringContentProvider(message), null); body.addFieldPart(MESSAGE_KEY_MESSAGE, new StringContentProvider(message), null);
} }
String title = this.title;
if (title != null) { if (title != null) {
if (title.length() > MAX_TITLE_LENGTH) { if (title.length() > MAX_TITLE_LENGTH) {
throw new IllegalArgumentException(String throw new IllegalArgumentException(String
@ -187,6 +197,7 @@ public class PushoverMessageBuilder {
body.addFieldPart(MESSAGE_KEY_TITLE, new StringContentProvider(title), null); body.addFieldPart(MESSAGE_KEY_TITLE, new StringContentProvider(title), null);
} }
String device = this.device;
if (device != null) { if (device != null) {
if (device.length() > MAX_DEVICE_LENGTH) { if (device.length() > MAX_DEVICE_LENGTH) {
logger.warn("Skip 'device' as it is longer than {} characters. Got: {}.", MAX_DEVICE_LENGTH, device); logger.warn("Skip 'device' as it is longer than {} characters. Got: {}.", MAX_DEVICE_LENGTH, device);
@ -224,6 +235,14 @@ public class PushoverMessageBuilder {
} }
} }
if (!ttl.isZero()) {
if (priority == EMERGENCY_PRIORITY) {
logger.warn("TTL value of {} will be ignored for emergency priority.", ttl);
}
body.addFieldPart(MESSAGE_KEY_TTL, new StringContentProvider(String.valueOf(ttl.getSeconds())), null);
}
String url = this.url;
if (url != null) { if (url != null) {
if (url.length() > MAX_URL_LENGTH) { if (url.length() > MAX_URL_LENGTH) {
throw new IllegalArgumentException(String throw new IllegalArgumentException(String
@ -231,6 +250,7 @@ public class PushoverMessageBuilder {
} }
body.addFieldPart(MESSAGE_KEY_URL, new StringContentProvider(url), null); body.addFieldPart(MESSAGE_KEY_URL, new StringContentProvider(url), null);
String urlTitle = this.urlTitle;
if (urlTitle != null) { if (urlTitle != null) {
if (urlTitle.length() > MAX_URL_TITLE_LENGTH) { if (urlTitle.length() > MAX_URL_TITLE_LENGTH) {
throw new IllegalArgumentException( throw new IllegalArgumentException(
@ -245,9 +265,9 @@ public class PushoverMessageBuilder {
body.addFieldPart(MESSAGE_KEY_SOUND, new StringContentProvider(sound), null); body.addFieldPart(MESSAGE_KEY_SOUND, new StringContentProvider(sound), null);
} }
String attachment = this.attachment;
if (attachment != null) { if (attachment != null) {
String localAttachment = attachment; if (attachment.startsWith("http")) { // support data HTTP(S) scheme
if (localAttachment.startsWith("http")) { // support data HTTP(S) scheme
RawType rawImage = HttpUtil.downloadImage(attachment, 10000); RawType rawImage = HttpUtil.downloadImage(attachment, 10000);
if (rawImage == null) { if (rawImage == null) {
throw new IllegalArgumentException( throw new IllegalArgumentException(
@ -255,9 +275,9 @@ public class PushoverMessageBuilder {
} }
addFilePart(createTempFile(rawImage.getBytes()), addFilePart(createTempFile(rawImage.getBytes()),
contentType == null ? rawImage.getMimeType() : contentType); contentType == null ? rawImage.getMimeType() : contentType);
} else if (localAttachment.startsWith("data:")) { // support data URI scheme } else if (attachment.startsWith("data:")) { // support data URI scheme
try { try {
RawType rawImage = RawType.valueOf(localAttachment); RawType rawImage = RawType.valueOf(attachment);
addFilePart(createTempFile(rawImage.getBytes()), addFilePart(createTempFile(rawImage.getBytes()),
contentType == null ? rawImage.getMimeType() : contentType); contentType == null ? rawImage.getMimeType() : contentType);
} catch (IllegalArgumentException e) { } catch (IllegalArgumentException e) {

View File

@ -101,6 +101,7 @@ public class PushoverAccountHandler extends BaseThingHandler {
*/ */
public List<Sound> getSounds() { public List<Sound> getSounds() {
try { try {
PushoverAPIConnection connection = this.connection;
if (connection != null) { if (connection != null) {
List<Sound> sounds = connection.getSounds(); List<Sound> sounds = connection.getSounds();
if (sounds != null) { if (sounds != null) {
@ -145,6 +146,7 @@ public class PushoverAccountHandler extends BaseThingHandler {
} }
public boolean sendMessage(PushoverMessageBuilder messageBuilder) { public boolean sendMessage(PushoverMessageBuilder messageBuilder) {
PushoverAPIConnection connection = this.connection;
if (connection != null) { if (connection != null) {
try { try {
return connection.sendMessage(messageBuilder); return connection.sendMessage(messageBuilder);
@ -160,6 +162,7 @@ public class PushoverAccountHandler extends BaseThingHandler {
} }
public String sendPriorityMessage(PushoverMessageBuilder messageBuilder) { public String sendPriorityMessage(PushoverMessageBuilder messageBuilder) {
PushoverAPIConnection connection = this.connection;
if (connection != null) { if (connection != null) {
try { try {
return connection.sendPriorityMessage(messageBuilder); return connection.sendPriorityMessage(messageBuilder);
@ -175,6 +178,7 @@ public class PushoverAccountHandler extends BaseThingHandler {
} }
public boolean cancelPriorityMessage(String receipt) { public boolean cancelPriorityMessage(String receipt) {
PushoverAPIConnection connection = this.connection;
if (connection != null) { if (connection != null) {
try { try {
return connection.cancelPriorityMessage(receipt); return connection.cancelPriorityMessage(receipt);

View File

@ -64,6 +64,8 @@ sendMessageActionInputMessageLabel = Message
sendMessageActionInputMessageDescription = Message to be sent. sendMessageActionInputMessageDescription = Message to be sent.
sendMessageActionInputTitleLabel = Title sendMessageActionInputTitleLabel = Title
sendMessageActionInputTitleDescription = The title of the message. sendMessageActionInputTitleDescription = The title of the message.
sendMessageActionInputTTLLabel = Time to Live
sendMessageActionInputTTLDescription = The Time to Live of the message.
sendMessageToDeviceActionLabel = send a plain text message to a specific device sendMessageToDeviceActionLabel = send a plain text message to a specific device
sendMessageToDeviceActionDescription = This method is used to send a message to a specific device. sendMessageToDeviceActionDescription = This method is used to send a message to a specific device.
sendMessageActionInputDeviceLabel = Device sendMessageActionInputDeviceLabel = Device

View File

@ -18,6 +18,8 @@ import static org.junit.jupiter.api.Assertions.assertThrows;
import static org.mockito.ArgumentMatchers.any; import static org.mockito.ArgumentMatchers.any;
import static org.mockito.Mockito.when; import static org.mockito.Mockito.when;
import java.time.Duration;
import org.eclipse.jdt.annotation.NonNullByDefault; import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable; import org.eclipse.jdt.annotation.Nullable;
import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.BeforeEach;
@ -47,6 +49,7 @@ public class PushoverActionsTest {
private static final String URL = "https://www.test.com"; private static final String URL = "https://www.test.com";
private static final String URL_TITLE = "Some Link"; private static final String URL_TITLE = "Some Link";
private static final String RECEIPT = "12345"; private static final String RECEIPT = "12345";
private static final Duration TTL = Duration.ofSeconds(15);
private final ThingActions thingActionsStub = new ThingActions() { private final ThingActions thingActionsStub = new ThingActions() {
@Override @Override
@ -85,19 +88,26 @@ public class PushoverActionsTest {
} }
@Test @Test
public void testSendMessageWithoutTitle() { public void testSendMessage() {
pushoverThingActions.setThingHandler(mockPushoverAccountHandler); pushoverThingActions.setThingHandler(mockPushoverAccountHandler);
boolean sent = PushoverActions.sendMessage(pushoverThingActions, MESSAGE, null); boolean sent = PushoverActions.sendMessage(pushoverThingActions, MESSAGE);
assertThat(sent, is(true)); assertThat(sent, is(true));
} }
@Test @Test
public void testSendMessage() { public void testSendMessageWithTitle() {
pushoverThingActions.setThingHandler(mockPushoverAccountHandler); pushoverThingActions.setThingHandler(mockPushoverAccountHandler);
boolean sent = PushoverActions.sendMessage(pushoverThingActions, MESSAGE, TITLE); boolean sent = PushoverActions.sendMessage(pushoverThingActions, MESSAGE, TITLE);
assertThat(sent, is(true)); assertThat(sent, is(true));
} }
@Test
public void testSendMessageWithTitleAndTTL() {
pushoverThingActions.setThingHandler(mockPushoverAccountHandler);
boolean sent = PushoverActions.sendMessage(pushoverThingActions, MESSAGE, TITLE, TTL);
assertThat(sent, is(true));
}
// sendURLMessage // sendURLMessage
@Test @Test
public void testSendURLMessageThingActionsIsNotPushoverThingActions() { public void testSendURLMessageThingActionsIsNotPushoverThingActions() {
@ -132,6 +142,13 @@ public class PushoverActionsTest {
assertThat(sent, is(true)); assertThat(sent, is(true));
} }
@Test
public void testSendURLMessageWithTTL() {
pushoverThingActions.setThingHandler(mockPushoverAccountHandler);
boolean sent = PushoverActions.sendURLMessage(pushoverThingActions, MESSAGE, TITLE, URL, URL_TITLE, TTL);
assertThat(sent, is(true));
}
// sendPriorityMessage // sendPriorityMessage
@Test @Test
public void testSendPriorityMessageThingActionsIsNotPushoverThingActions() { public void testSendPriorityMessageThingActionsIsNotPushoverThingActions() {