[pushbullet] Remove unimplemented channels and thing parameters (#17482)

* [pushbullet] Remove unimplemented channels and thing parameters

Signed-off-by: jsetton <jeremy.setton@gmail.com>
Signed-off-by: Ciprian Pascu <contact@ciprianpascu.ro>
This commit is contained in:
Jeremy 2024-10-05 15:46:49 -04:00 committed by Ciprian Pascu
parent 405ed71f25
commit 4b7557f4da
8 changed files with 64 additions and 199 deletions

View File

@ -1,45 +1,25 @@
# Pushbullet Binding # Pushbullet Binding
The Pushbullet binding allows you to notify iOS, Android & Windows 10 Phone & Desktop devices of a message using the Pushbullet API web service. The Pushbullet binding allows you to notify supported devices of a message using the [Pushbullet API](https://docs.pushbullet.com).
To get started you need to [create an account](#creating-an-account) and [obtain an access token](#obtaining-an-access-token).
## Supported Things ## Supported Things
This binding supports a generic "bot" which is a representation of the client. | Thing | Type | Description |
| ----- | ----- | ------------------------- |
## Discovery | bot | Thing | Bot to send messages with |
This binding provides no discovery.
The desired bots must be configured manually or via a things file.
## Binding Configuration
The binding has no configuration options itself, all configuration is done at 'Things' level.
## Thing Configuration ## Thing Configuration
### Bot (`bot`) ### `bot`
The bot thing is used to send messages to other recipients. | Parameter | Required | Description |
It has the following parameters: | --------- | :------: | ---------------------------------------------------- |
| token | Yes | Access token obtained from the account settings page |
| Config | Description | Required | Advanced |
|------------|------------------------------------------------------------------|----------|----------|
| token | Pushbullet [API token](#obtaining-an-api-key) to send to devices | Yes | False |
| name | Explicit Name, for later use when the bot can receive messages | No | True |
| apiUrlBase | Address of own Pushbullet server, for testing purposes | No | True |
```java
Thing pushbullet:bot:r2d2 "R2D2" @ "Somewhere" [ token = "verysecretwonttellyou" ]
```
## Channels ## Channels
| Channel ID | Channel Description | Supported item type | Advanced | Currently the binding does not support any channels.
|------------|-------------------------------------------------|----------------------|----------|
| recipient | for later use when the bot can receive messages | String | False |
| title | for later use when the bot can receive messages | String | False |
| message | for later use when the bot can receive messages | String | False |
## Rule Action ## Rule Action
@ -71,110 +51,47 @@ For the `sendPushbulletNote` action, parameter `message` is required.
Likewise, for `sendPushbulletLink`, `url` and for `sendPushbulletFile`, `content` parameters are required. Likewise, for `sendPushbulletLink`, `url` and for `sendPushbulletFile`, `content` parameters are required.
Any other parameters for these actions are optional and can be set to `null`. Any other parameters for these actions are optional and can be set to `null`.
Examples: ## Full Example
demo.things:
```java
Thing pushbullet:bot:r2d2 [ token="<ACCESS_TOKEN>" ]
```
demo.rules
```java ```java
val actions = getActions("pushbullet", "pushbullet:bot:r2d2") val actions = getActions("pushbullet", "pushbullet:bot:r2d2")
// push a note
actions.sendPushbulletNote("someone@example.com", "Note Example", "This is the pushed note.") actions.sendPushbulletNote("someone@example.com", "Note Example", "This is the pushed note.")
// push a link
actions.sendPushbulletLink("someone@example.com", "Link Example", "This is the pushed link", "https://example.com") actions.sendPushbulletLink("someone@example.com", "Link Example", "This is the pushed link", "https://example.com")
// push a file
actions.sendPushbulletFile("someone@example.com", "File Example", "This is the pushed file", "https://example.com/image.png") actions.sendPushbulletFile("someone@example.com", "File Example", "This is the pushed file", "https://example.com/image.png")
actions.sendPushbulletFile("someone@example.com", null, null, "/path/to/somefile.pdf", "document.pdf") actions.sendPushbulletFile("someone@example.com", null, null, "/path/to/somefile.pdf", "document.pdf")
// use toFullString method to push the content of an Image item
actions.sendPushbulletFile("someone@example.com", ImageItem.state.toFullString) actions.sendPushbulletFile("someone@example.com", ImageItem.state.toFullString)
``` ```
## Full Example ## Creating an account
_Provide a full usage example based on textual configuration files (*.things, *.items, *.sitemap)._ A Pushbullet account is linked to either a Google or Facebook account.
pushbullet.things: - Go to the [Pushbullet](https://www.pushbullet.com) website.
- Select either "Sign up with Google" or "Sign up with Facebook".
```java
Thing pushbullet:bot:r2d2 "R2D2" @ "Somewhere" [ token = "verysecretwonttellyou" ]
```
pushbullet.items
```java
Switch Pushbullet_R2D2_Button "Pushbullet Action bot R2D2"
```
pushbullet.sitemap
```java
sitemap pushbullet label="Pushbullet"
{
Switch item=Pushbullet_R2D2_Button
}
```
pushbullet.rules
```java
rule "Pushbullet R2D2 changed"
when
Item Pushbullet_R2D2_Button changed
then
logInfo(filename, "Button R2D2 changed - OH2...")
if (Pushbullet_R2D2_Button.state == ON)
{
sendCommand(Pushbullet_R2D2_Button, OFF)
val actions = getActions("pushbullet", "pushbullet:bot:r2d2")
logInfo(filename, "Actions for 'R2D2' are: " + actions)
if (actions != null)
{
val result = actions.sendPushbulletNote("someone@example.com", "Title R2D2 OH2", "This has been sent by the new R2D2 bot")
logInfo(filename, "Result of send action is: " + result)
}
}
end
```
## Creating an account for your bot(s)
The pushbullet accounts are bound to either Google or Facebook accounts.
- Create a bot account with either Facebook or Google
- Go to "<https://www.pushbullet.com/>"
- Chose to either "Sign up with Google" or "Sign up with Facebook".
- Complete the signup process as guided by the pushbullet web site. - Complete the signup process as guided by the pushbullet web site.
- Continue with "Obtaining an API key".
## Obtaining an API key ## Obtaining an access token
The API keys are bound to the pushbullet account. An access token is linked to a Pushbullet account.
- Go to the pushbullet site. - Go to your [Pushbullet account settings](https://www.pushbullet.com/#settings/account) page.
- Log in with either your personal account or the one you created for your bot.
- Go to "<https://www.pushbullet.com/#settings/account>"
- Click on "Create Access Token". - Click on "Create Access Token".
- Copy the token created on the site. - Copy the created access token.
You must at least provide an API token (Private or Alias Key from Pushbullet.com) and a message in some manner before a message can be pushed.
All other parameters are optional.
If you use an alias key, the parameters (device, icon, sound, vibration) are overwritten by the alias setting on pushbullet.
## Rate limits ## Rate limits
As of 2019, free accounts have a limit of 100 pushes per month. As of 2024, free accounts are [limited](https://docs.pushbullet.com/#limits) to 500 pushes per month.
This action does not evaluate the rate limiting headers though. Going over will result in a warning message being logged indicating when your account rate limit will reset.
## Translation
This project is being translated on transifex.
If you want to help, please join the project at the URL:
- <https://www.transifex.com/hakan42/openhab-binding-pushbullet/dashboard/>
## Libraries
This action has been written without using libraries as jpushbullet or jpushbullet2.
Both of those libraries use various libraries themselves which makes integrating them into openHAB a challenge.
## pushbullet API
- <https://docs.pushbullet.com/>
- <https://docs.pushbullet.com/#push-limit>

View File

@ -34,20 +34,19 @@ public class PushbulletBindingConstants {
public static final Set<ThingTypeUID> SUPPORTED_THING_TYPES_UIDS = Set.of(THING_TYPE_BOT); public static final Set<ThingTypeUID> SUPPORTED_THING_TYPES_UIDS = Set.of(THING_TYPE_BOT);
// List of all Channel ids
public static final String RECIPIENT = "recipient";
public static final String TITLE = "title";
public static final String MESSAGE = "message";
// Thing properties // Thing properties
public static final String PROPERTY_EMAIL = "email"; public static final String PROPERTY_EMAIL = "email";
public static final String PROPERTY_NAME = "name"; public static final String PROPERTY_NAME = "name";
// Binding logic constants // Binding logic constants
public static final String API_BASE_URL = "https://api.pushbullet.com/v2";
public static final String API_ENDPOINT_PUSHES = "/pushes"; public static final String API_ENDPOINT_PUSHES = "/pushes";
public static final String API_ENDPOINT_UPLOAD_REQUEST = "/upload-request"; public static final String API_ENDPOINT_UPLOAD_REQUEST = "/upload-request";
public static final String API_ENDPOINT_USERS_ME = "/users/me"; public static final String API_ENDPOINT_USERS_ME = "/users/me";
public static final String HEADER_RATELIMIT_RESET = "X-Ratelimit-Reset";
public static final String IMAGE_FILE_NAME = "image.jpg"; public static final String IMAGE_FILE_NAME = "image.jpg";
public static final int MAX_UPLOAD_SIZE = 26214400; public static final int MAX_UPLOAD_SIZE = 26214400;

View File

@ -13,7 +13,6 @@
package org.openhab.binding.pushbullet.internal; package org.openhab.binding.pushbullet.internal;
import org.eclipse.jdt.annotation.NonNullByDefault; import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
/** /**
* The {@link PushbulletConfiguration} class contains fields mapping thing configuration parameters. * The {@link PushbulletConfiguration} class contains fields mapping thing configuration parameters.
@ -24,21 +23,9 @@ import org.eclipse.jdt.annotation.Nullable;
@NonNullByDefault @NonNullByDefault
public class PushbulletConfiguration { public class PushbulletConfiguration {
private @Nullable String name;
private String token = ""; private String token = "";
private String apiUrlBase = "https://api.pushbullet.com/v2";
public @Nullable String getName() {
return name;
}
public String getAccessToken() { public String getAccessToken() {
return token; return token;
} }
public String getApiUrlBase() {
return apiUrlBase;
}
} }

View File

@ -12,6 +12,8 @@
*/ */
package org.openhab.binding.pushbullet.internal; package org.openhab.binding.pushbullet.internal;
import static org.openhab.binding.pushbullet.internal.PushbulletBindingConstants.*;
import java.time.Instant; import java.time.Instant;
import java.util.Objects; import java.util.Objects;
import java.util.concurrent.ExecutionException; import java.util.concurrent.ExecutionException;
@ -49,12 +51,11 @@ import com.google.gson.JsonSyntaxException;
*/ */
@NonNullByDefault @NonNullByDefault
public class PushbulletHttpClient { public class PushbulletHttpClient {
private static final String AGENT = "openHAB/" + OpenHAB.getVersion();
private static final String USER_AGENT = "openHAB/" + OpenHAB.getVersion();
private static final int TIMEOUT = 30; // in seconds private static final int TIMEOUT = 30; // in seconds
private static final String HEADER_RATELIMIT_RESET = "X-Ratelimit-Reset";
private final Logger logger = LoggerFactory.getLogger(PushbulletHttpClient.class); private final Logger logger = LoggerFactory.getLogger(PushbulletHttpClient.class);
private final Gson gson = new GsonBuilder().registerTypeHierarchyAdapter(Instant.class, new InstantDeserializer()) private final Gson gson = new GsonBuilder().registerTypeHierarchyAdapter(Instant.class, new InstantDeserializer())
@ -95,7 +96,7 @@ public class PushbulletHttpClient {
*/ */
public <T> T executeRequest(String apiEndpoint, @Nullable Object body, Class<T> responseType) public <T> T executeRequest(String apiEndpoint, @Nullable Object body, Class<T> responseType)
throws PushbulletApiException { throws PushbulletApiException {
String url = config.getApiUrlBase() + apiEndpoint; String url = API_BASE_URL + apiEndpoint;
String accessToken = config.getAccessToken(); String accessToken = config.getAccessToken();
Request request = newRequest(url).header("Access-Token", accessToken); Request request = newRequest(url).header("Access-Token", accessToken);
@ -142,7 +143,7 @@ public class PushbulletHttpClient {
* @return the new Request object with default parameters * @return the new Request object with default parameters
*/ */
private Request newRequest(String url) { private Request newRequest(String url) {
return httpClient.newRequest(url).agent(AGENT).timeout(TIMEOUT, TimeUnit.SECONDS); return httpClient.newRequest(url).agent(USER_AGENT).timeout(TIMEOUT, TimeUnit.SECONDS);
} }
/** /**

View File

@ -78,12 +78,7 @@ public class PushbulletHandler extends BaseThingHandler {
@Override @Override
public void handleCommand(ChannelUID channelUID, Command command) { public void handleCommand(ChannelUID channelUID, Command command) {
logger.debug("About to handle {} on {}", command, channelUID); // do nothing
// Future improvement: If recipient is already set, send a push on a command channel change
// check reconnect channel of the unifi binding for that
logger.debug("The Pushbullet binding is a read-only binding and cannot handle command '{}'.", command);
} }
@Override @Override

View File

@ -7,20 +7,8 @@ thing-type.pushbullet.bot.label = Pushbullet Bot
thing-type.pushbullet.bot.description = Bot to send messages with. thing-type.pushbullet.bot.description = Bot to send messages with.
# thing types config # thing types config
thing-type.config.pushbullet.bot.name.label = Name
thing-type.config.pushbullet.bot.name.description = Explicit Name of Bot, if wanted
thing-type.config.pushbullet.bot.token.label = Access Token thing-type.config.pushbullet.bot.token.label = Access Token
thing-type.config.pushbullet.bot.token.description = Access token obtained from the account settings page thing-type.config.pushbullet.bot.token.description = Access token obtained from the account settings page.
thing-type.config.pushbullet.bot.apiUrlBase.label = API Server
thing-type.config.pushbullet.bot.apiUrlBase.description = The Pushbullet API Server to use, for local testing
# channel types
channel-type.pushbullet.recipient-channel.label = Recipient
channel-type.pushbullet.recipient-channel.description = Mail address or Channel Name
channel-type.pushbullet.title-channel.label = Title
channel-type.pushbullet.title-channel.description = Title of the message
channel-type.pushbullet.message-channel.label = Message
channel-type.pushbullet.message-channel.description = The text that is to be sent
# action # action
actionSendPushbulletNoteLabel = publish a Pushbullet message actionSendPushbulletNoteLabel = publish a Pushbullet message
@ -42,6 +30,3 @@ actionSendPushbulletFileInputContent = File Content
actionSendPushbulletFileInputContentDesc = The File Content to publish actionSendPushbulletFileInputContentDesc = The File Content to publish
actionSendPushbulletFileInputName = File Name actionSendPushbulletFileInputName = File Name
actionSendPushbulletFileInputNameDesc = The File Name to publish actionSendPushbulletFileInputNameDesc = The File Name to publish
# error texts
offline.conf-error-httpresponseexception = The pushbullet server reported an error, possibly an expired token. Check on web site

View File

@ -8,51 +8,18 @@
<label>Pushbullet Bot</label> <label>Pushbullet Bot</label>
<description>Bot to send messages with.</description> <description>Bot to send messages with.</description>
<channels> <properties>
<channel id="recipient" typeId="recipient-channel"/> <property name="thingTypeVersion">1</property>
<channel id="title" typeId="title-channel"/> </properties>
<channel id="message" typeId="message-channel"/>
</channels>
<config-description> <config-description>
<parameter name="name" type="text" required="false">
<label>Name</label>
<description>Explicit Name of Bot, if wanted</description>
<advanced>true</advanced>
</parameter>
<parameter name="token" type="text" required="true"> <parameter name="token" type="text" required="true">
<label>Access Token</label> <label>Access Token</label>
<description>Access token obtained from the account settings page</description> <description>Access token obtained from the account settings page.</description>
</parameter>
<parameter name="apiUrlBase" type="text" required="true">
<label>API Server</label>
<description>The Pushbullet API Server to use, for local testing</description>
<default>https://api.pushbullet.com/v2</default>
<advanced>true</advanced>
</parameter> </parameter>
</config-description> </config-description>
</thing-type> </thing-type>
<channel-type id="recipient-channel">
<item-type>String</item-type>
<label>Recipient</label>
<description>Mail address or Channel Name</description>
</channel-type>
<channel-type id="title-channel">
<item-type>String</item-type>
<label>Title</label>
<description>Title of the message</description>
</channel-type>
<channel-type id="message-channel">
<item-type>String</item-type>
<label>Message</label>
<description>The text that is to be sent</description>
</channel-type>
</thing:thing-descriptions> </thing:thing-descriptions>

View File

@ -0,0 +1,14 @@
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<update:update-descriptions xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:update="https://openhab.org/schemas/update-description/v1.0.0"
xsi:schemaLocation="https://openhab.org/schemas/update-description/v1.0.0 https://openhab.org/schemas/update-description-1.0.0.xsd">
<thing-type uid="pushbullet:bot">
<instruction-set targetVersion="1">
<remove-channel id="recipient"/>
<remove-channel id="title"/>
<remove-channel id="message"/>
</instruction-set>
</thing-type>
</update:update-descriptions>