[plex] Initial contribution (#15057)

* aronbeurskens plex baseline

Signed-off-by: Michael Lobstein <michael.lobstein@gmail.com>

* review changes

Signed-off-by: Michael Lobstein <michael.lobstein@gmail.com>

* Clean-up PlexApiConnector exception handling

Signed-off-by: Michael Lobstein <michael.lobstein@gmail.com>

* Additional clean-up

Signed-off-by: Michael Lobstein <michael.lobstein@gmail.com>

* review changes

Signed-off-by: Michael Lobstein <michael.lobstein@gmail.com>

* review changes

Signed-off-by: Michael Lobstein <michael.lobstein@gmail.com>

---------

Signed-off-by: Michael Lobstein <michael.lobstein@gmail.com>
This commit is contained in:
mlobstein 2023-06-10 00:22:02 -05:00 committed by GitHub
parent ddffda3522
commit 44a78b99f5
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
27 changed files with 2477 additions and 0 deletions

View File

@ -267,6 +267,7 @@
/bundles/org.openhab.binding.pjlinkdevice/ @nils /bundles/org.openhab.binding.pjlinkdevice/ @nils
/bundles/org.openhab.binding.playstation/ @FluBBaOfWard /bundles/org.openhab.binding.playstation/ @FluBBaOfWard
/bundles/org.openhab.binding.plclogo/ @falkena /bundles/org.openhab.binding.plclogo/ @falkena
/bundles/org.openhab.binding.plex/ @aronbeurskens
/bundles/org.openhab.binding.plugwise/ @wborn /bundles/org.openhab.binding.plugwise/ @wborn
/bundles/org.openhab.binding.plugwiseha/ @lsiepel /bundles/org.openhab.binding.plugwiseha/ @lsiepel
/bundles/org.openhab.binding.powermax/ @lolodomo /bundles/org.openhab.binding.powermax/ @lolodomo

View File

@ -1326,6 +1326,11 @@
<artifactId>org.openhab.binding.plclogo</artifactId> <artifactId>org.openhab.binding.plclogo</artifactId>
<version>${project.version}</version> <version>${project.version}</version>
</dependency> </dependency>
<dependency>
<groupId>org.openhab.addons.bundles</groupId>
<artifactId>org.openhab.binding.plex</artifactId>
<version>${project.version}</version>
</dependency>
<dependency> <dependency>
<groupId>org.openhab.addons.bundles</groupId> <groupId>org.openhab.addons.bundles</groupId>
<artifactId>org.openhab.binding.plugwise</artifactId> <artifactId>org.openhab.binding.plugwise</artifactId>

View File

@ -0,0 +1,13 @@
This content is produced and maintained by the openHAB project.
* Project home: https://www.openhab.org
== Declared Project Licenses
This program and the accompanying materials are made available under the terms
of the Eclipse Public License 2.0 which is available at
https://www.eclipse.org/legal/epl-2.0/.
== Source Code
https://github.com/openhab/openhab-addons

View File

@ -0,0 +1,167 @@
# PLEX Binding
This binding can read information from multiple PLEX players connected to a PLEX server.
It can be used for multiple scenarios:
* Drive light changes based on player status. For instances turn off the lights when movie starts playing and turn them back on when movie is stopped/paused
* Create a page that displays currently played media of one or more player connected to the server.
* Send social media messages when player plays new media
* Inform what the end time of the currently played media is
The binding can also control `PLAY/PAUSE/NEXT/PREVIOUS` the players which can be used for:
* Start playing some music when someone enters a room
* Pause the movie when motion is detected
## Supported Things
This binding supports 2 things.
- `server`: The PLEX server will act as a bridge to read out the information from all connected players
- `player`: A PLEX client of any type / os connected to the server.
## Discovery
For the auto discovery to work correctly you first need to configure and add the `PLEX Server` Thing.
Next step is to *PLAY* something on the desired player. Only when media is played on the player it will show up in the auto discovery!
## Thing Configuration
The PLEX Server needs to be configured first. The hostname of the PLEX server is mandatory and the either the PLEX token (recommended) or the username/password of the PLEX server (not recommended).
Then find the PLEX token please follow the instructions from the PLEX support forum:
1. Sign in to your Plex account in Plex Web App
2. Browse to a library item and view the XML for it
3. Look in the URL and find the token as the X-Plex-Token value
### `PLEX Server` Thing Configuration
| Name | Type | Description | Default | Required | Advanced |
|-------------|---------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|---------|----------|---------|
| host | text | PLEX host name or IP address | N/A | yes | no |
| portNumber | integer | Port Number (leave blank if PLEX installed on default port) | 32400 | no | no |
| refreshRate | integer | Interval in seconds at which PLEX server status is polled | 5 | no | no |
| username | text | If you're using Plex Home you need to supply the username and password of your Plex account here. If you don't want to enter your credentials you can also directly set your account token below instead. | N/A | no | no |
| password | text | If you're using Plex Home you need to supply the username and password of your Plex account here. If you don't want to enter your credentials you can also directly set your account token below instead. | N/A | no | no |
| token | text | The authentication token when username/password is left blank | N/A | no | no |
### `PLEX Player` Thing Configuration
You can add multiple PLEX players. You can choose to find the player by autodiscovery or add them manually.
#### Autodiscovery
Turn on the player you want to add and *play* some media on it. Navigate to `/settings/things/add/plex` and start the auto discover.
The player will be found and you can add it.
#### Manual adding a player Thing
When you want to add them manually go to the following url [https://plex.tv/devices.xml] and login when needed.
It will display the following XML file.
```xml
<MediaContainer publicAddress="XXX.XXX.XXX.XXX">
<Device name="iPhone" publicAddress="XXX.XXX.XXX.XXX" product="Plex for iOS" productVersion="8.4" platform="iOS" platformVersion="15.5" device="iPhone" model="14,5" vendor="Apple" provides="client,controller,sync-target,player,pubsub-player,provider-playback" clientIdentifier="B03466F7-BEEB-405F-A315-C7BBAA2D3FAE" version="8.4" id="547394701" token="XXX" createdAt="1633194400" lastSeenAt="1655715607" screenResolution="1170x2532" screenDensity="3">
<SyncList itemsCompleteCount="0" totalSize="0" version="2"/>
<Connection uri="http://192.168.1.194:32500"/>
</Device>
<Device name="Chrome" publicAddress="XXX.XXX.XXX.XXX" product="Plex Web" productVersion="4.83.2" platform="Chrome" platformVersion="102.0" device="Linux" model="hosted" vendor="" provides="" clientIdentifier="e29nk766fd48skpm8uuu1x9l" version="4.83.2" id="660497510" token="XXX" createdAt="1655714525" lastSeenAt="1655714526" screenResolution="1920x975,1920x1080" screenDensity=""> </Device>
<Device name="MY PLEX SERVER" publicAddress="XXX.XXX.XXX.XXX" product="Plex Media Server" productVersion="1.27.0.5897-3940636f2" platform="Linux" platformVersion="20.04.4 LTS (Focal Fossa)" device="PC" model="x86_64" vendor="Ubuntu" provides="server" clientIdentifier="906a992fc4c5722595f36732838bcc330700b2af" version="1.27.0.5897-3940636f2" id="282416069" token="XXX" createdAt="1560609688" lastSeenAt="1655709241" screenResolution="" screenDensity="">
<Connection uri="http://[2001:1c05:380f:5700:642:1aff:fe08:1c22]:32400"/>
<Connection uri="http://192.168.1.2:32400"/>
</Device>
<Device name="BRAVIA 4K UR2" publicAddress="XXX.XXX.XXX.XXX" product="Plex for Android (TV)" productVersion="8.26.2.29389" platform="Android" platformVersion="9" device="BRAVIA 4K UR2" model="BRAVIA_UR2_4K" vendor="Sony" provides="player,pubsub-player,controller" clientIdentifier="97d2510bd3942159-com-plexapp-android" version="8.26.2.29389" id="403947762" token="XXX" createdAt="1601489892" lastSeenAt="1655701261" screenResolution="1920x1080" screenDensity="320">
<Connection uri="http://192.168.1.19:32500"/>
</Device>
<Device name="SHIELD Android TV" publicAddress="XXX.XXX.XXX.XXX" product="Plex for Android (TV)" productVersion="8.26.2.29389" platform="Android" platformVersion="11" device="SHIELD Android TV" model="mdarcy" vendor="NVIDIA" provides="player,pubsub-player,controller" clientIdentifier="2c098c67afd0ca79-com-plexapp-android" version="8.26.2.29389" id="508114660" token="XXX" createdAt="1625317867" lastSeenAt="1655693424" screenResolution="1920x1080" screenDensity="320">
<Connection uri="http://192.168.1.6:32500"/>
</Device>
<Device name="iPad" publicAddress="XXX.XXX.XXX.XXX" product="Plex for iOS" productVersion="8.0" platform="iOS" platformVersion="14.7.1" device="iPad" model="5,3" vendor="Apple" provides="client,controller,sync-target,player,pubsub-player,provider-playback" clientIdentifier="D9629798-4B91-4375-8844-C62400573E42" version="8.0" id="617442968" token="XXX" createdAt="1646760020" lastSeenAt="1647973844" screenResolution="2048x1536" screenDensity="2">
<SyncList itemsCompleteCount="0" totalSize="0" version="2"/>
<Connection uri="http://192.168.1.220:32500"/>
</Device>
<Device name="MacBook-Pro.local" publicAddress="XXX.XXX.XXX.XXX" product="Plex for Mac" productVersion="1.41.0.2876-e960c9ca" platform="osx" platformVersion="12.2" device="" model="standalone" vendor="" provides="client,player,pubsub-player" clientIdentifier="5ehipgz2ca60ikqnv9jrgojx" version="1.41.0.2876-e960c9ca" id="507110703" token="XXX" createdAt="1625081186" lastSeenAt="1647973509" screenResolution="1680x1050,1680x1050" screenDensity=""> </Device>
</MediaContainer>
```
Find the `Device` block of the player you want to add and fill in the `clientIdentifier` as `playerID`
| Name | Type | Description | Default | Required | Advanced |
|-------------|---------|--------------------------------------------------------------------------------------------|---------|----------|---------|
| playerID | text | The unique identifier of the player. `clientIdentifier` from [https://plex.tv/devices.xml] | N/A | yes | no |
## Channels
The PLEX Player supports the following channels:
| Channel | Type | Read/Write | Description |
|----------------------|----------|------------|-----------------------------------------------------------------------|
| currentPlayers | Number | RO | The number of players currently configured to watch on PLEX |
| currentPlayersActive | Number | RO | The number of players currently being used on PLEX |
| state | String | RO | The current state of the Player (BUFFERING, PLAYING, PAUSED, STOPPED) |
| power | Switch | RO | The power status of the player |
| title | String | RO | The title of media that is playing |
| type | String | RO | The current type of playing media |
| endtime | DateTime | RO | Time at which the media that is playing will end |
| progress | Dimmer | RO | The current progress of playing media |
| art | String | RO | The URL of the background art for currently playing media |
| thumb | String | RO | The URL of the cover art for currently playing media |
| player | Player | RW | The control channel for the player `PLAY/PAUSE/NEXT/PREVIOUS` |
## Full Example
`.things` file:
```java
Bridge plex:server:plexrServer "Bridge Plex : Plex" [host="IP.Address.Or.Hostname", token="SadhjsajjA3AG", refreshRate=5]
{
Thing plex:player:MyViewerName01 "My Viewer Name 01" [playerID="ClientIdentifierFromDevices.XML1"]
Thing plex:player:MyViewerName02 "My Viewer Name 02" [playerID="ClientIdentifierFromDevices.XML2"]
}
```
`.items` file
```java
String BridgePlexCurrent "Current players" {channel="plex:server:plexrServer:currentPlayers"}
String BridgePlexCurrentActive "Current players active" {channel="plex:server:plexrServer:currentPlayersActive"}
Switch PlexTVPower01 "Power" {channel="plex:player:MyViewerName01:power"}
String PlexTVStatus01 "Status [%s]" {channel="plex:player:MyViewerName01:state"}
String PlexTVTitle01 "Title [%s]" {channel="plex:player:MyViewerName01:title"}
String PlexTVType01 "Type [%s]" {channel="plex:player:MyViewerName01:type"}
String PlexTVEndTime01 "End time" {channel="plex:player:MyViewerName01:endtime"}
Dimmer PlexTVProgress01 "Progress [%.1f%%]" {channel="plex:player:MyViewerName01:progress"}
String PlexTVCover1 "Cover" {channel="plex:player:MyViewerName01:thumb"}
String ShellArt01 "Background art" {channel="plex:player:MyViewerName01:art"}
Switch PlexTVPower02 "Power" {channel="plex:player:MyViewerName02:power"}
String PlexTVStatus02 "Status [%s]" {channel="plex:player:MyViewerName02:state"}
String PlexTVTitle02 "Title [%s]" {channel="plex:player:MyViewerName02:title"}
String PlexTVType02 "Type [%s]" {channel="plex:player:MyViewerName02:type"}
String PlexTVEndTime02 "End time" {channel="plex:player:MyViewerName02:endtime"}
Dimmer PlexTVProgress02 "Progress [%.1f%%]" {channel="plex:player:MyViewerName02:progress"}
String PlexTVCover2 "Cover" {channel="plex:player:MyViewerName02:thumb"}
String ShellArt02 "Background art" {channel="plex:player:MyViewerName02:art"}
```
`.rules` file
```java
rule "Send telegram with title for My Viewer Name 01"
when
Item PlexTVTitle01 changed
then
val telegramActionPlexBot = getActions("telegram","telegram:telegramBot:PlexBot")
telegramActionPlexBot.sendTelegram("Bedroom Roku is watching %s", PlexTVTitle01.state.toString)
end
rule "Send telegram with title for My Viewer Name 02"
when
Item PlexTVTitle02 changed
then
val telegramActionPlexBot = getActions("telegram","telegram:telegramBot:PlexBot")
telegramActionPlexBot.sendTelegram("Bedroom Roku is watching %s", PlexTVTitle02.state.toString)
end
```

View File

@ -0,0 +1,17 @@
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://maven.apache.org/POM/4.0.0"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>org.openhab.addons.bundles</groupId>
<artifactId>org.openhab.addons.reactor.bundles</artifactId>
<version>4.0.0-SNAPSHOT</version>
</parent>
<artifactId>org.openhab.binding.plex</artifactId>
<name>openHAB Add-ons :: Bundles :: Plex Binding</name>
</project>

View File

@ -0,0 +1,10 @@
<?xml version="1.0" encoding="UTF-8"?>
<features name="org.openhab.binding.plex-${project.version}" xmlns="http://karaf.apache.org/xmlns/features/v1.4.0">
<repository>mvn:org.openhab.core.features.karaf/org.openhab.core.features.karaf.openhab-core/${ohc.version}/xml/features</repository>
<feature name="openhab-binding-plex" description="Plex Binding" version="${project.version}">
<feature>openhab-runtime-base</feature>
<feature dependency="true">openhab.tp-jaxb</feature>
<bundle start-level="80">mvn:org.openhab.addons.bundles/org.openhab.binding.plex/${project.version}</bundle>
</feature>
</features>

View File

@ -0,0 +1,58 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.plex.discovery;
import static org.openhab.binding.plex.internal.PlexBindingConstants.*;
import java.util.HashMap;
import java.util.Map;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.openhab.binding.plex.internal.handler.PlexServerHandler;
import org.openhab.core.config.discovery.AbstractDiscoveryService;
import org.openhab.core.config.discovery.DiscoveryResult;
import org.openhab.core.config.discovery.DiscoveryResultBuilder;
import org.openhab.core.thing.ThingTypeUID;
import org.openhab.core.thing.ThingUID;
/**
* @author Brian Homeyer - Initial contribution
* @author Aron Beurskens - Binding development
*/
@NonNullByDefault
public class PlexDiscoveryService extends AbstractDiscoveryService {
private final PlexServerHandler bridgeHandler;
public PlexDiscoveryService(PlexServerHandler bridgeHandler) {
super(SUPPORTED_THING_TYPES_UIDS, 10, false);
this.bridgeHandler = bridgeHandler;
}
@Override
protected void startScan() {
for (String machineId : bridgeHandler.getAvailablePlayers()) {
ThingUID bridgeUID = bridgeHandler.getThing().getUID();
ThingTypeUID thingTypeUID = UID_PLAYER;
ThingUID playerThingUid = new ThingUID(UID_PLAYER, bridgeUID, machineId);
Map<String, Object> properties = new HashMap<>();
properties.put(CONFIG_PLAYER_ID, machineId);
DiscoveryResult discoveryResult = DiscoveryResultBuilder.create(playerThingUid).withThingType(thingTypeUID)
.withProperties(properties).withBridge(bridgeUID).withRepresentationProperty(CONFIG_PLAYER_ID)
.withLabel("PLEX Player (" + machineId + ")").build();
thingDiscovered(discoveryResult);
}
}
}

View File

@ -0,0 +1,77 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.plex.internal;
import java.util.Collections;
import java.util.Set;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.openhab.core.thing.ThingTypeUID;
/**
* The {@link PlexBindingConstants} class defines common constants, which are
* used across the whole binding.
*
* @author Brian Homeyer - Initial contribution
* @author Aron Beurskens - Binding development
*/
@NonNullByDefault
public class PlexBindingConstants {
private static final String BINDING_ID = "plex";
// Bridge thing
public static final String THING_TYPE_SERVER = "server";
public static final ThingTypeUID UID_SERVER = new ThingTypeUID(BINDING_ID, THING_TYPE_SERVER);
public static final Set<ThingTypeUID> SUPPORTED_SERVER_THING_TYPES_UIDS = Set.of(UID_SERVER);
// Monitor things
public static final String THING_TYPE_PLAYER = "player";
public static final ThingTypeUID UID_PLAYER = new ThingTypeUID(BINDING_ID, THING_TYPE_PLAYER);
// Collection of monitor thing types
public static final Set<ThingTypeUID> SUPPORTED_PLAYER_THING_TYPES_UIDS = Set.of(UID_PLAYER);
// Collection of all supported thing types
public static final Set<ThingTypeUID> SUPPORTED_THING_TYPES_UIDS = Collections.unmodifiableSet(
Stream.concat(SUPPORTED_PLAYER_THING_TYPES_UIDS.stream(), SUPPORTED_SERVER_THING_TYPES_UIDS.stream())
.collect(Collectors.toSet()));
// General purpose stuff
public static final int DEFAULT_REFRESH_PERIOD_SEC = 5;
// Config parameters
// Server
public static final String CONFIG_HOST = "host";
public static final String CONFIG_PORT_NUMBER = "portNumber";
public static final String CONFIG_TOKEN = "token";
public static final String CONFIG_REFRESH_RATE = "refreshRate";
// Player parameters
public static final String CONFIG_PLAYER_ID = "playerID";
public static final String CONFIG_PLAYER_NAME = "playerName";
// List of all Channel ids
// Server
public static final String CHANNEL_SERVER_COUNT = "currentPlayers";
public static final String CHANNEL_SERVER_COUNTACTIVE = "currentPlayersActive";
// Player
public static final String CHANNEL_PLAYER_STATE = "state";
public static final String CHANNEL_PLAYER_TITLE = "title";
public static final String CHANNEL_PLAYER_TYPE = "type";
public static final String CHANNEL_PLAYER_POWER = "power";
public static final String CHANNEL_PLAYER_ART = "art";
public static final String CHANNEL_PLAYER_THUMB = "thumb";
public static final String CHANNEL_PLAYER_PROGRESS = "progress";
public static final String CHANNEL_PLAYER_ENDTIME = "endtime";
public static final String CHANNEL_PLAYER_CONTROL = "player";
public static final String CHANNEL_PLAYER_USER = "user";
}

View File

@ -0,0 +1,91 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.plex.internal;
import static org.openhab.binding.plex.internal.PlexBindingConstants.*;
import java.util.Hashtable;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
import org.openhab.binding.plex.discovery.PlexDiscoveryService;
import org.openhab.binding.plex.internal.handler.PlexPlayerHandler;
import org.openhab.binding.plex.internal.handler.PlexServerHandler;
import org.openhab.core.config.discovery.DiscoveryService;
import org.openhab.core.io.net.http.HttpClientFactory;
import org.openhab.core.thing.Bridge;
import org.openhab.core.thing.Thing;
import org.openhab.core.thing.ThingTypeUID;
import org.openhab.core.thing.binding.BaseThingHandlerFactory;
import org.openhab.core.thing.binding.ThingHandler;
import org.openhab.core.thing.binding.ThingHandlerFactory;
import org.osgi.framework.ServiceRegistration;
import org.osgi.service.component.annotations.Activate;
import org.osgi.service.component.annotations.Component;
import org.osgi.service.component.annotations.Reference;
/**
* The {@link PlexHandlerFactory} is responsible for creating things and thing
* handlers.
*
* @author Brian Homeyer - Initial contribution
* @author Aron Beurskens - Binding development
*/
@NonNullByDefault
@Component(configurationPid = "binding.plex", service = ThingHandlerFactory.class)
public class PlexHandlerFactory extends BaseThingHandlerFactory {
private final HttpClientFactory httpClientFactory;
private @Nullable ServiceRegistration<?> plexDiscoveryServiceRegistration;
@Activate
public PlexHandlerFactory(final @Reference HttpClientFactory httpClientFactory) {
this.httpClientFactory = httpClientFactory;
}
@Override
public boolean supportsThingType(ThingTypeUID thingTypeUID) {
return SUPPORTED_THING_TYPES_UIDS.contains(thingTypeUID);
}
@Override
protected @Nullable ThingHandler createHandler(Thing thing) {
ThingTypeUID thingTypeUID = thing.getThingTypeUID();
if (SUPPORTED_SERVER_THING_TYPES_UIDS.contains(thingTypeUID)) {
PlexServerHandler handler = new PlexServerHandler((Bridge) thing, httpClientFactory);
registerPlexDiscoveryService(handler);
return handler;
} else if (SUPPORTED_PLAYER_THING_TYPES_UIDS.contains(thingTypeUID)) {
return new PlexPlayerHandler(thing);
}
return null;
}
@Override
protected synchronized void removeHandler(ThingHandler thingHandler) {
if (thingHandler instanceof PlexServerHandler) {
ServiceRegistration<?> plexDiscoveryServiceRegistration = this.plexDiscoveryServiceRegistration;
if (plexDiscoveryServiceRegistration != null) {
// remove discovery service, if bridge handler is removed
plexDiscoveryServiceRegistration.unregister();
}
}
}
private void registerPlexDiscoveryService(PlexServerHandler handler) {
PlexDiscoveryService discoveryService = new PlexDiscoveryService(handler);
if (bundleContext != null) {
this.plexDiscoveryServiceRegistration = bundleContext.registerService(DiscoveryService.class.getName(),
discoveryService, new Hashtable<>());
}
}
}

View File

@ -0,0 +1,41 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.plex.internal;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.openhab.core.thing.binding.BaseDynamicStateDescriptionProvider;
import org.openhab.core.thing.i18n.ChannelTypeI18nLocalizationService;
import org.openhab.core.thing.type.DynamicStateDescriptionProvider;
import org.osgi.service.component.annotations.Component;
import org.osgi.service.component.annotations.Reference;
/**
* Dynamic provider of state options while leaving other state description fields as original.
*
* @author Brian Homeyer - Initial contribution
* @author Aron Beurskens - Binding development
*/
@Component(service = { DynamicStateDescriptionProvider.class, PlexStateDescriptionOptionProvider.class })
@NonNullByDefault
public class PlexStateDescriptionOptionProvider extends BaseDynamicStateDescriptionProvider {
@Reference
protected void setChannelTypeI18nLocalizationService(
final ChannelTypeI18nLocalizationService channelTypeI18nLocalizationService) {
this.channelTypeI18nLocalizationService = channelTypeI18nLocalizationService;
}
protected void unsetChannelTypeI18nLocalizationService(
final ChannelTypeI18nLocalizationService channelTypeI18nLocalizationService) {
this.channelTypeI18nLocalizationService = null;
}
}

View File

@ -0,0 +1,27 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.plex.internal.config;
import org.eclipse.jdt.annotation.NonNullByDefault;
/**
* The {@link PlexPlayerConfiguration} is the class used to match the
* thing configuration.
*
* @author Brian Homeyer - Initial contribution
* @author Aron Beurskens - Binding development
*/
@NonNullByDefault
public class PlexPlayerConfiguration {
public String playerID = "";
}

View File

@ -0,0 +1,33 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.plex.internal.config;
import org.eclipse.jdt.annotation.NonNullByDefault;
/**
* The {@link PlexServerConfiguration} is the class used to match the
* bridge configuration.
*
* @author Brian Homeyer - Initial contribution
* @author Aron Beurskens - Binding development
*/
@NonNullByDefault
public class PlexServerConfiguration {
public String host = "";
public Integer portNumber = 32400;
public String token = "";
public Integer refreshRate = 5;
public String username = "";
public String password = "";
public String scheme = "";
}

View File

@ -0,0 +1,277 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.plex.internal.dto;
import java.util.List;
import com.thoughtworks.xstream.annotations.XStreamAlias;
import com.thoughtworks.xstream.annotations.XStreamAsAttribute;
import com.thoughtworks.xstream.annotations.XStreamImplicit;
/**
*
* @author Brian Homeyer - Initial contribution
* @author Aron Beurskens - Binding development
*/
@XStreamAlias("MediaContainer")
public class MediaContainer {
@XStreamAsAttribute
private Integer size;
@XStreamImplicit
@XStreamAsAttribute
private List<Video> video = null;
@XStreamImplicit
@XStreamAsAttribute
private List<Track> track = null;
@XStreamImplicit
@XStreamAsAttribute
private List<Device> device = null;
public List<Device> getDevice() {
return device;
}
public Integer getSize() {
return size;
}
public List<Video> getVideo() {
return video;
}
public List<Track> getTrack() {
return track;
}
/**
* Returns a list of video or track objects, depends on what is playing
*/
public List<? extends MediaType> getMediaTypes() {
if (video != null) {
return video;
}
if (track != null) {
return track;
}
return null;
}
public class MediaType {
@XStreamAsAttribute
private String title;
@XStreamAsAttribute
private String thumb;
@XStreamAsAttribute
private String art;
@XStreamAsAttribute
private String grandparentThumb;
@XStreamAsAttribute
private String grandparentTitle;
@XStreamAsAttribute
private String parentThumb;
@XStreamAsAttribute
private String parentTitle;
@XStreamAsAttribute
private long viewOffset;
@XStreamAsAttribute
private String type;
@XStreamAsAttribute
private String sessionKey;
@XStreamAlias("Media")
private Media media;
@XStreamAlias("Player")
private Player player;
@XStreamAlias("User")
private User user;
public String getGrandparentThumb() {
return this.grandparentThumb;
}
public void setGrandparentThumb(String grandparentThumb) {
this.grandparentThumb = grandparentThumb;
}
public String getGrandparentTitle() {
return this.grandparentTitle;
}
public void setGrandparentTitle(String grandparentTitle) {
this.grandparentTitle = grandparentTitle;
}
public String getParentThumb() {
return this.parentThumb;
}
public void setParentThumb(String parentThumb) {
this.parentThumb = parentThumb;
}
public String getParentTitle() {
return this.parentTitle;
}
public void setParentTitle(String parentTitle) {
this.parentTitle = parentTitle;
}
public Media getMedia() {
return this.media;
}
public Player getPlayer() {
return this.player;
}
public String getTitle() {
return title;
}
public void setTitle(String title) {
this.title = title;
}
public long getViewOffset() {
return this.viewOffset;
}
public String getType() {
return this.type;
}
public String getThumb() {
return this.thumb;
}
public void setThumb(String art) {
this.thumb = art;
}
public String getArt() {
return art;
}
public void setArt(String art) {
this.art = art;
}
public String getSessionKey() {
return sessionKey;
}
public void setSessionKey(String sessionKey) {
this.sessionKey = sessionKey;
}
public User getUser() {
return user;
}
public class Player {
@XStreamAsAttribute
private String machineIdentifier;
@XStreamAsAttribute
private String state;
@XStreamAsAttribute
private String local;
public String getMachineIdentifier() {
return this.machineIdentifier;
}
public String getState() {
return this.state;
}
public void setState(String state) {
this.state = state;
}
public String getLocal() {
return this.local;
}
}
public class Media {
@XStreamAsAttribute
private long duration;
public long getDuration() {
return this.duration;
}
}
@XStreamAlias("User")
public class User {
@XStreamAsAttribute
private Integer id;
public Integer getId() {
return this.id;
}
@XStreamAsAttribute
private String title;
public String getTitle() {
return this.title;
}
}
}
@XStreamAlias("Video")
public class Video extends MediaType {
}
@XStreamAlias("Track")
public class Track extends MediaType {
}
@XStreamAlias("Device")
public class Device {
@XStreamAsAttribute
private String name;
public String getName() {
return this.name;
}
@XStreamAlias("Connection")
@XStreamImplicit
private List<Connection> connection = null;
public List<Connection> getConnection() {
return connection;
}
public class Connection {
@XStreamAsAttribute
private String protocol;
@XStreamAsAttribute
private String address;
public String getProtocol() {
return this.protocol;
}
public String getAddress() {
return this.address;
}
}
}
}

View File

@ -0,0 +1,166 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.plex.internal.dto;
import java.util.List;
import com.google.gson.annotations.Expose;
import com.google.gson.annotations.SerializedName;
/**
* @author Brian Homeyer - Initial contribution
* @author Aron Beurskens - Binding development
*/
public class NotificationContainer {
@SerializedName("NotificationContainer")
@Expose
private InnerNotificationContainer notificationContainer;
public InnerNotificationContainer getNotificationContainer() {
return notificationContainer;
}
public void setNotificationContainer(InnerNotificationContainer notificationContainer) {
this.notificationContainer = notificationContainer;
}
public class InnerNotificationContainer {
@SerializedName("type")
@Expose
private String type;
@SerializedName("size")
@Expose
private Integer size;
@SerializedName("PlaySessionStateNotification")
@Expose
private List<PlaySessionStateNotification> playSessionStateNotification = null;
public String getType() {
return type;
}
public void setType(String type) {
this.type = type;
}
public Integer getSize() {
return size;
}
public void setSize(Integer size) {
this.size = size;
}
public List<PlaySessionStateNotification> getPlaySessionStateNotification() {
return playSessionStateNotification;
}
public void setPlaySessionStateNotification(List<PlaySessionStateNotification> playSessionStateNotification) {
this.playSessionStateNotification = playSessionStateNotification;
}
public class PlaySessionStateNotification {
@SerializedName("sessionKey")
@Expose
private String sessionKey;
@SerializedName("guid")
@Expose
private String guid;
@SerializedName("ratingKey")
@Expose
private String ratingKey;
@SerializedName("url")
@Expose
private String url;
@SerializedName("key")
@Expose
private String key;
@SerializedName("viewOffset")
@Expose
private Integer viewOffset;
@SerializedName("playQueueItemID")
@Expose
private Integer playQueueItemID;
@SerializedName("state")
@Expose
private String state;
public String getSessionKey() {
return sessionKey;
}
public void setSessionKey(String sessionKey) {
this.sessionKey = sessionKey;
}
public String getGuid() {
return guid;
}
public void setGuid(String guid) {
this.guid = guid;
}
public String getRatingKey() {
return ratingKey;
}
public void setRatingKey(String ratingKey) {
this.ratingKey = ratingKey;
}
public String getUrl() {
return url;
}
public void setUrl(String url) {
this.url = url;
}
public String getKey() {
return key;
}
public void setKey(String key) {
this.key = key;
}
public Integer getViewOffset() {
return viewOffset;
}
public void setViewOffset(Integer viewOffset) {
this.viewOffset = viewOffset;
}
public Integer getPlayQueueItemID() {
return playQueueItemID;
}
public void setPlayQueueItemID(Integer playQueueItemID) {
this.playQueueItemID = playQueueItemID;
}
public String getState() {
return state;
}
public void setState(String state) {
this.state = state;
}
}
}
}

View File

@ -0,0 +1,40 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.plex.internal.dto;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
/**
* The {@link PlexPlayerState} is the class used to map the
* player states for the player things.
*
* @author Brian Homeyer - Initial contribution
* @author Aron Beurskens - Binding development
*/
@NonNullByDefault
public enum PlexPlayerState {
STOPPED,
BUFFERING,
PLAYING,
PAUSED;
public static @Nullable PlexPlayerState of(String state) {
for (PlexPlayerState playerState : values()) {
if (playerState.toString().toLowerCase().equals(state)) {
return playerState;
}
}
return null;
}
}

View File

@ -0,0 +1,177 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.plex.internal.dto;
import java.math.BigDecimal;
import java.math.MathContext;
import java.math.RoundingMode;
import java.util.Date;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* The {@link PlexSession} is the class used internally by the PlexPlayer things
* to keep state of updates from the Bridge.
*
* @author Brian Homeyer - Initial contribution
* @author Aron Beurskens - Binding development
*/
public class PlexSession {
private String title;
private String thumb;
private String art;
private long viewOffset;
private String type;
private BigDecimal progress;
private String machineIdentifier;
private PlexPlayerState state;
private String local;
private long duration;
private Date endTime;
private String sessionKey = "";
private Integer userId;
private String userTitle = "";
private final Logger logger = LoggerFactory.getLogger(PlexSession.class);
public String getSessionKey() {
return sessionKey;
}
public void setSessionKey(String sessionKey) {
this.sessionKey = sessionKey;
}
public String getTitle() {
return title;
}
public Date getEndTime() {
return endTime;
}
public void setEndTime(Date endTime) {
this.endTime = endTime;
}
public void setTitle(String title) {
this.title = title;
}
public String getThumb() {
return thumb;
}
public void setThumb(String thumb) {
this.thumb = thumb;
}
public long getViewOffset() {
return viewOffset;
}
public void setViewOffset(long viewOffset) {
this.viewOffset = viewOffset;
updateProgress();
}
public String getType() {
return type;
}
public void setType(String type) {
this.type = type;
}
public BigDecimal getProgress() {
return progress;
}
public void setProgress(BigDecimal progress) {
this.progress = progress;
}
public String getMachineIdentifier() {
return machineIdentifier;
}
public void setMachineIdentifier(String machineIdentifier) {
this.machineIdentifier = machineIdentifier;
}
public PlexPlayerState getState() {
return state;
}
public void setState(PlexPlayerState state) {
this.state = state;
}
public String getLocal() {
return local;
}
public void setLocal(String local) {
this.local = local;
}
public long getDuration() {
return duration;
}
public void setDuration(long duration) {
this.duration = duration;
}
public String getArt() {
return art;
}
public void setArt(String art) {
this.art = art;
}
public Integer getUserId() {
return userId;
}
public void setUserId(Integer userId) {
this.userId = userId;
}
public String getUserTitle() {
return userTitle;
}
public void setUserTitle(String userTitle) {
this.userTitle = userTitle;
}
private void updateProgress() {
try {
if (this.duration > 0) {
BigDecimal progress = new BigDecimal("100")
.divide(new BigDecimal(this.duration), new MathContext(100, RoundingMode.HALF_UP))
.multiply(new BigDecimal(this.viewOffset)).setScale(2, RoundingMode.HALF_UP);
progress = BigDecimal.ZERO.max(progress);
progress = new BigDecimal("100").min(progress);
this.endTime = new Date(System.currentTimeMillis() + (this.duration - this.viewOffset));
this.progress = progress;
}
} catch (Exception e) {
logger.debug("An exception occurred while polling the updating Progress: '{}'", e.getMessage());
}
}
}

View File

@ -0,0 +1,53 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.plex.internal.dto;
import com.thoughtworks.xstream.annotations.XStreamAlias;
/**
* The {@link User} Maps the XML response from the PLEX API
*
* @author Brian Homeyer - Initial contribution
* @author Aron Beurskens - Binding development
*/
@XStreamAlias("user")
public class User {
private String username;
private String email;
@XStreamAlias("authentication-token")
private String authenticationToken;
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getEmail() {
return email;
}
public void setEmail(String email) {
this.email = email;
}
public String getAuthenticationToken() {
return authenticationToken;
}
public void setAuthenticationToken(String authenticationToken) {
this.authenticationToken = authenticationToken;
}
}

View File

@ -0,0 +1,392 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.plex.internal.handler;
import java.io.IOException;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.Base64;
import java.util.Properties;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
import org.eclipse.jetty.client.HttpClient;
import org.eclipse.jetty.http.HttpHeader;
import org.eclipse.jetty.websocket.api.Session;
import org.eclipse.jetty.websocket.api.annotations.OnWebSocketClose;
import org.eclipse.jetty.websocket.api.annotations.OnWebSocketConnect;
import org.eclipse.jetty.websocket.api.annotations.OnWebSocketError;
import org.eclipse.jetty.websocket.api.annotations.OnWebSocketMessage;
import org.eclipse.jetty.websocket.api.annotations.WebSocket;
import org.eclipse.jetty.websocket.client.ClientUpgradeRequest;
import org.eclipse.jetty.websocket.client.WebSocketClient;
import org.openhab.binding.plex.internal.config.PlexServerConfiguration;
import org.openhab.binding.plex.internal.dto.MediaContainer;
import org.openhab.binding.plex.internal.dto.MediaContainer.Device;
import org.openhab.binding.plex.internal.dto.MediaContainer.Device.Connection;
import org.openhab.binding.plex.internal.dto.NotificationContainer;
import org.openhab.binding.plex.internal.dto.User;
import org.openhab.core.i18n.ConfigurationException;
import org.openhab.core.io.net.http.HttpUtil;
import org.openhab.core.library.types.NextPreviousType;
import org.openhab.core.library.types.PlayPauseType;
import org.openhab.core.types.Command;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.google.gson.Gson;
import com.thoughtworks.xstream.XStream;
import com.thoughtworks.xstream.io.xml.StaxDriver;
/**
* The {@link PlexApiConnector} is responsible for communications with the PLEX server
*
* @author Brian Homeyer - Initial contribution
* @author Aron Beurskens - Binding development
*/
@NonNullByDefault
public class PlexApiConnector {
private static final int REQUEST_TIMEOUT_MS = 2000;
private static final String TOKEN_HEADER = "X-Plex-Token";
private static final String SIGNIN_URL = "https://plex.tv/users/sign_in.xml";
private static final String CLIENT_ID = "928dcjhd-91ka-la91-md7a-0msnan214563";
private static final String API_URL = "https://plex.tv/api/resources?includeHttps=1";
private final HttpClient httpClient;
private WebSocketClient wsClient = new WebSocketClient();
private PlexSocket plexSocket = new PlexSocket();
private final Logger logger = LoggerFactory.getLogger(PlexApiConnector.class);
private @Nullable PlexUpdateListener listener;
private final XStream xStream = new XStream(new StaxDriver());
private Gson gson = new Gson();
private boolean isShutDown = false;
private @Nullable ScheduledFuture<?> socketReconnect;
private ScheduledExecutorService scheduler;
private @Nullable URI uri;
private String username = "";
private String password = "";
private String token = "";
private String host = "";
private int port = 0;
private static String scheme = "";
public PlexApiConnector(ScheduledExecutorService scheduler, HttpClient httpClient) {
this.scheduler = scheduler;
this.httpClient = httpClient;
setupXstream();
}
public void setParameters(PlexServerConfiguration connProps) {
username = connProps.username;
password = connProps.password;
token = connProps.token;
host = connProps.host;
port = connProps.portNumber;
wsClient = new WebSocketClient();
plexSocket = new PlexSocket();
}
private String getSchemeWS() {
return "http".equals(scheme) ? "ws" : "wss";
}
public boolean hasToken() {
return !token.isBlank();
}
/**
* Base configuration for XStream
*/
private void setupXstream() {
xStream.allowTypesByWildcard(
new String[] { User.class.getPackageName() + ".**", MediaContainer.class.getPackageName() + ".**" });
xStream.setClassLoader(PlexApiConnector.class.getClassLoader());
xStream.ignoreUnknownElements();
xStream.processAnnotations(User.class);
xStream.processAnnotations(MediaContainer.class);
}
/**
* Fetch the XML data and parse it through xStream to get a MediaContainer object
*
* @return
*/
public @Nullable MediaContainer getSessionData() {
try {
String url = "http://" + host + ":" + String.valueOf(port) + "/status/sessions" + "?X-Plex-Token=" + token;
logger.debug("Getting session data '{}'", url);
MediaContainer mediaContainer = doHttpRequest("GET", url, getClientHeaders(), MediaContainer.class);
return mediaContainer;
} catch (IOException e) {
logger.debug("An exception occurred while polling the PLEX Server: '{}'", e.getMessage());
return null;
}
}
/**
* Assemble the URL to include the Token
*
* @param url The url portion that is returned from the sessions call
* @return the completed url that will be usable
*/
public String getURL(String url) {
String artURL = scheme + "://" + host + ":" + String.valueOf(port + url + "?X-Plex-Token=" + token);
return artURL;
}
/**
* This method will get an X-Token from the PLEX server if one is not provided in the bridge config
* and use this in the communication with the plex server
*/
public void getToken() {
User user;
String authString = Base64.getEncoder().encodeToString((username + ":" + password).getBytes());
Properties headers = getClientHeaders();
headers.put("Authorization", "Basic " + authString);
try {
user = doHttpRequest("POST", SIGNIN_URL, headers, User.class);
} catch (IOException e) {
logger.debug("An exception occurred while fetching PLEX user token :'{}'", e.getMessage(), e);
throw new ConfigurationException("Error occurred while fetching PLEX user token, please check config");
}
if (user.getAuthenticationToken() != null) {
token = user.getAuthenticationToken();
logger.debug("PLEX login successful using username/password");
} else {
throw new ConfigurationException("Invalid credentials for PLEX account, please check config");
}
}
/**
* This method will get the Api information from the PLEX servers.
*/
public boolean getApi() {
try {
MediaContainer api = doHttpRequest("GET", API_URL, getClientHeaders(), MediaContainer.class);
logger.debug("MediaContainer {}", api.getSize());
if (api.getDevice() != null) {
for (Device tmpDevice : api.getDevice()) {
if (tmpDevice.getConnection() != null) {
for (Connection tmpConn : tmpDevice.getConnection()) {
if (host.equals(tmpConn.getAddress())) {
scheme = tmpConn.getProtocol();
logger.debug(
"PLEX Api fetched. Found configured PLEX server in Api request, applied. Protocol used : {}",
scheme);
return true;
}
}
}
}
}
return false;
} catch (IOException e) {
logger.debug("An exception occurred while fetching API :'{}'", e.getMessage(), e);
}
return false;
}
/**
* Make an HTTP request and return the class object that was used when calling.
*
* @param <T> Class being used(dto)
* @param method GET/POST
* @param url What URL to call
* @param headers Additional headers that will be used
* @param type class type for the XML parsing
* @return Returns a class object from the data returned by the call
* @throws IOException
*/
private <T> T doHttpRequest(String method, String url, Properties headers, Class<T> type) throws IOException {
String response = HttpUtil.executeUrl(method, url, headers, null, null, REQUEST_TIMEOUT_MS);
@SuppressWarnings("unchecked")
T obj = (T) xStream.fromXML(response);
logger.debug("HTTP response {}", response);
return obj;
}
/**
* Fills in the header information for any calls to PLEX services
*
* @return Property headers
*/
private Properties getClientHeaders() {
Properties headers = new Properties();
headers.put(HttpHeader.USER_AGENT, "openHAB / PLEX binding "); // + VERSION);
headers.put("X-Plex-Client-Identifier", CLIENT_ID);
headers.put("X-Plex-Product", "openHAB");
headers.put("X-Plex-Version", "");
headers.put("X-Plex-Device", "JRE11");
headers.put("X-Plex-Device-Name", "openHAB");
headers.put("X-Plex-Provides", "controller");
headers.put("X-Plex-Platform", "Java");
headers.put("X-Plex-Platform-Version", "JRE11");
if (hasToken()) {
headers.put(TOKEN_HEADER, token);
}
return headers;
}
/**
* Register callback to PlexServerHandler
*
* @param listener function to call
*/
public void registerListener(PlexUpdateListener listener) {
this.listener = listener;
}
/**
* Dispose method, cleans up the websocket starts the reconnect logic
*/
public void dispose() {
isShutDown = true;
try {
wsClient.stop();
ScheduledFuture<?> socketReconnect = this.socketReconnect;
if (socketReconnect != null) {
socketReconnect.cancel(true);
this.socketReconnect = null;
}
httpClient.stop();
} catch (Exception e) {
logger.debug("Could not stop webSocketClient, message {}", e.getMessage());
}
}
/**
* Connect to the websocket
*/
public void connect() {
logger.debug("Connecting to WebSocket");
try {
wsClient = new WebSocketClient(httpClient);
uri = new URI(getSchemeWS() + "://" + host + ":32400/:/websockets/notifications?X-Plex-Token=" + token); // WS_ENDPOINT_TOUCHWAND);
} catch (URISyntaxException e) {
logger.debug("URI not valid {} message {}", uri, e.getMessage());
return;
}
wsClient.setConnectTimeout(2000);
ClientUpgradeRequest request = new ClientUpgradeRequest();
try {
isShutDown = false;
wsClient.start();
wsClient.connect(plexSocket, uri, request);
} catch (Exception e) {
logger.debug("Could not connect webSocket URI {} message {}", uri, e.getMessage(), e);
}
}
/**
* PlexSocket class to handle the websocket connection to the PLEX server
*/
@WebSocket(maxIdleTime = 360000) // WEBSOCKET_IDLE_TIMEOUT_MS)
public class PlexSocket {
@OnWebSocketClose
public void onClose(int statusCode, String reason) {
logger.debug("Connection closed: {} - {}", statusCode, reason);
if (!isShutDown) {
logger.debug("PLEX websocket closed - reconnecting");
asyncWeb();
}
}
@OnWebSocketConnect
public void onConnect(Session session) {
logger.debug("PLEX Socket connected to {}", session.getRemoteAddress().getAddress());
}
@OnWebSocketMessage
public void onMessage(String msg) {
NotificationContainer notification = gson.fromJson(msg, NotificationContainer.class);
if (notification != null) {
PlexUpdateListener listenerLocal = listener;
if (listenerLocal != null && notification.getNotificationContainer().getType().equals("playing")) {
listenerLocal.onItemStatusUpdate(
notification.getNotificationContainer().getPlaySessionStateNotification().get(0)
.getSessionKey(),
notification.getNotificationContainer().getPlaySessionStateNotification().get(0)
.getState());
}
}
}
@OnWebSocketError
public void onError(Throwable cause) {
if (!isShutDown) {
logger.debug("WebSocket onError - reconnecting");
asyncWeb();
}
}
private void asyncWeb() {
ScheduledFuture<?> mySocketReconnect = socketReconnect;
if (mySocketReconnect == null || mySocketReconnect.isDone()) {
socketReconnect = scheduler.schedule(PlexApiConnector.this::connect, 5, TimeUnit.SECONDS); // WEBSOCKET_RECONNECT_INTERVAL_SEC,
}
}
}
/**
* Handles control commands to the plex player.
*
* Supports:
* - Play / Pause
* - Previous / Next
*
* @param command The control command
* @param playerID The ID of the PLEX player
*/
public void controlPlayer(Command command, String playerID) {
String commandPath = null;
if (command instanceof PlayPauseType) {
if (command.equals(PlayPauseType.PLAY)) {
commandPath = "/player/playback/play";
}
if (command.equals(PlayPauseType.PAUSE)) {
commandPath = "/player/playback/pause";
}
}
if (command instanceof NextPreviousType) {
if (command.equals(NextPreviousType.PREVIOUS)) {
commandPath = "/player/playback/skipPrevious";
}
if (command.equals(NextPreviousType.NEXT)) {
commandPath = "/player/playback/skipNext";
}
}
if (commandPath != null) {
try {
String url = "http://" + host + ":" + String.valueOf(port) + commandPath;
Properties headers = getClientHeaders();
headers.put("X-Plex-Target-Client-Identifier", playerID);
HttpUtil.executeUrl("GET", url, headers, null, null, REQUEST_TIMEOUT_MS);
} catch (IOException e) {
logger.debug("An exception occurred trying to send command '{}' to the player: {}", commandPath,
e.getMessage());
}
} else {
logger.warn("Could not match command '{}' to an action", command);
}
}
}

View File

@ -0,0 +1,192 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.plex.internal.handler;
import static org.openhab.binding.plex.internal.PlexBindingConstants.*;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.openhab.binding.plex.internal.config.PlexPlayerConfiguration;
import org.openhab.binding.plex.internal.dto.MediaContainer.MediaType;
import org.openhab.binding.plex.internal.dto.PlexPlayerState;
import org.openhab.binding.plex.internal.dto.PlexSession;
import org.openhab.core.library.types.PlayPauseType;
import org.openhab.core.library.types.StringType;
import org.openhab.core.thing.Bridge;
import org.openhab.core.thing.ChannelUID;
import org.openhab.core.thing.Thing;
import org.openhab.core.thing.ThingStatus;
import org.openhab.core.thing.ThingStatusDetail;
import org.openhab.core.thing.binding.BaseThingHandler;
import org.openhab.core.types.Command;
import org.openhab.core.types.RefreshType;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* The {@link PlexBindingConstants} class defines common constants, which are
* used across the whole binding.
*
* @author Brian Homeyer - Initial contribution
* @author Aron Beurskens - Binding development
*/
@NonNullByDefault
public class PlexPlayerHandler extends BaseThingHandler {
private @NonNullByDefault({}) String playerID;
private PlexSession currentSessionData;
private boolean foundInSession;
private final Logger logger = LoggerFactory.getLogger(PlexPlayerHandler.class);
public PlexPlayerHandler(Thing thing) {
super(thing);
currentSessionData = new PlexSession();
}
/**
* Initialize the player thing, check the bridge status and hang out waiting
* for the session data to get polled.
*/
@Override
public void initialize() {
PlexPlayerConfiguration config = getConfigAs(PlexPlayerConfiguration.class);
foundInSession = false;
playerID = config.playerID;
logger.debug("Initializing PLEX player : {}", playerID);
updateStatus(ThingStatus.UNKNOWN);
}
/**
* Currently only the 'player' channel accepts commands, all others are read-only
*/
@Override
public void handleCommand(ChannelUID channelUID, Command command) {
if (command instanceof RefreshType) {
logger.debug("REFRESH not implemented");
return;
}
Bridge bridge = getBridge();
PlexServerHandler bridgeHandler = bridge == null ? null : (PlexServerHandler) bridge.getHandler();
if (bridgeHandler == null) {
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.HANDLER_MISSING_ERROR, "No bridge associated");
} else {
switch (channelUID.getId()) {
case CHANNEL_PLAYER_CONTROL:
bridgeHandler.getPlexAPIConnector().controlPlayer(command, playerID);
break;
default:
logger.debug("Channel {} not implemented/supported to control player {}", channelUID.getId(),
this.thing.getUID());
}
}
}
/**
* This is really just to set these all back to false so when we refresh the data it's
* updated for Power On/Off. This is only called from the Server Bridge.
*
* @param foundInSession Will always be false, so this can probably be changed.
*/
public void setFoundInSession(boolean foundInSession) {
this.foundInSession = foundInSession;
}
/**
* Returns the session key from the current player
*
* @return
*/
public String getSessionKey() {
return currentSessionData.getSessionKey();
}
/**
* Called when this thing gets its configuration changed.
*/
@Override
public void thingUpdated(Thing thing) {
dispose();
this.thing = thing;
initialize();
}
/**
* Refreshes all the data from the session XML call. This is called from the bridge
*
* @param sessionData The Video section of the XML(which is what pertains to the player)
*/
public void refreshSessionData(MediaType sessionData) {
currentSessionData.setState(PlexPlayerState.of(sessionData.getPlayer().getState()));
currentSessionData.setDuration(sessionData.getMedia().getDuration());
currentSessionData.setMachineIdentifier(sessionData.getPlayer().getMachineIdentifier());
currentSessionData.setViewOffset(sessionData.getViewOffset());
currentSessionData.setTitle(sessionData.getTitle());
currentSessionData.setType(sessionData.getType());
currentSessionData.setThumb(sessionData.getThumb());
currentSessionData.setArt(sessionData.getArt());
currentSessionData.setLocal(sessionData.getPlayer().getLocal());
currentSessionData.setSessionKey(sessionData.getSessionKey());
currentSessionData.setUserId(sessionData.getUser().getId());
currentSessionData.setUserTitle(sessionData.getUser().getTitle());
foundInSession = true;
updateStatus(ThingStatus.ONLINE);
}
/**
* Update just the state, this status comes from the websocket.
*
* @param state - The state to update it to.
*/
public synchronized void updateStateChannel(String state) {
currentSessionData.setState(PlexPlayerState.of(state));
updateState(new ChannelUID(getThing().getUID(), CHANNEL_PLAYER_STATE),
new StringType(String.valueOf(foundInSession ? currentSessionData.getState() : "Stopped")));
}
/**
* Updates the channel states to match reality.
*/
public synchronized void updateChannels() {
updateState(new ChannelUID(getThing().getUID(), CHANNEL_PLAYER_STATE),
new StringType(String.valueOf(foundInSession ? currentSessionData.getState() : "Stopped")));
updateState(new ChannelUID(getThing().getUID(), CHANNEL_PLAYER_POWER),
new StringType(String.valueOf(foundInSession ? "ON" : "OFF")));
updateState(new ChannelUID(getThing().getUID(), CHANNEL_PLAYER_TITLE),
new StringType(String.valueOf(foundInSession ? currentSessionData.getTitle() : "")));
updateState(new ChannelUID(getThing().getUID(), CHANNEL_PLAYER_TYPE),
new StringType(String.valueOf(foundInSession ? currentSessionData.getType() : "")));
updateState(new ChannelUID(getThing().getUID(), CHANNEL_PLAYER_ART),
new StringType(String.valueOf(foundInSession ? currentSessionData.getArt() : "")));
updateState(new ChannelUID(getThing().getUID(), CHANNEL_PLAYER_THUMB),
new StringType(String.valueOf(foundInSession ? currentSessionData.getThumb() : "")));
updateState(new ChannelUID(getThing().getUID(), CHANNEL_PLAYER_PROGRESS),
new StringType(String.valueOf(foundInSession ? currentSessionData.getProgress() : "0")));
updateState(new ChannelUID(getThing().getUID(), CHANNEL_PLAYER_ENDTIME),
new StringType(String.valueOf(foundInSession ? currentSessionData.getEndTime() : "")));
updateState(new ChannelUID(getThing().getUID(), CHANNEL_PLAYER_ENDTIME),
new StringType(String.valueOf(foundInSession ? currentSessionData.getEndTime() : "")));
updateState(new ChannelUID(getThing().getUID(), CHANNEL_PLAYER_USER),
new StringType(String.valueOf(foundInSession ? currentSessionData.getUserTitle() : "")));
// Make sure player control is in sync with the play state
if (currentSessionData.getState() == PlexPlayerState.PLAYING) {
updateState(new ChannelUID(getThing().getUID(), CHANNEL_PLAYER_CONTROL), PlayPauseType.PLAY);
}
if (currentSessionData.getState() == PlexPlayerState.PAUSED) {
updateState(new ChannelUID(getThing().getUID(), CHANNEL_PLAYER_CONTROL), PlayPauseType.PAUSE);
}
}
}

View File

@ -0,0 +1,323 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.plex.internal.handler;
import static org.openhab.binding.plex.internal.PlexBindingConstants.*;
import java.math.BigDecimal;
import java.util.ArrayList;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jdt.annotation.Nullable;
import org.eclipse.jetty.client.HttpClient;
import org.eclipse.jetty.util.ssl.SslContextFactory;
import org.openhab.binding.plex.internal.config.PlexServerConfiguration;
import org.openhab.binding.plex.internal.dto.MediaContainer;
import org.openhab.binding.plex.internal.dto.MediaContainer.MediaType;
import org.openhab.core.i18n.ConfigurationException;
import org.openhab.core.io.net.http.HttpClientFactory;
import org.openhab.core.library.types.StringType;
import org.openhab.core.thing.Bridge;
import org.openhab.core.thing.ChannelUID;
import org.openhab.core.thing.Thing;
import org.openhab.core.thing.ThingStatus;
import org.openhab.core.thing.ThingStatusDetail;
import org.openhab.core.thing.binding.BaseBridgeHandler;
import org.openhab.core.thing.binding.ThingHandler;
import org.openhab.core.thing.util.ThingWebClientUtil;
import org.openhab.core.types.Command;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* The {@link PlexServerHandler} is responsible for creating the
* Bridge Thing for a PLEX Server.
*
* @author Brian Homeyer - Initial contribution
* @author Aron Beurskens - Binding development
*/
@NonNullByDefault
public class PlexServerHandler extends BaseBridgeHandler implements PlexUpdateListener {
private final Logger logger = LoggerFactory.getLogger(PlexServerHandler.class);
private final HttpClientFactory httpClientFactory;
private @Nullable HttpClient httpClient;
// Maintain mapping of handler and players
private final Map<String, PlexPlayerHandler> playerHandlers = new ConcurrentHashMap<>();
private PlexServerConfiguration config = new PlexServerConfiguration();
private PlexApiConnector plexAPIConnector;
private @Nullable ScheduledFuture<?> pollingJob;
private volatile boolean isRunning = false;
public PlexServerHandler(Bridge bridge, HttpClientFactory httpClientFactory) {
super(bridge);
this.httpClientFactory = httpClientFactory;
plexAPIConnector = new PlexApiConnector(scheduler, httpClientFactory.getCommonHttpClient());
logger.debug("Initializing server handler");
}
public PlexApiConnector getPlexAPIConnector() {
return plexAPIConnector;
}
/**
* Initialize the Bridge set the config paramaters for the PLEX Server and
* start the refresh Job.
*/
@Override
public void initialize() {
final String httpClientName = ThingWebClientUtil.buildWebClientConsumerName(thing.getUID(), null);
try {
SslContextFactory.Server sslContextFactory = new SslContextFactory.Server();
sslContextFactory.setEndpointIdentificationAlgorithm(null);
HttpClient localHttpClient = httpClient = httpClientFactory.createHttpClient(httpClientName,
sslContextFactory);
localHttpClient.start();
plexAPIConnector = new PlexApiConnector(scheduler, localHttpClient);
} catch (Exception e) {
logger.error(
"Long running HttpClient for PlexServerHandler {} cannot be started. Creating Handler failed. Exception: {}",
httpClientName, e.getMessage(), e);
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR);
return;
}
config = getConfigAs(PlexServerConfiguration.class);
if (!config.host.isEmpty()) { // Check if a hostname is set
plexAPIConnector.setParameters(config);
} else {
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
"Host must be specified, check configuration");
return;
}
if (!plexAPIConnector.hasToken()) {
// No token is set by config, let's see if we can fetch one from username/password
logger.debug("Token is not set, trying to fetch one");
if (config.username.isEmpty() || config.password.isEmpty()) {
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
"Username, password and Token is not set, unable to connect to PLEX without. ");
return;
} else {
try {
plexAPIConnector.getToken();
} catch (ConfigurationException e) {
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR, e.getMessage());
return;
} catch (Exception e) {
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR, e.getMessage());
return;
}
}
}
logger.debug("Fetch API with config, {}", config.toString());
if (!plexAPIConnector.getApi()) {
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.CONFIGURATION_ERROR,
"Unable to fetch API, token may be wrong?");
return;
}
isRunning = true;
onUpdate(); // Start the session refresh
scheduler.execute(() -> { // Start the web socket
synchronized (this) {
if (isRunning) {
final HttpClient localHttpClient = this.httpClient;
if (localHttpClient != null) {
PlexApiConnector localSockets = plexAPIConnector = new PlexApiConnector(scheduler,
localHttpClient);
localSockets.setParameters(config);
localSockets.registerListener(this);
localSockets.connect();
}
}
}
});
}
/**
* Not currently used, all channels in this thing are read-only.
*/
@Override
public void handleCommand(ChannelUID channelUID, Command command) {
return;
}
/**
* Gets a list of all the players currently being used w/ a status of local. This
* is used for discovery only.
*
* @return
*/
public List<String> getAvailablePlayers() {
List<String> availablePlayers = new ArrayList<String>();
MediaContainer sessionData = plexAPIConnector.getSessionData();
if (sessionData != null && sessionData.getSize() > 0) {
for (MediaType tmpMeta : sessionData.getMediaTypes()) {
if (tmpMeta != null && playerHandlers.get(tmpMeta.getPlayer().getMachineIdentifier()) == null) {
if ("1".equals(tmpMeta.getPlayer().getLocal())) {
availablePlayers.add(tmpMeta.getPlayer().getMachineIdentifier());
}
}
}
}
return availablePlayers;
}
/**
* Called when a new player thing has been added. We add it to the hash map so we can
* keep track of things.
*/
@Override
public synchronized void childHandlerInitialized(ThingHandler childHandler, Thing childThing) {
String playerID = (String) childThing.getConfiguration().get(CONFIG_PLAYER_ID);
playerHandlers.put(playerID, (PlexPlayerHandler) childHandler);
logger.debug("Bridge: Monitor handler was initialized for {} with id {}", childThing.getUID(), playerID);
}
/**
* Called when a player has been removed from the system.
*/
@Override
public synchronized void childHandlerDisposed(ThingHandler childHandler, Thing childThing) {
String playerID = (String) childThing.getConfiguration().get(CONFIG_PLAYER_ID);
playerHandlers.remove(playerID);
logger.debug("Bridge: Monitor handler was disposed for {} with id {}", childThing.getUID(), playerID);
}
/**
* Basically a callback method for the websocket handling
*/
@Override
public void onItemStatusUpdate(String sessionKey, String state) {
try {
for (Map.Entry<String, PlexPlayerHandler> entry : playerHandlers.entrySet()) {
if (entry.getValue().getSessionKey().equals(sessionKey)) {
entry.getValue().updateStateChannel(state);
}
}
} catch (Exception e) {
logger.debug("Failed setting item status : {}", e.getMessage());
}
}
/**
* Clears the foundInSession field for the configured players, then it sets the
* data for the machineIds that are found in the session data set. This allows
* us to determine if a device is on or off.
*
* @param sessionData The MediaContainer object that is pulled from the XML result of
* a call to the session data on PLEX.
*/
@SuppressWarnings("null")
private void refreshStates(MediaContainer sessionData) {
int playerCount = 0;
int playerActiveCount = 0;
Iterator<PlexPlayerHandler> valueIterator = playerHandlers.values().iterator();
while (valueIterator.hasNext()) {
playerCount++;
valueIterator.next().setFoundInSession(false);
}
if (sessionData != null && sessionData.getSize() > 0) { // Cover condition where nothing is playing
for (MediaContainer.MediaType tmpMeta : sessionData.getMediaTypes()) { // Roll through mediaType objects
// looking for machineID
if (playerHandlers.get(tmpMeta.getPlayer().getMachineIdentifier()) != null) { // if we have a player
// configured, update
// it
tmpMeta.setArt(plexAPIConnector.getURL(tmpMeta.getArt()));
if (tmpMeta.getType().equals("episode")) {
tmpMeta.setThumb(plexAPIConnector.getURL(tmpMeta.getGrandparentThumb()));
tmpMeta.setTitle(tmpMeta.getGrandparentTitle() + " : " + tmpMeta.getTitle());
} else if (tmpMeta.getType().equals("track")) {
tmpMeta.setThumb(plexAPIConnector.getURL(tmpMeta.getThumb()));
tmpMeta.setTitle(tmpMeta.getGrandparentTitle() + " - " + tmpMeta.getParentTitle() + " - "
+ tmpMeta.getTitle());
} else {
tmpMeta.setThumb(plexAPIConnector.getURL(tmpMeta.getThumb()));
}
playerHandlers.get(tmpMeta.getPlayer().getMachineIdentifier()).refreshSessionData(tmpMeta);
playerActiveCount++;
}
}
}
updateState(new ChannelUID(getThing().getUID(), CHANNEL_SERVER_COUNT),
new StringType(String.valueOf(playerCount)));
updateState(new ChannelUID(getThing().getUID(), CHANNEL_SERVER_COUNTACTIVE),
new StringType(String.valueOf(playerActiveCount)));
}
/**
* Refresh all the configured players
*/
private void refreshAllPlayers() {
Iterator<PlexPlayerHandler> valueIterator = playerHandlers.values().iterator();
while (valueIterator.hasNext()) {
valueIterator.next().updateChannels();
}
}
/**
* This is called to start the refresh job and also to reset that refresh job when a config change is done.
*/
private synchronized void onUpdate() {
ScheduledFuture<?> pollingJob = this.pollingJob;
if (pollingJob == null || pollingJob.isCancelled()) {
int pollingInterval = ((BigDecimal) getConfig().get(CONFIG_REFRESH_RATE)).intValue();
this.pollingJob = scheduler.scheduleWithFixedDelay(pollingRunnable, 1, pollingInterval, TimeUnit.SECONDS);
}
}
/**
* The refresh job, pulls the session data and then calls refreshAllPlayers which will have them send
* out their current status.
*/
private Runnable pollingRunnable = () -> {
try {
MediaContainer plexSessionData = plexAPIConnector.getSessionData();
if (plexSessionData != null) {
refreshStates(plexSessionData);
updateStatus(ThingStatus.ONLINE);
} else {
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR,
"PLEX is not returning valid session data");
}
refreshAllPlayers();
} catch (Exception e) {
updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.COMMUNICATION_ERROR, String
.format("An exception occurred while polling the PLEX Server: '%s'", e.getMessage()).toString());
}
};
@Override
public void dispose() {
logger.debug("Disposing PLEX Bridge Handler.");
isRunning = false;
plexAPIConnector.dispose();
ScheduledFuture<?> pollingJob = this.pollingJob;
if (pollingJob != null && !pollingJob.isCancelled()) {
pollingJob.cancel(true);
this.pollingJob = null;
}
}
}

View File

@ -0,0 +1,24 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.plex.internal.handler;
import org.eclipse.jdt.annotation.NonNullByDefault;
/**
* @author Brian Homeyer - Initial contribution
* @author Aron Beurskens - Binding development
*/
@NonNullByDefault
public interface PlexUpdateListener {
void onItemStatusUpdate(String sessionKey, String state);
}

View File

@ -0,0 +1,11 @@
<?xml version="1.0" encoding="UTF-8"?>
<addon:addon id="plex" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:addon="https://openhab.org/schemas/addon/v1.0.0"
xsi:schemaLocation="https://openhab.org/schemas/addon/v1.0.0 https://openhab.org/schemas/addon-1.0.0.xsd">
<type>binding</type>
<name>Plex Binding</name>
<description>Monitor your PLEX server and players</description>
<connection>hybrid</connection>
</addon:addon>

View File

@ -0,0 +1,52 @@
<?xml version="1.0" encoding="UTF-8"?>
<config-description:config-descriptions
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:config-description="https://openhab.org/schemas/config-description/v1.0.0"
xsi:schemaLocation="https://openhab.org/schemas/config-description/v1.0.0 https://openhab.org/schemas/config-description-1.0.0.xsd">
<config-description uri="thing-type:plex:server">
<parameter-group name="config-info">
<label>PLEX Bridge Configuration</label>
</parameter-group>
<parameter-group name="auth-info">
<label>Authentication Information</label>
</parameter-group>
<parameter name="host" type="text" required="true" groupName="config-info">
<label>Server</label>
<description>PLEX host name or IP address</description>
<context>network-address</context>
</parameter>
<parameter name="portNumber" type="integer" min="1" max="65535" required="false" groupName="config-info">
<label>Port Number</label>
<description>Port Number (leave blank if PLEX installed on default port)</description>
<default>32400</default>
</parameter>
<parameter name="refreshRate" type="integer" min="2" unit="s" required="true" groupName="config-info">
<label>Refresh Interval</label>
<description>Interval in seconds at which PLEX server status is polled</description>
<default>5</default>
</parameter>
<parameter name="token" type="text" required="false" groupName="auth-info">
<label>X-Token</label>
<description>X-Token</description>
</parameter>
<parameter name="username" type="text" groupName="auth-info">
<label>Username</label>
<description>If you're using PLEX Home you need to supply the username and password of your PLEX account here. If you
don't want to enter your credentials you can also directly set your account token below instead.</description>
</parameter>
<parameter name="password" type="text" groupName="auth-info">
<context>password</context>
<label>Password</label>
<description>Password</description>
</parameter>
</config-description>
<config-description uri="thing-type:plex:player">
<parameter name="playerID" type="text" required="true">
<label>Player ID</label>
<description>The unique identifier of the player</description>
</parameter>
</config-description>
</config-description:config-descriptions>

View File

@ -0,0 +1,57 @@
# add-on
addon.plex.name = Plex Binding
addon.plex.description = Monitor your PLEX server and players
# thing types
thing-type.plex.player.label = PLEX Player
thing-type.plex.player.description = Represents a PLEX player
thing-type.plex.server.label = PLEX Server
thing-type.plex.server.description = Represents a PLEX Server
# thing types config
thing-type.config.plex.player.playerID.label = Player ID
thing-type.config.plex.player.playerID.description = The unique identifier of the player
thing-type.config.plex.server.group.auth-info.label = Authentication Information
thing-type.config.plex.server.group.config-info.label = PLEX Bridge Configuration
thing-type.config.plex.server.host.label = Server
thing-type.config.plex.server.host.description = PLEX host name or IP address
thing-type.config.plex.server.password.label = Password
thing-type.config.plex.server.password.description = Password
thing-type.config.plex.server.portNumber.label = Port Number
thing-type.config.plex.server.portNumber.description = Port Number (leave blank if PLEX installed on default port)
thing-type.config.plex.server.refreshRate.label = Refresh Interval
thing-type.config.plex.server.refreshRate.description = Interval in seconds at which PLEX server status is polled
thing-type.config.plex.server.token.label = X-Token
thing-type.config.plex.server.token.description = X-Token
thing-type.config.plex.server.username.label = Username
thing-type.config.plex.server.username.description = If you're using PLEX Home you need to supply the username and password of your PLEX account here. If you don't want to enter your credentials you can also directly set your account token below instead.
# channel types
channel-type.plex.art.label = Background Art
channel-type.plex.art.description = The URL of the background art for currently playing media
channel-type.plex.currentPlayers.label = Current Players
channel-type.plex.currentPlayers.description = The number of players currently configured to watch on PLEX
channel-type.plex.currentPlayersActive.label = Current Players Active
channel-type.plex.currentPlayersActive.description = The number of players currently being used on PLEX
channel-type.plex.endtime.label = End Time
channel-type.plex.endtime.description = Time at which the media that is playing will end
channel-type.plex.player.label = Player Control
channel-type.plex.player.description = The control channel for the player `PLAY/PAUSE/NEXT/PREVIOUS`
channel-type.plex.power.label = Player Power State
channel-type.plex.power.description = The power status of the player
channel-type.plex.progress.label = Media Progress
channel-type.plex.progress.description = The current progress of playing media
channel-type.plex.state.label = Player State
channel-type.plex.state.description = The current state of the Player
channel-type.plex.thumb.label = Cover Art
channel-type.plex.thumb.description = The URL of the cover art for currently playing media
channel-type.plex.title.label = Player Title
channel-type.plex.title.description = The title of media that is playing
channel-type.plex.type.label = Media Type
channel-type.plex.type.description = The current type of playing media
channel-type.plex.user.label = Player User
channel-type.plex.user.description = The title of the user controlling the player

View File

@ -0,0 +1,108 @@
<?xml version="1.0" encoding="UTF-8"?>
<thing:thing-descriptions bindingId="plex"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:thing="https://openhab.org/schemas/thing-description/v1.0.0"
xsi:schemaLocation="https://openhab.org/schemas/thing-description/v1.0.0 https://openhab.org/schemas/thing-description-1.0.0.xsd">
<bridge-type id="server">
<label>PLEX Server</label>
<description>Represents a PLEX Server</description>
<channels>
<channel id="currentPlayer" typeId="currentPlayers"/>
<channel id="currentPlayersActive" typeId="currentPlayersActive"/>
</channels>
<config-description-ref uri="thing-type:plex:server"/>
</bridge-type>
<thing-type id="player">
<supported-bridge-type-refs>
<bridge-type-ref id="server"/>
</supported-bridge-type-refs>
<label>PLEX Player</label>
<description>Represents a PLEX player</description>
<channels>
<channel id="state" typeId="state"/>
<channel id="power" typeId="power"/>
<channel id="title" typeId="title"/>
<channel id="type" typeId="type"/>
<channel id="endtime" typeId="endtime"/>
<channel id="progress" typeId="progress"/>
<channel id="art" typeId="art"/>
<channel id="thumb" typeId="thumb"/>
<channel id="player" typeId="player"/>
<channel id="user" typeId="user"/>
</channels>
<config-description-ref uri="thing-type:plex:player"/>
</thing-type>
<channel-type id="currentPlayers">
<item-type>Number</item-type>
<label>Current Players</label>
<description>The number of players currently configured to watch on PLEX</description>
<state readOnly="true"/>
</channel-type>
<channel-type id="currentPlayersActive">
<item-type>Number</item-type>
<label>Current Players Active</label>
<description>The number of players currently being used on PLEX</description>
<state readOnly="true"/>
</channel-type>
<channel-type id="state">
<item-type>String</item-type>
<label>Player State</label>
<description>The current state of the Player</description>
<state readOnly="true"/>
</channel-type>
<channel-type id="power">
<item-type>Switch</item-type>
<label>Player Power State</label>
<description>The power status of the player</description>
<state readOnly="true"/>
</channel-type>
<channel-type id="title">
<item-type>String</item-type>
<label>Player Title</label>
<description>The title of media that is playing</description>
<state readOnly="true"/>
</channel-type>
<channel-type id="type">
<item-type>String</item-type>
<label>Media Type</label>
<description>The current type of playing media</description>
<state readOnly="true"/>
</channel-type>
<channel-type id="endtime">
<item-type>DateTime</item-type>
<label>End Time</label>
<description>Time at which the media that is playing will end</description>
<state readOnly="true"/>
</channel-type>
<channel-type id="progress">
<item-type>Dimmer</item-type>
<label>Media Progress</label>
<description>The current progress of playing media</description>
</channel-type>
<channel-type id="art">
<item-type>String</item-type>
<label>Background Art</label>
<description>The URL of the background art for currently playing media</description>
<state readOnly="true"/>
</channel-type>
<channel-type id="thumb">
<item-type>String</item-type>
<label>Cover Art</label>
<description>The URL of the cover art for currently playing media</description>
<state readOnly="true"/>
</channel-type>
<channel-type id="player">
<item-type>Player</item-type>
<label>Player Control</label>
<description>The control channel for the player `PLAY/PAUSE/NEXT/PREVIOUS`</description>
</channel-type>
<channel-type id="user">
<item-type>String</item-type>
<label>Player User</label>
<description>The title of the user controlling the player</description>
<state readOnly="true"/>
</channel-type>
</thing:thing-descriptions>

View File

@ -0,0 +1,64 @@
/**
* Copyright (c) 2010-2023 Contributors to the openHAB project
*
* See the NOTICE file(s) distributed with this work for additional
* information.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License 2.0 which is available at
* http://www.eclipse.org/legal/epl-2.0
*
* SPDX-License-Identifier: EPL-2.0
*/
package org.openhab.binding.plex.internal.handler;
import static org.hamcrest.CoreMatchers.*;
import static org.hamcrest.MatcherAssert.assertThat;
import java.util.concurrent.ScheduledExecutorService;
import java.util.stream.Stream;
import org.eclipse.jdt.annotation.NonNullByDefault;
import org.eclipse.jetty.client.HttpClient;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.params.ParameterizedTest;
import org.junit.jupiter.params.provider.Arguments;
import org.junit.jupiter.params.provider.MethodSource;
import org.mockito.Mock;
import org.openhab.binding.plex.internal.config.PlexServerConfiguration;
/**
* Tests cases for {@link org.openhab.binding.plex.internal.handler.PlexApiConnector}.
*
* @author Brian Homeyer - Initial contribution
* @author Aron Beurskens - Added test
*/
@NonNullByDefault
public class PlexApiConnectorTest {
private @NonNullByDefault({}) @Mock ScheduledExecutorService scheduler;
private @NonNullByDefault({}) @Mock HttpClient httpClient;
private @NonNullByDefault({}) PlexApiConnector plexApiConnector;
@BeforeEach
public void setUp() {
plexApiConnector = new PlexApiConnector(scheduler, httpClient);
}
/**
* Test that the .hasToken check return the correct values.
*/
@ParameterizedTest(name = "{index} => token={0}, result={1}")
@MethodSource("tokenProvider")
public void testHasToken(String token, Boolean result) {
PlexServerConfiguration config = new PlexServerConfiguration();
config.token = token;
plexApiConnector.setParameters(config);
assertThat(plexApiConnector.hasToken(), is(result));
}
private static Stream<Arguments> tokenProvider() {
return Stream.of(Arguments.of("123", true), Arguments.of(" ", false),
Arguments.of("fdsjkghdf-dsjfhs-dsafkshj", true), Arguments.of("", false));
}
}

View File

@ -300,6 +300,7 @@
<module>org.openhab.binding.pjlinkdevice</module> <module>org.openhab.binding.pjlinkdevice</module>
<module>org.openhab.binding.playstation</module> <module>org.openhab.binding.playstation</module>
<module>org.openhab.binding.plclogo</module> <module>org.openhab.binding.plclogo</module>
<module>org.openhab.binding.plex</module>
<module>org.openhab.binding.plugwise</module> <module>org.openhab.binding.plugwise</module>
<module>org.openhab.binding.plugwiseha</module> <module>org.openhab.binding.plugwiseha</module>
<module>org.openhab.binding.powermax</module> <module>org.openhab.binding.powermax</module>