diff --git a/CODEOWNERS b/CODEOWNERS index 486a1c2e5d6..28ec2b43f63 100755 --- a/CODEOWNERS +++ b/CODEOWNERS @@ -235,6 +235,7 @@ /bundles/org.openhab.binding.mpd/ @stefanroellin /bundles/org.openhab.binding.mqtt/ @ccutrer /bundles/org.openhab.binding.mqtt.espmilighthub/ @Skinah +/bundles/org.openhab.binding.mqtt.fpp/ @computergeek1507 /bundles/org.openhab.binding.mqtt.generic/ @ccutrer /bundles/org.openhab.binding.mqtt.homeassistant/ @antroids @ccutrer /bundles/org.openhab.binding.mqtt.homie/ @ccutrer diff --git a/bundles/org.openhab.binding.mqtt.fpp/NOTICE b/bundles/org.openhab.binding.mqtt.fpp/NOTICE new file mode 100644 index 00000000000..38d625e3492 --- /dev/null +++ b/bundles/org.openhab.binding.mqtt.fpp/NOTICE @@ -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 diff --git a/bundles/org.openhab.binding.mqtt.fpp/README.md b/bundles/org.openhab.binding.mqtt.fpp/README.md new file mode 100644 index 00000000000..19d46cac76a --- /dev/null +++ b/bundles/org.openhab.binding.mqtt.fpp/README.md @@ -0,0 +1,92 @@ +# FPP Binding + +Binding to control Falcon Player (FPP) Devices using MQTT and HTTP. Status messages are recieved over MQTT and Commands are HTTP Commands. + +## Discovery + +Autodiscovering is not supported. We have to define the things manually. + +## Supported Things + +The binding supports one Thing `player` that represents the Falcon Player. +## Thing Configuration + +| Parameter | Description | Required | Default | +|--------------|-----------------------------------------|----------|---------| +| `playerIP` | IP Address or Host Name of FPP Devive | Y | | +| `playerMQTT` | MQTT Topic of FPP Devive Status Updates | Y | | + +## Channels + +| Channel | Type | Description | +|----------------------------------------|--------------------|-------------------------------------------| +| `player` | Player | Play/Stop Current Playlist. | +| `volume` | Dimmer | Playback Audio Volume. | +| `status` | String | Playback Status. | +| `mode` | String | Playback Mode. | +| `uptime` | Number:Time | Device Uptime. | +| `testing-enabled` | Switch | Enabled/Disable Sending Testing Data. | +| `current-sequence` | String (read only) | Currently Playing Sequence File. | +| `current-song` | String (read only) | Currently Playing Audio/Media File. | +| `current-playlist` | String (read only) | Currently Playing Playlist. | +| `seconds-played` | Number:Time | Sequence Playback time in secs. | +| `seconds-remaining` | Number:Time | Sequence Playback time remaining in secs. | +| `last-playlist` | String | Lasted Played Playlist. | +| `bridging-enabled` | Switch | Is Recieving Bridge Data. | +| `multisync-enabled` | Switch | Multisync Mode Enabled. | +| `scheduler-current-playlist` | String (read only) | Scheduler Current Playlist. | +| `scheduler-current-playlist-start` | String (read only) | Scheduler Current Playlist Start Time. | +| `scheduler-current-playlist-end` | String (read only) | Scheduler Current Playlist End Time. | +| `scheduler-current-playlist-stop-type` | String (read only) | Scheduler Current Playlist End Type. | +| `scheduler-next-playlist` | String (read only) | Next Scheduled Playlist. | +| `scheduler-next-playlist-start` | String (read only) | Next Scheduled Start Time. | + + +## Full Example + +To use these examples for textual configuration, you must already have a configured MQTT `broker` thing, and know its unique ID. +This UID will be used in the things file and will replace the text `myBroker`. +The first line in the things file will create a `broker` thing and this can be removed if you have already setup a broker in another file or via the UI already. + +### fpp.things + +```java +Bridge mqtt:broker:myBroker [ host="localhost", secure=false, password="*******", qos=1, username="user"] +Thing mqtt:player:myBroker:mainPlayer "Main Player" (mqtt:broker:myBroker) @ "MQTT" +``` + +### fpp.items + +```java +Player FPP_Player "FPP Player" {channel="mqtt:player:myBroker:mainPlayer:player"} +Dimmer Audio_Volume "Audio Volume" {channel="mqtt:player:myBroker:mainPlayer:volume"} +String Current_Sequence "Current Sequence" {channel="mqtt:player:myBroker:mainPlayer:current-sequence"} +String Current_Song "Current Song" {channel="mqtt:player:myBroker:mainPlayer:current-song"} +String Current_Playlist "Current Playlist" {channel="mqtt:player:myBroker:mainPlayer:current-playlist"} +String Status "FPP Status" {channel="mqtt:player:myBroker:mainPlayer:status"} +String Mode "FPP Mode" {channel="mqtt:player:myBroker:mainPlayer:mode"} +String Last_Playlist "Last Playlist" {channel="mqtt:player:myBroker:mainPlayer:last-playlist"} +Number:Time Seconds_Played "Seconds Played [%d %unit%]" {channel="mqtt:player:myBroker:mainPlayer:seconds-played"} +Number:Time Seconds_Remaining "Seconds Remaining [%d %unit%]" {channel="mqtt:player:myBroker:mainPlayer:seconds-remaining"} +Switch Testing "Testing Mode" {channel="mqtt:player:myBroker:mainPlayer:testing-enabled"} +Switch Multisync "Multisync" {channel="mqtt:player:myBroker:mainPlayer:multisync-enabled"} +``` + +### fpp.sitemap + +```perl +Text label="Main Player" +{ + Player item=FPP_Player + Switch item=Testing + Slider item=Audio_Volume + Text item=Current_Sequence + Text item=Current_Song + Text item=Current_Playlist + Text item=Status + Text item=Mode + Selection item=Last_Playlist + Switch item=Testing + Switch item=Multisync +} +``` diff --git a/bundles/org.openhab.binding.mqtt.fpp/pom.xml b/bundles/org.openhab.binding.mqtt.fpp/pom.xml new file mode 100644 index 00000000000..b6c67c8aade --- /dev/null +++ b/bundles/org.openhab.binding.mqtt.fpp/pom.xml @@ -0,0 +1,24 @@ + + + + 4.0.0 + + + org.openhab.addons.bundles + org.openhab.addons.reactor.bundles + 4.3.0-SNAPSHOT + + + org.openhab.binding.mqtt.fpp + openHAB Add-ons :: Bundles :: MQTT FPP + + + + org.openhab.addons.bundles + org.openhab.binding.mqtt + ${project.version} + provided + + + diff --git a/bundles/org.openhab.binding.mqtt.fpp/src/main/feature/feature.xml b/bundles/org.openhab.binding.mqtt.fpp/src/main/feature/feature.xml new file mode 100644 index 00000000000..ca8ff9fdadf --- /dev/null +++ b/bundles/org.openhab.binding.mqtt.fpp/src/main/feature/feature.xml @@ -0,0 +1,12 @@ + + + mvn:org.openhab.core.features.karaf/org.openhab.core.features.karaf.openhab-core/${ohc.version}/xml/features + + + openhab-runtime-base + openhab-transport-mqtt + mvn:org.openhab.addons.bundles/org.openhab.binding.mqtt/${project.version} + mvn:org.openhab.addons.bundles/org.openhab.binding.mqtt.fpp/${project.version} + + + diff --git a/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/ConfigOptions.java b/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/ConfigOptions.java new file mode 100644 index 00000000000..9ffcada0992 --- /dev/null +++ b/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/ConfigOptions.java @@ -0,0 +1,26 @@ +/** + * Copyright (c) 2010-2024 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.mqtt.fpp.internal; + +import org.eclipse.jdt.annotation.NonNullByDefault; + +/** + * The {@link ConfigOptions} Holds the config for the settings. + * + * @author Scott Hanson - Initial contribution + */ +@NonNullByDefault +public class ConfigOptions { + public String playerAddress = ""; + public String playerMQTTTopic = ""; +} diff --git a/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/FPPBindingConstants.java b/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/FPPBindingConstants.java new file mode 100644 index 00000000000..9c27ea30ec2 --- /dev/null +++ b/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/FPPBindingConstants.java @@ -0,0 +1,73 @@ +/** + * Copyright (c) 2010-2024 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.mqtt.fpp.internal; + +import static org.openhab.binding.mqtt.MqttBindingConstants.BINDING_ID; + +import java.util.Set; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.openhab.core.thing.ThingTypeUID; + +/** + * The {@link FPPBindingConstants} class defines common constants, which are + * used across the whole binding. + * + * @author Scott Hanson - Initial contribution + */ +@NonNullByDefault +public class FPPBindingConstants { + // falcon/player/FPP/fppd_status + public static final String STATUS_TOPIC = "fppd_status"; + public static final String VERSION_TOPIC = "version"; + public static final String PLAYLIST_TOPIC = "playlist"; + public static final String MQTT_PREFIX = "falcon/player/"; + + // List of all Thing Type UIDs + public static final ThingTypeUID THING_TYPE_PLAYER = new ThingTypeUID(BINDING_ID, "player"); + + public static final Set SUPPORTED_THING_TYPES = Set.of(THING_TYPE_PLAYER); + + // Channels + public static final String CHANNEL_PLAYER = "player"; + public static final String CHANNEL_STATUS = "status"; + public static final String CHANNEL_VOLUME = "volume"; + public static final String CHANNEL_MODE = "mode"; + public static final String CHANNEL_CURRENT_SEQUENCE = "current-sequence"; + public static final String CHANNEL_CURRENT_SONG = "current-song"; + public static final String CHANNEL_CURRENT_PLAYLIST = "current-playlist"; + public static final String CHANNEL_SEC_PLAYED = "seconds-played"; + public static final String CHANNEL_SEC_REMAINING = "seconds-remaining"; + public static final String CHANNEL_UPTIME = "uptime"; + public static final String CHANNEL_BRIDGING = "bridging-enabled"; + public static final String CHANNEL_MULTISYNC = "multisync-enabled"; + public static final String CHANNEL_TESTING = "testing-enabled"; + public static final String CHANNEL_LAST_PLAYLIST = "last-playlist"; + + public static final String CHANNEL_SCHEDULER_STATUS = "scheduler-status"; + public static final String CHANNEL_SCHEDULER_CURRENT_PLAYLIST = "scheduler-current-playlist"; + public static final String CHANNEL_SCHEDULER_CURRENT_PLAYLIST_START = "scheduler-current-playlist-start"; + public static final String CHANNEL_SCHEDULER_CURRENT_PLAYLIST_END = "scheduler-current-playlist-end"; + public static final String CHANNEL_SCHEDULER_CURRENT_PLAYLIST_STOP_TYPE = "scheduler-current-playlist-stop-type"; + public static final String CHANNEL_SCHEDULER_NEXT_PLAYLIST = "scheduler-next-playlist"; + public static final String CHANNEL_SCHEDULER_NEXT_PLAYLIST_START = "scheduler-next-playlist-start"; + + public static final String PROPERTY_UUID = "uuid"; + public static final String PROPERTY_SOFTWARE_VERSION = "Software Version"; + + // Status + public static final String CONNECTED = "connected"; + public static final String CHANNEL_STATUS_NAME = "status-name"; + + public static final String TESTING = "testing"; +} diff --git a/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/FPPHandlerFactory.java b/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/FPPHandlerFactory.java new file mode 100644 index 00000000000..96d132ff765 --- /dev/null +++ b/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/FPPHandlerFactory.java @@ -0,0 +1,59 @@ +/** + * Copyright (c) 2010-2024 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.mqtt.fpp.internal; + +import static org.openhab.binding.mqtt.fpp.internal.FPPBindingConstants.SUPPORTED_THING_TYPES; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.eclipse.jdt.annotation.Nullable; +import org.openhab.binding.mqtt.fpp.internal.handler.FPPPlayerHandler; +import org.openhab.core.thing.Thing; +import org.openhab.core.thing.ThingRegistry; +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.service.component.annotations.Activate; +import org.osgi.service.component.annotations.Component; +import org.osgi.service.component.annotations.Reference; + +/** + * The {@link FPPHandlerFactory} is responsible for creating things and thing + * handlers. + * + * @author Scott Hanson - Initial contribution + */ +@Component(service = ThingHandlerFactory.class) +@NonNullByDefault +public class FPPHandlerFactory extends BaseThingHandlerFactory { + private final ThingRegistry thingRegistry; + + @Activate + public FPPHandlerFactory(final @Reference ThingRegistry thingRegistry) { + this.thingRegistry = thingRegistry; + } + + @Override + public boolean supportsThingType(ThingTypeUID thingTypeUID) { + return SUPPORTED_THING_TYPES.contains(thingTypeUID); + } + + @Override + protected @Nullable ThingHandler createHandler(Thing thing) { + ThingTypeUID thingTypeUID = thing.getThingTypeUID(); + if (SUPPORTED_THING_TYPES.contains(thingTypeUID)) { + return new FPPPlayerHandler(thing, thingRegistry); + } + return null; + } +} diff --git a/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/FPPStatus.java b/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/FPPStatus.java new file mode 100644 index 00000000000..25adf6dd805 --- /dev/null +++ b/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/FPPStatus.java @@ -0,0 +1,156 @@ +/** + * Copyright (c) 2010-2024 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.mqtt.fpp.internal; + +import com.google.gson.annotations.Expose; +import com.google.gson.annotations.SerializedName; + +/** + * The {@link FPPStstus} is responsible for storing + * the FPP JSON Status data. + * + * @author Scott Hanson - Initial contribution + */ +public class FPPStatus { + /* + * { + * "MQTT" : + * { + * "configured" : true, + * "connected" : true + * }, + * "bridging" : false, + * "current_playlist" : + * { + * "count" : "0", + * "description" : "", + * "index" : "0", + * "playlist" : "", + * "type" : "" + * }, + * "current_sequence" : "", + * "current_song" : "", + * "dateStr" : "Sun Jan 7", + * "fppd" : "running", + * "mode" : 2, + * "mode_name" : "player", + * "multisync" : true, + * "next_playlist" : + * { + * "playlist" : "No playlist scheduled.", + * "start_time" : "" + * }, + * "repeat_mode" : "0", + * "scheduler" : + * { + * "enabled" : 1, + * "nextPlaylist" : + * { + * "playlistName" : "No playlist scheduled.", + * "scheduledStartTime" : 0, + * "scheduledStartTimeStr" : "" + * }, + * "status" : "idle" + * }, + * "seconds_played" : "0", + * "seconds_remaining" : "0", + * "sensors" : + * [ + * { + * "formatted" : "55.0", + * "label" : "CPU: ", + * "postfix" : "", + * "prefix" : "", + * "value" : 55.017000000000003, + * "valueType" : "Temperature" + * } + * ], + * "status" : 0, + * "status_name" : "idle", + * "time" : "Sun Jan 07 19:15:25 EST 2024", + * "timeStr" : "07:15 PM", + * "timeStrFull" : "07:15:25 PM", + * "time_elapsed" : "00:00", + * "time_remaining" : "00:00", + * "uptime" : "2 days, 01:30:04", + * "uptimeDays" : 2.0625462962962962, + * "uptimeHours" : 1.5011111111111111, + * "uptimeMinutes" : 30.066666666666666, + * "uptimeSeconds" : 4, + * "uptimeStr" : "2 days, 1 hours, 30 minutes, 4 seconds", + * "uptimeTotalSeconds" : 178204, + * "uuid" : "M1-10000000fd93cfe5", + * "volume" : 48 + * } + * + */ + @SerializedName("status") + @Expose + public int status; + + @SerializedName("status_name") + @Expose + public String status_name; + + @SerializedName("mode_name") + @Expose + public String mode_name; + + @SerializedName("current_sequence") + @Expose + public String current_sequence; + + @SerializedName("current_song") + @Expose + public String current_song; + + @SerializedName("time_elapsed") + @Expose + public String time_elapsed; + + @SerializedName("uptimeTotalSeconds") + @Expose + public long uptimeTotalSeconds; + + @SerializedName("seconds_played") + @Expose + public int seconds_played; + + @SerializedName("seconds_remaining") + @Expose + public int seconds_remaining; + + @SerializedName("volume") + @Expose + public int volume; + + @SerializedName("uuid") + @Expose + public String uuid; + + @SerializedName("multisync") + @Expose + public boolean multisync; + + @SerializedName("current_playlist") + @Expose + public FPPPlaylist current_playlist; + + @SerializedName("bridging") + @Expose + public boolean bridging; + + @SerializedName("scheduler") + @Expose + public FPPScheduler scheduler; +} diff --git a/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/dto/FPPCurrentPlaylist.java b/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/dto/FPPCurrentPlaylist.java new file mode 100644 index 00000000000..3b4a4a3d918 --- /dev/null +++ b/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/dto/FPPCurrentPlaylist.java @@ -0,0 +1,40 @@ +/** + * Copyright (c) 2010-2024 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.mqtt.fpp.internal; + +import com.google.gson.annotations.Expose; +import com.google.gson.annotations.SerializedName; + +/** + * The {@link FPPCurrentPlaylist} is responsible for storing + * the FPP JSON Current Playlist data. + * + * @author Scott Hanson - Initial contribution + */ +public class FPPCurrentPlaylist { + @SerializedName("playlistName") + @Expose + public String playlistName; + + @SerializedName("scheduledStartTimeStr") + @Expose + public String scheduledStartTimeStr; + + @SerializedName("scheduledEndTimeStr") + @Expose + public String scheduledEndTimeStr; + + @SerializedName("stopTypeStr") + @Expose + public String stopTypeStr; +} diff --git a/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/dto/FPPNextPlaylist.java b/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/dto/FPPNextPlaylist.java new file mode 100644 index 00000000000..34c8f8c123b --- /dev/null +++ b/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/dto/FPPNextPlaylist.java @@ -0,0 +1,45 @@ +/** + * Copyright (c) 2010-2024 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.mqtt.fpp.internal; + +import com.google.gson.annotations.Expose; +import com.google.gson.annotations.SerializedName; + +/** + * The {@link FPPNextPlaylist} is responsible for storing + * the FPP JSON Next Playlist data. + * + * @author Scott Hanson - Initial contribution + */ +public class FPPNextPlaylist { + /* + * "nextPlaylist" : + * { + * "playlistName" : "No playlist scheduled.", + * "scheduledStartTime" : 0, + * "scheduledStartTimeStr" : "" + * }, + */ + + @SerializedName("playlistName") + @Expose + public String playlistName; + + @SerializedName("scheduledStartTimeStr") + @Expose + public String scheduledStartTimeStr; + + @SerializedName("scheduledStartTime") + @Expose + public int scheduledStartTime; +} diff --git a/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/dto/FPPPlaylist.java b/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/dto/FPPPlaylist.java new file mode 100644 index 00000000000..0a2495bdf0f --- /dev/null +++ b/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/dto/FPPPlaylist.java @@ -0,0 +1,53 @@ +/** + * Copyright (c) 2010-2024 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.mqtt.fpp.internal; + +import com.google.gson.annotations.Expose; +import com.google.gson.annotations.SerializedName; + +/** + * The {@link FPPPlaylist} is responsible for storing + * the FPP JSON Status data. + * + * @author Scott Hanson - Initial contribution + */ +public class FPPPlaylist { + /* + * + * "current_playlist" : + * { + * "count" : "0", + * "description" : "", + * "index" : "0", + * "playlist" : "", + * "type" : "" + * } + * + */ + + @SerializedName("playlist") + @Expose + public String playlist; + + @SerializedName("description") + @Expose + public String description; + + @SerializedName("count") + @Expose + public int count; + + @SerializedName("index") + @Expose + public int index; +} diff --git a/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/dto/FPPScheduler.java b/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/dto/FPPScheduler.java new file mode 100644 index 00000000000..783ec83a5e6 --- /dev/null +++ b/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/dto/FPPScheduler.java @@ -0,0 +1,40 @@ +/** + * Copyright (c) 2010-2024 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.mqtt.fpp.internal; + +import com.google.gson.annotations.Expose; +import com.google.gson.annotations.SerializedName; + +/** + * The {@link FPPPlaylist} is responsible for storing + * the FPP JSON FPPScheduler data. + * + * @author Scott Hanson - Initial contribution + */ +public class FPPScheduler { + @SerializedName("currentPlaylist") + @Expose + public FPPCurrentPlaylist currentPlaylist; + + @SerializedName("nextPlaylist") + @Expose + public FPPNextPlaylist nextPlaylist; + + @SerializedName("enabled") + @Expose + public int enabled; + + @SerializedName("status") + @Expose + public String status; +} diff --git a/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/handler/FPPPlayerHandler.java b/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/handler/FPPPlayerHandler.java new file mode 100644 index 00000000000..bb89b7562a3 --- /dev/null +++ b/bundles/org.openhab.binding.mqtt.fpp/src/main/java/org/openhab/binding/mqtt/fpp/internal/handler/FPPPlayerHandler.java @@ -0,0 +1,282 @@ +/** + * Copyright (c) 2010-2024 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.mqtt.fpp.internal.handler; + +import static org.openhab.binding.mqtt.fpp.internal.FPPBindingConstants.*; + +import java.io.ByteArrayInputStream; +import java.io.IOException; +import java.math.BigDecimal; +import java.nio.charset.StandardCharsets; +import java.util.Properties; +import java.util.concurrent.ExecutionException; +import java.util.concurrent.TimeUnit; +import java.util.concurrent.TimeoutException; + +import org.eclipse.jdt.annotation.NonNullByDefault; +import org.eclipse.jdt.annotation.Nullable; +import org.openhab.binding.mqtt.fpp.internal.ConfigOptions; +import org.openhab.binding.mqtt.fpp.internal.FPPStatus; +import org.openhab.binding.mqtt.handler.AbstractBrokerHandler; +import org.openhab.core.io.net.http.HttpUtil; +import org.openhab.core.io.transport.mqtt.MqttBrokerConnection; +import org.openhab.core.io.transport.mqtt.MqttMessageSubscriber; +import org.openhab.core.library.types.IncreaseDecreaseType; +import org.openhab.core.library.types.NextPreviousType; +import org.openhab.core.library.types.OnOffType; +import org.openhab.core.library.types.PercentType; +import org.openhab.core.library.types.PlayPauseType; +import org.openhab.core.library.types.QuantityType; +import org.openhab.core.library.types.StringType; +import org.openhab.core.library.unit.Units; +import org.openhab.core.thing.Bridge; +import org.openhab.core.thing.ChannelUID; +import org.openhab.core.thing.Thing; +import org.openhab.core.thing.ThingRegistry; +import org.openhab.core.thing.ThingStatus; +import org.openhab.core.thing.ThingStatusDetail; +import org.openhab.core.thing.ThingStatusInfo; +import org.openhab.core.thing.binding.BaseThingHandler; +import org.openhab.core.thing.binding.ThingHandler; +import org.openhab.core.types.Command; +import org.openhab.core.types.RefreshType; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import com.google.gson.Gson; + +/** + * The {@link FPPPlayerHandler} is responsible for handling commands of the globes, which are then + * sent to one of the bridges to be sent out by MQTT. + * + * @author Scott Hanson - Initial contribution + */ +@NonNullByDefault +public class FPPPlayerHandler extends BaseThingHandler implements MqttMessageSubscriber { + private final Logger logger = LoggerFactory.getLogger(this.getClass()); + private @Nullable MqttBrokerConnection connection; + + private String fullStatusTopic = ""; + private String fullVersionTopic = ""; + private int lastKnownVolume = 50; + private String lastPlaylist = ""; + private ConfigOptions config = new ConfigOptions(); + + private final Gson gson = new Gson(); + + public FPPPlayerHandler(Thing thing, ThingRegistry thingRegistry) { + super(thing); + } + + private void processIncomingState(String messageJSON) { + FPPStatus data = gson.fromJson(messageJSON, FPPStatus.class); + updateState(CHANNEL_STATUS, new StringType(data.status_name)); + updateState(CHANNEL_PLAYER, data.status == 1 ? PlayPauseType.PLAY : PlayPauseType.PAUSE); + lastKnownVolume = data.volume; + updateState(CHANNEL_VOLUME, new PercentType(data.volume)); + updateState(CHANNEL_MODE, new StringType(data.mode_name)); + updateState(CHANNEL_CURRENT_SEQUENCE, new StringType(data.current_sequence)); + updateState(CHANNEL_CURRENT_SONG, new StringType(data.current_song)); + updateState(CHANNEL_CURRENT_PLAYLIST, new StringType(data.current_playlist.playlist)); + updateState(CHANNEL_SEC_PLAYED, new QuantityType<>(new BigDecimal(data.seconds_played), Units.SECOND)); + updateState(CHANNEL_SEC_REMAINING, new QuantityType<>(new BigDecimal(data.seconds_remaining), Units.SECOND)); + updateState(CHANNEL_UPTIME, new QuantityType<>(new BigDecimal(data.uptimeTotalSeconds), Units.SECOND)); + updateState(CHANNEL_BRIDGING, OnOffType.from(data.bridging)); + updateState(CHANNEL_MULTISYNC, OnOffType.from(data.multisync)); + updateState(CHANNEL_TESTING, data.status_name.equals(TESTING) ? OnOffType.ON : OnOffType.OFF); + + updateState(CHANNEL_SCHEDULER_STATUS, new StringType(data.scheduler.status)); + updateState(CHANNEL_SCHEDULER_NEXT_PLAYLIST, new StringType(data.scheduler.nextPlaylist.playlistName)); + updateState(CHANNEL_SCHEDULER_NEXT_PLAYLIST_START, + new StringType(data.scheduler.nextPlaylist.scheduledStartTimeStr)); + + if (data.scheduler.currentPlaylist != null) { + updateState(CHANNEL_SCHEDULER_CURRENT_PLAYLIST, + new StringType(data.scheduler.currentPlaylist.playlistName)); + updateState(CHANNEL_SCHEDULER_CURRENT_PLAYLIST_START, + new StringType(data.scheduler.currentPlaylist.playlistName)); + updateState(CHANNEL_SCHEDULER_CURRENT_PLAYLIST_END, + new StringType(data.scheduler.currentPlaylist.playlistName)); + updateState(CHANNEL_SCHEDULER_CURRENT_PLAYLIST_STOP_TYPE, + new StringType(data.scheduler.currentPlaylist.playlistName)); + } + + if (!data.current_playlist.playlist.isEmpty()) { + lastPlaylist = data.current_playlist.playlist; + updateState(CHANNEL_LAST_PLAYLIST, new StringType(lastPlaylist)); + } + + thing.setProperty(PROPERTY_UUID, data.uuid); + } + + @Override + public void handleCommand(ChannelUID channelUID, Command command) { + if (command instanceof RefreshType) { + return; + } + String channelId = channelUID.getId(); + if (channelId.equals(CHANNEL_PLAYER)) { + if (command == PlayPauseType.PAUSE || command == OnOffType.OFF) { + executeGet("/api/playlists/stop"); + } else if (command == PlayPauseType.PLAY || command == OnOffType.ON) { + if (!lastPlaylist.isEmpty()) { + executeGet("/api/playlist/" + lastPlaylist + "/start"); + } + } else if (command == NextPreviousType.NEXT) { + executeGet("/api/command/Next Playlist Item"); + } else if (command == NextPreviousType.PREVIOUS) { + executeGet("/api/command/Prev Playlist Item"); + } + } + if (channelId.equals(CHANNEL_VOLUME)) { + Integer volume = null; + if (command instanceof PercentType percentCommand) { + volume = percentCommand.intValue(); + } else if (command == OnOffType.OFF) { + volume = 0; + } else if (command == OnOffType.ON) { + volume = lastKnownVolume; + } else if (command == IncreaseDecreaseType.INCREASE) { + if (lastKnownVolume < 100) { + lastKnownVolume++; + volume = lastKnownVolume; + } + } else if (command == IncreaseDecreaseType.DECREASE) { + if (lastKnownVolume > 0) { + lastKnownVolume--; + volume = lastKnownVolume; + } + } + if (volume != null) { + lastKnownVolume = volume; + executePost("/api/system/volume", "{\"volume\":" + lastKnownVolume + "}"); + updateState(CHANNEL_VOLUME, new PercentType(lastKnownVolume)); + } + } + if (channelId.equals(CHANNEL_TESTING)) { + if (command == OnOffType.OFF) { + executePost("/api/testmode", GetTestMode(0)); + } else if (command == OnOffType.ON) { + executePost("/api/testmode", GetTestMode(1)); + } + } + } + + private String GetTestMode(int enable) { + return "{\"mode\":\"RGBChase\",\"subMode\":\"RGBChase-RGB\",\"cycleMS\":1000,\"colorPattern\":\"FF000000FF000000FF\",\"enabled\":" + + enable + ",\"channelSet\": \"1-1048576\",\"channelSetType\": \"channelRange\"}"; + } + + @Override + public void initialize() { + updateStatus(ThingStatus.UNKNOWN); + config = getConfigAs(ConfigOptions.class); + if (!config.playerMQTTTopic.isEmpty()) { + fullStatusTopic = config.playerMQTTTopic + "/" + STATUS_TOPIC; + fullVersionTopic = config.playerMQTTTopic + "/" + VERSION_TOPIC; + bridgeStatusChanged(getBridgeStatus()); + updateStatus(ThingStatus.ONLINE); + } + } + + @Override + public void processMessage(String topic, byte[] payload) { + String state = new String(payload, StandardCharsets.UTF_8); + logger.trace("Received the following new FPP state:{}:{}", topic, state); + + if (topic.endsWith(STATUS_TOPIC)) { + processIncomingState(state); + } else if (topic.endsWith(VERSION_TOPIC)) { + thing.setProperty(PROPERTY_SOFTWARE_VERSION, state); + updateStatus(ThingStatus.ONLINE); + } + } + + public ThingStatusInfo getBridgeStatus() { + Bridge b = getBridge(); + if (b != null) { + return b.getStatusInfo(); + } else { + return new ThingStatusInfo(ThingStatus.OFFLINE, ThingStatusDetail.BRIDGE_OFFLINE, null); + } + } + + @Override + public void bridgeStatusChanged(ThingStatusInfo bridgeStatusInfo) { + if (bridgeStatusInfo.getStatus() == ThingStatus.OFFLINE) { + updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.BRIDGE_OFFLINE); + connection = null; + return; + } + + Bridge localBridge = this.getBridge(); + if (localBridge == null) { + updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.BRIDGE_UNINITIALIZED, + "Bridge is missing or offline, you need to setup a working MQTT broker first."); + return; + } + ThingHandler handler = localBridge.getHandler(); + if (handler instanceof AbstractBrokerHandler abh) { + final MqttBrokerConnection connection; + try { + connection = abh.getConnectionAsync().get(5000, TimeUnit.MILLISECONDS); + } catch (InterruptedException | ExecutionException | TimeoutException ignored) { + updateStatus(ThingStatus.OFFLINE, ThingStatusDetail.BRIDGE_UNINITIALIZED, + "Bridge handler has no valid broker connection!"); + return; + } + this.connection = connection; + updateStatus(ThingStatus.UNKNOWN); + if (!fullStatusTopic.isEmpty()) { + connection.subscribe(fullStatusTopic, this); + connection.subscribe(fullVersionTopic, this); + } + } + return; + } + + private @Nullable String executeGet(String url) { + String response = null; + try { + response = HttpUtil.executeUrl("GET", "http://" + config.playerAddress + url, 5000); + + } catch (IOException e) { + logger.warn("Failed HTTP Post", e); + } + return response; + } + + private boolean executePost(String url, String json) { + try { + Properties header = new Properties(); + header.put("Accept", "application/json"); + header.put("Content-Type", "application/json"); + String response = HttpUtil.executeUrl("POST", "http://" + config.playerAddress + url, header, + new ByteArrayInputStream(json.getBytes()), "application/json", 5000); + + return !response.isEmpty(); + } catch (IOException e) { + logger.warn("Failed HTTP Post", e); + } + return false; + } + + @Override + public void dispose() { + MqttBrokerConnection localConnection = connection; + if (localConnection != null) { + localConnection.unsubscribe(fullStatusTopic, this); + localConnection.unsubscribe(fullVersionTopic, this); + } + } +} diff --git a/bundles/org.openhab.binding.mqtt.fpp/src/main/resources/OH-INF/config/config.xml b/bundles/org.openhab.binding.mqtt.fpp/src/main/resources/OH-INF/config/config.xml new file mode 100644 index 00000000000..82c725d815a --- /dev/null +++ b/bundles/org.openhab.binding.mqtt.fpp/src/main/resources/OH-INF/config/config.xml @@ -0,0 +1,19 @@ + + + + + + network-address + + Player IP Address or Host Name + + + name + + MQTT Player Status Topic + + + diff --git a/bundles/org.openhab.binding.mqtt.fpp/src/main/resources/OH-INF/i18n/mqtt.properties b/bundles/org.openhab.binding.mqtt.fpp/src/main/resources/OH-INF/i18n/mqtt.properties new file mode 100644 index 00000000000..542c765a319 --- /dev/null +++ b/bundles/org.openhab.binding.mqtt.fpp/src/main/resources/OH-INF/i18n/mqtt.properties @@ -0,0 +1,56 @@ +# thing types + +thing.label.player=FPP Player +thing.description.player=FPP Player + +# thing types config + +thing-type.config.mqtt.player.playerAddress.label = Player Address +thing-type.config.mqtt.player.playerAddress.description = Player IP Address or Host Name +thing-type.config.mqtt.player.playerMQTTTopic.label = MQTT Player Topic +thing-type.config.mqtt.player.playerMQTTTopic.description = MQTT Player Status Topic + +# channel types + +channel-type.mqtt.player.status.label = Status +channel-type.mqtt.player.status.description = FPP Player Status +channel-type.mqtt.player.mode.label = Mode +channel-type.mqtt.player.mode.description = FPP Player Mode +channel-type.mqtt.player.current-sequence.label = Current Sequence +channel-type.mqtt.player.current-sequence.description = FPP Current Sequence +channel-type.mqtt.player.current-song.label = Current Song +channel-type.mqtt.player.current-song.description = FPP Current Song +channel-type.mqtt.player.current-playlist.label = Current Playlist +channel-type.mqtt.player.current-playlist.description = FPP Current Playlist +channel-type.mqtt.player.seconds-played.label = Seconds Played +channel-type.mqtt.player.seconds-played.description = FPP Seconds Played +channel-type.mqtt.player.seconds-remaining.label = Seconds Remaining +channel-type.mqtt.player.seconds-remaining.description = FPP Seconds Remaining +channel-type.mqtt.player.uptime.label = Uptime +channel-type.mqtt.player.uptime.description = FPP System Uptime (time after start) +channel-type.mqtt.player.player.label = Player +channel-type.mqtt.player.player.description = FPP Player Control +channel-type.mqtt.player.volume.label = Volume +channel-type.mqtt.player.volume.description = FPP Volume of the Output +channel-type.mqtt.player.bridging-enabled.label = Bridging +channel-type.mqtt.player.bridging-enabled.description = FPP Recieving Bridge Data +channel-type.mqtt.player.multisync-enabled.label = Multisync +channel-type.mqtt.player.multisync-enabled.description = FPP Multisync Mode Enabled +channel-type.mqtt.player.testing-enabled.label = Testing +channel-type.mqtt.player.testing-enabled.description = FPP Is In Test Mode +channel-type.mqtt.player.last-playlist.label = Last Run Playlist +channel-type.mqtt.player.last-playlist.description = FPP Last Run Playlist +channel-type.mqtt.player.scheduler-status.label = Scheduler Status +channel-type.mqtt.player.scheduler-status.description = FPP Scheduler Status +channel-type.mqtt.player.scheduler-current-playlist.label = Scheduler Current Playlist +channel-type.mqtt.player.scheduler-current-playlist.description = FPP Scheduler Current Playlist +channel-type.mqtt.player.scheduler-current-playlist-start.label = Scheduler Current Playlist Start +channel-type.mqtt.player.scheduler-current-playlist-start.description = FPP Scheduler Current Playlist Start Time +channel-type.mqtt.player.scheduler-current-playlist-end.label = Scheduler Current Playlist End +channel-type.mqtt.player.scheduler-current-playlist-end.description = FPP Scheduler Current Playlist End Time +channel-type.mqtt.player.scheduler-current-playlist-stop-type.label = Scheduler Current Playlist Stop Type +channel-type.mqtt.player.scheduler-current-playlist-stop-type.description = FPP Scheduler Current Playlist Stop Type +channel-type.mqtt.player.scheduler-next-playlist.label = Scheduler Next Playlist +channel-type.mqtt.player.scheduler-next-playlist.description = FPP Scheduler Next Playlist +channel-type.mqtt.player.scheduler-next-playlist-start.label = Scheduler Next Playlist Start +channel-type.mqtt.player.scheduler-next-playlist-start.description = FPP Scheduler Next Playlist Start Time diff --git a/bundles/org.openhab.binding.mqtt.fpp/src/main/resources/OH-INF/thing/thing-types.xml b/bundles/org.openhab.binding.mqtt.fpp/src/main/resources/OH-INF/thing/thing-types.xml new file mode 100644 index 00000000000..99076559f5d --- /dev/null +++ b/bundles/org.openhab.binding.mqtt.fpp/src/main/resources/OH-INF/thing/thing-types.xml @@ -0,0 +1,169 @@ + + + + + + + + + FPP Player + + + + + + + + + + + + + + + + + + + + + + + + + + + + unknown + unknown + + + + + + String + + FPP Player Status + + + + String + + FPP Player Mode + + + + String + + FPP Current Sequence + + + + String + + FPP Current Song + + + + String + + FPP Current Playlist + + + + Number:Time + + FPP Seconds Played + + + + Number:Time + + FPP Seconds Remaining + + + + Number:Time + + FPP System Uptime (time after start) + + + + Player + + FPP Player Control + + + Dimmer + + FPP Volume of the Output + + + Switch + + FPP Recieving Bridge Data + + + + Switch + + FPP Multisync Mode Enabled + + + + Switch + + FPP Is In Test Mode + + + String + + FPP Last Run Playlist + + + + String + + FPP Scheduler Status + + + + String + + FPP Scheduler Current Playlist + + + + String + + FPP Scheduler Current Playlist Start Time + + + + String + + FPP Scheduler Current Playlist End Time + + + + String + + FPP Scheduler Current Playlist Stop Type + + + + String + + FPP Scheduler Next Playlist + + + + String + + FPP Scheduler Next Playlist Start Time + + + diff --git a/bundles/pom.xml b/bundles/pom.xml index b3a32868783..94b76644521 100644 --- a/bundles/pom.xml +++ b/bundles/pom.xml @@ -270,6 +270,7 @@ org.openhab.binding.mpd org.openhab.binding.mqtt org.openhab.binding.mqtt.espmilighthub + org.openhab.binding.mqtt.fpp org.openhab.binding.mqtt.generic org.openhab.binding.mqtt.homeassistant org.openhab.binding.mqtt.homie diff --git a/features/openhab-addons/src/main/resources/footer.xml b/features/openhab-addons/src/main/resources/footer.xml index 383959548d4..9b241a74a70 100644 --- a/features/openhab-addons/src/main/resources/footer.xml +++ b/features/openhab-addons/src/main/resources/footer.xml @@ -28,6 +28,7 @@ mvn:ch.obermuhlner/big-math/2.3.2 mvn:org.openhab.addons.bundles/org.openhab.binding.mqtt/${project.version} mvn:org.openhab.addons.bundles/org.openhab.binding.mqtt.espmilighthub/${project.version} + mvn:org.openhab.addons.bundles/org.openhab.binding.mqtt.fpp/${project.version} mvn:org.openhab.addons.bundles/org.openhab.binding.mqtt.generic/${project.version} mvn:org.openhab.addons.bundles/org.openhab.binding.mqtt.homeassistant/${project.version} mvn:org.openhab.addons.bundles/org.openhab.binding.mqtt.homie/${project.version}