Vielleicht sind hier ein paar Leute, die mitdenken und/oder sich schon Gedanken darüber gemacht haben. Ich habe ja bei vielen Geräten die Möglichkeit REST oder MQTT zur Datenübertragung zu verwenden. Zumindest fehlt ab openHAB 2.5 ein Eventbus, so dass von einer zur nächsten openHAB-Instanz eine MQTT-Lösung nicht gerade sinnig erscheint. Das Remote openHAB Binding verwendet wiederum ebenfalls die REST API. Versuchen wir doch einfach mal beide Seiten zu beleuchten:
Anwendungsfälle MQTT
- MQTT wird bspw. auch bei kleinen selbstgebauten Sensoren/Aktoren die über Raspberry Pi's oder Arduino gesteuert werden eingesetzt, weil es leichtgewichtig ist.
- Ein paar Sensoren/Aktoren aus der Industrie verwenden MQTT
- Bis einschließlich openHAB 2.4 ein MQTT Eventbus über bspw. das Legacy MQTT 1.x Binding, welches openHAB ohne REST über mehrere Instanzen verteilen konnte. Hierzu gibt es verschiedene Netztopologien und Aufbauten.
- Fahrzeuge, Roboter etc.
- Eingebettete Systeme
- Rechner und Rechnerinfrastrukturen
Anwendungsfälle REST
- Android/iOS App bspw. von openHAB
- Clientseitige Software wie bspw. eine lokal installierte openHAB-Instanz
- Serverseitige Software wie bspw. die Software openHAB Cloud, die z.B. auf https://myopenhab.org/ gehostet ist und von jedem selbst eingesetzt werden könnte: https://github.com/openhab/openhab-cloud
- Amazon Alexa über openHAB Cloud
- IFTTT über eine selbst gehostete openHAB Cloud, da es seit 2019 von myopenhab.org abgeschalten worden ist, wegen zu hohen Serverlasten.

Zu unterscheiden gibt es hier, dass es zwei REST APIs gibt. Eine auf der lokalen openHAB Instanz und eine durch die Software openHAB Cloud:
Verwendet man diese Software über myopenhab.org, dann wäre dies theoretisch die URL zur REST API:
https://myopenhab.org/rest/items
Bzw. noch mit den entsprechenden Anmeldeinformationen. Auf der lokalen openHAB-Instanz entsprechend mit:
[url]https://<ip>:8080/rest/items[/url]
An diesem Beispiel kann man ableiten, dass die Software openHAB Cloud erfolgreich REST-Abfragen stellt, um auf die Daten der lokalen openHAB-Instanz zuzugreifen. Diese wird quasi geklont und wäre im World Wide Web verfügbar (mit Sicherheitseinschränkungen wie bspw. das man bei myopenhab.org eingelogt sein muss, oAuth 2.0 bei Anwendungen implementiert sein müssen und der Benutzer den Zugriff bestätigen muss). Heißt in der openHAB Cloud REST API befinden sich zu jeder Zeit (außer durch Übertragungsfehler/-probleme), dieselben Daten, wie lokal. Bedient man openHAB also über diese Software extern, werden diese Änderungen aber auch über REST an die lokale openHAB-Instanz gesendet.
So funktioniert dann bspw. auch IFTTT, Alexa oder die Smartphone App. Und Kritik könnten jetzt sagen, dass die Schilderung mit Client-Server jetzt auch nicht ganz zutreffend ist auf die lokale openHAB-Instanz... Ist ja nur ein Beispiel.
Als nächstes schauen wir uns die technische Seite von beiden ein bisschen näher an.
MQTT
- hieß ursprünglich Message Queuing Telemetry Transport
- Netzwerkprotokoll für Machine-to-Machine-Kommunikation (M2M)
- Übertragung von Telemetriedaten in Form von Nachrichten zwischen Geräten (trotz hoher Verzögerungen oder beschränkter Netzwerke)
- Die Internet Assigned Numbers Authority (IANA) reserviert für MQTT die Ports 1883 und 8883
- MQTT-Nachrichten können mit dem TLS-Protokoll verschlüsselt werden
- Problemlose Umsetzung in vielen Programmier- und Skriptsprachen wie Java, Jython, JavaScript, Python usw.
- Broker-Client-Architektur
- Publisher-Subscriber-Pattern
- Nutzt TCP/IP

Nehmt es mir nicht übel, wenn ich der Einfachheit her aus Wikipedia kopiere:
Interessant ist, dass ein MQTT-Server („Broker“) die gesamte Datenlage seiner Kommunikationspartner hält, und so als Zustands-Datenbank benutzt werden kann. So ist es möglich, kleine unperformante MQTT-Geräte mit einem MQTT-Broker zu verbinden, wobei die Geräte Daten einsammeln und/oder Befehle entgegennehmen, während ein komplexes Lagebild nur auf dem MQTT-Broker entsteht und hier oder durch einen leistungsfähigen Kommunikationspartner ausgewertet werden kann. Stelleingriffe können so von einer oder mehreren leistungsfähigen Instanzen an den MQTT-Broker übermittelt und auf die einzelnen Geräte verbreitet werden. Dadurch eignet sich MQTT sehr gut für Automatisierungslösungen und findet im Bereich IoT durch die einfache Verwendung große Verbreitung.
Schauen wir uns mal ein Beispiel zur Client-Broker-Architektur an:

Den Broker können wir wie eine Art Server vorstellen, der Anfragen verteilt und managt. Ein Sensor stellt bspw. seine Sensorinformationen über einen Kanal/Channel zur Verfügung bzw. published diese Informationen. Dazu muss dieser Sensor beim Broker registriert sein. Ein oder mehrere Client/s greifen diese Daten nach ihrer Registrierung beim Broker ab, wenn sie vom entsprechenden Kanal das richtig Thema bzw. Topic kennen. Dies ist ein Subscribe-Vorgang.

Das besondere an dem Publisher-Subscriber-Vorgang ist: Solange der Sensor published, werden Informationen bereitgetstellt. Jeder Client kann diese Informationen solange abfragen, wie er sie benötigt. Er kann Zeit versetzt kommen, er kann die Verbindung unterbrechen und wieder neu aufnehmen. Anders als bei einem Request-Response-Modell muss er nicht jedes mal neu eine Anfrage stellen, um letzten Endes Daten zu erhalten. Man kann sich diese als eine Leitung vorstellen, in dem einfach nur gehorcht wird, ob Daten verfügbar sind. Ändern sich diese, werden auch Änderungen ausgegeben. Sind sie nicht mehr vorhanden, wird nichts ausgegeben und wird nicht mehr nachgefragt, sind sie immer noch in der Leitung für andere Clients. Man "horcht" quasi mit.
MQTT ist ein Client-Server-Protokoll. Clients senden dem Server (“Broker”) nach Verbindungsaufbau Nachrichten mit einem Topic, welches die Nachricht hierarchisch einstuft; zum Beispiel Küche/Kühlschrank/Temperatur oder Auto/Rad/3/Luftdruck. Clients können diese Topics abonnieren, wobei der Server die empfangenen Nachrichten an die entsprechenden Abonnenten weiterleitet.
Nachrichten bestehen immer aus einem Topic und dem Nachrichteninhalt. Nachrichten werden mit einer definierbaren Quality of Service versendet: at most once (die Nachricht wird einmal gesendet und kommt bei Verbindungsunterbrechung möglicherweise nicht an), at least once (die Nachricht wird so lange gesendet, bis der Empfang bestätigt wird, und kann beim Empfänger mehrfach ankommen) und exactly once (hierbei wird sichergestellt, dass die Nachricht auch bei Verbindungsunterbrechung genau einmal ankommt). Außerdem kann mit dem Retain-Flag der Server angewiesen werden, die Nachricht zu diesem Topic zwischenzuspeichern. Clients, die dieses Thema neu abonnieren, bekommen als erstes die zwischengespeicherte Nachricht zugestellt.
Beim Verbindungsaufbau können Clients einen „letzten Willen“ in Form einer Nachricht definieren. Falls die Verbindung zum Client verloren geht, wird diese Nachricht publiziert und dabei an die entsprechenden Abonnenten gesendet.
MQTT wird üblicherweise über TCP benutzt und hat einen 2-Byte-Header. Das erste Byte enthält den Nachrichtentyp (4 Bit), den Quality of Service (2 Bit) und ein Retain-Flag.

Beispielablauf einer MQTT-Verbindung mit Publish und Subscribe. Die erste Nachricht von Client B wird vom Broker aufgrund des gesetzten Retain-Flags gespeichert.
Es gibt folgende Nachrichten-Typen:
- CONNECT
- CONNACK
- PUBLISH
- PUBACK
- PUBREC
- PUBREL
- PUBCOMP
- SUBSCRIBE
- SUBACK
- UNSUBSCRIBE
- UNSUBACK
- PINGREQ
- PINGRESP
- DISCONNECT
Daran schließt sich ein variabler Teil an, der das MQTT-Topic, also das Thema enthält. Abschließend kommt die Payload, also der Dateninhalt, der unter dem Thema veröffentlicht wird.
Die Themen sind hierarchisch organisiert. Zum Beispiel:
Code: Alles auswählen
Neujahrsansprache/1984/audio/ogg
Neujahrsansprache/1984/audio/mp3
Neujahrsansprache/1984/video
Neujahrsansprache/1984/text/ascii
Neujahrsansprache/1984/text/odt
Mit einem + kann eine Hierarchie-Ebene als Wildcard gesetzt werden.
Code: Alles auswählen
Neujahrsansprache/1984/# // Alles der Neujahrsansprache von 1984
Neujahrsansprache/+/text/ascii // Die ASCII-Texte aller Neujahrsansprachen
Neujahrsansprache/+/audio/# // Alle Audio-Formate aller Neujahrsansprachen
REST
- heißt Representational State Transfer (Repräsentative Zustandsübertragung)
- Paradigma für die Softwarearchitektur von verteilten Systemen, insbesondere für Webservices
- Abstraktion der Struktur und des Verhaltens des World Wide Web
- Ziel: Ein Architekturstil zu schaffen, der den Anforderungen des modernen Web besser genügt. Dabei unterscheidet sich REST vor allem in der Forderung nach einer einheitlichen Schnittstelle von anderen Architekturstilen.
- Netzwerkprotokoll für Machine-to-Machine-Kommunikation (M2M)
- REST kodiert keine Methodeninformation in den URI, da der URI Ort und Namen der Ressource angibt, nicht aber die Funktionalität, die der Web-Dienst zu der Ressource anbietet (im vgl. zu SOAP, WDSL oder RPC)
- Der Vorteil von REST liegt darin, dass im WWW bereits ein Großteil der für REST nötigen Infrastruktur (z. B. Web- und Application-Server, HTTP-fähige Clients, HTML- und XML-Parser, Sicherheitsmechanismen) vorhanden ist, und viele Web-Dienste per se REST-konform sind --> Eine Ressource kann dabei über verschiedene Medientypen dargestellt werden (Repräsentation der Ressource).
- Die Bezeichnung „Representational State Transfer“ soll den Übergang vom aktuellen Zustand zum nächsten Zustand (state) einer Applikation verbildlichen. Dieser Zustandsübergang erfolgt durch den Transfer der Daten, die den nächsten Zustand repräsentieren
- So ist ein Online-Dienst, der lediglich unveränderte Seiteninhalte nach dem Internetstandard HTTP anbietet, bereits REST-konform. Dynamisch erzeugte Seiten folgen diesem Paradigma jedoch oft nicht. So bieten beispielsweise Nachrichtenseiten sich ständig ändernde Informationen mit sowohl unterschiedlichem Format als auch Inhalt an, die nur schwer automatisch verarbeitet werden können. Bliebe das Format unverändert, so wäre eine wichtige REST-Eigenschaft erfüllt. So wäre eine Webseite, auf der ständig die aktuelle Uhrzeit in immer demselben Format abrufbar ist, REST-konform
- Problemlose Umsetzung in vielen Programmier- und Skriptsprachen wie Java, Jython, JavaScript, Python usw.
- Client-Server-Architektur
- Zustandslosigkeit
- Caching
- Einheitliche Schnittstellen
- Mehrsprachige Systeme
- Request-Response-Pattern
- nutzt das http-Protokoll
- Nutzt TCP/IP (in seltenen Fällen auch UDP)
- GET: Fordert die angegebene Ressource vom Server an. GET weist keine Nebeneffekte auf. Der Zustand am Server wird nicht verändert, weshalb GET als sicher bezeichnet wird.
- PUT: Die angegebene Ressource wird angelegt. Wenn die Ressource bereits existiert, wird sie geändert.
- POST: Fügt eine neue (Sub-)Ressource unterhalb der angegebenen Ressource ein. Da die neue Ressource noch keinen URI besitzt, adressiert der URI die übergeordnete Ressource. Als Ergebnis wird der neue Ressourcenlink dem Client zurückgegeben. POST kann im weiteren Sinne auch dazu verwendet werden, Operationen abzubilden, die von keiner anderen Methode abgedeckt werden.
- DELETE: Löscht die angegebene Ressource.
Code: Alles auswählen
items
get /items
Get all available items.
put /items
Adds a list of items to the registry or updates the existing items.
delete /items/{itemName}/members/{memberItemName}
Removes an existing member from a group item.
put /items/{itemName}/members/{memberItemName}
Adds a new member to a group item.
delete /items/{itemname}
Removes an item from the registry.
get /items/{itemname}
Gets a single item.
post /items/{itemname}
Sends a command to an item.
put /items/{itemname}
Adds a new item to the registry or updates the existing item.
delete /items/{itemname}/metadata/{namespace}
Removes metadata from an item.
put /items/{itemname}/metadata/{namespace}
Adds metadata to an item.
get /items/{itemname}/state
Gets the state of an item.
put /items/{itemname}/state
Updates the state of an item.
delete /items/{itemname}/tags/{tag}
Removes a tag from an item.
put /items/{itemname}/tags/{tag}
Adds a tag to an item.
Bei der DNS-Versionierung wird die Version als Bestandteil der DNS-Domain behandelt.
Code: Alles auswählen
http://v1.api.foo.com/customer/1234
http://v1_1.api.foo.com/customer/1234
http://v2.api.foo.com/customer/1234
http://v2_2.api.foo.com/customer/1234
Bei der URL-Versionierung wird die Version der Schnittstelle im Pfad der URL angegeben:
Code: Alles auswählen
http://foo.com/api/v1/customer/1234
http://foo.com/api/v1.1/customer/1234
http://foo.com/api/v2.0/customer/1234
http://foo.com/api/v2.2/customer/1234
Übertragen werden Daten in der Regel im JSON-Format:

Die JavaScript Object Notation ist ein kompaktes Datenformat in einer einfach lesbaren Textform für den Datenaustausch zwischen Anwendungen. JSON ist von der Programmiersprache unabhängig. Parser und Generatoren existieren in allen verbreiteten Sprachen. JSON wurde ursprünglich von Douglas Crockford spezifiziert.
Ein Beispiel aus openHAB könnte so aussehen:
Code: Alles auswählen
link "http://<ip>:8080/…oT_Hue_Lampe6_Helligkeit"
state "185,57,0"
editable false
type "Color"
name "iIoT_Hue_Lampe6_Helligkeit"
label "Farbe"
tags
0 "Lighting"
groupNames []
Eine allgemeine Darstellung sehe bspw. so aus:
[img][/img]
Ein Beispiel für mehrere Ressourcen-Abfragen wäre ja bereits ein einfacher Webseiten-Aufruf, der so aussehen könnte:

Jedes Item wäre bspw. auch eine einzelne Ressource.
Polling
Polling ist eine zyklische Abfrage, um den Status (Hard- oder Software) oder das Ereignis einer Wertänderung mit zyklischem Abfragen zu ermitteln.
Hardware, deren Status benötigt wird, besteht häufig aus Ports elektrischer Schaltungen, Schnittstellen oder externen Geräten. Software-Zustände, die häufig abgefragt werden, sind Dateisperren auf anderen Rechnern oder Semaphore. Ein möglicher Zweck des Pollings ist das aktive Warten auf Zustandsänderungen, auch Spinning genannt. Eine andere Form ist die Abfrage jeweils einmal in einem Abtastzyklus, oder die Abfrage nach jeweils einer anderen Aktivität.
Polling ist unter folgenden Bedingungen sinnvoll:
- Das einfache explizit serielle und bestimmbare Verhalten von Polling-Programmierungen kann gegenüber potentiell komplexeren parallelen Alternativen erwünscht sein.
- Es handelt sich um eine sehr schnelle Abfrage beispielsweise einer Hardwareanschaltung, die binnen Mikrosekunden den gewünschten Zustand einnimmt.
- Es ist eine zyklische Aktivität vorhanden, in der das Polling stattfinden kann, während alternative Lösungen mehr Aufwand erfordern. Dies ist oft bei zyklischen Regelungen der Fall.
- Es existieren keine weiteren Verbindungen zum Partner außer der Abfragemöglichkeit. Das ist etwa bei lose gekoppelten Systemen gegeben, beispielsweise beim Abfragen, ob eine Datei auf einem Dateisystem (gegebenenfalls auf einem entfernten Rechner) existiert oder freigegeben ist, die von einem anderen Programm bereitgestellt wird oder gesperrt ist.
- Durch Polling ist häufig die Leistungsfähigkeit von Programmen nicht-deterministisch (z. B. variierende FPS) und meistens geringer als mit Alternativlösungen.
- Die Effizienz bei der Verwendung von Systemressourcen kann signifikant geringer sein als Alternativansätze. Beispielsweise kann eine naive Polling-Implementierung, z. B. ein mit maximaler Geschwindigkeit pollendes Programm, die Systemlast auf 100 % treiben (mit unnötigem Stromverbrauch und Abwärme), wogegen eine Event-basierte- oder Hardwareinterrupt-Lösung praktisch keine Systemressourcen verbraucht.
Das Remote openHAB Binding verwendet wie bereits erwähnt die REST-Schnittstelle von openHAB. Genauer SSE REST:
Server-Sent Events (SSE) ist eine Server-Push-Technologie, die es einem Client ermöglicht, automatische Aktualisierungen von einem Server über eine HTTP-Verbindung zu empfangen, und beschreibt, wie Server die Datenübertragung in Richtung Clients initiieren können, sobald eine erste Client-Verbindung hergestellt wurde. Sie wird üblicherweise verwendet, um Nachrichten-Updates oder kontinuierliche Datenströme an einen Browser-Client zu senden und wurde entwickelt, um das native, browserübergreifende Streaming durch eine JavaScript-API namens EventSource zu verbessern, über die ein Client eine bestimmte URL anfordert, um einen Ereignisstrom zu empfangen. Die Server-Sent Events Event Source-API ist als Teil von HTML5 durch das W3C standardisiert.
Grafisch kann man sich dies wie folgt vorstellen:
[img][/img]
Fazit
Ich hoffe es ist klar, dass der Unterschied an den Requests-Responses im Vergleich zu Publish-Subscribe der eigentliche Unterschied ist. Mehr oder weniger vergleicht man nicht MQTT mit REST sondern mit HTTP. Damit Änderungen erkannt werden, müsste man bei MQTT nichts triggern. Ohne SSE müsste man permanent beim Server nachfragen, ob sich Daten verändern würden. Mit SSE ist dies schon sehr viel angenehmer, da automatisch bei Änderungen quasi getriggert wird.
Sollte man nicht das Remote openHAB Binding verwenden und eigene REST-Schnittstellen zu openHAB entwickeln wollen, müsste man genau diesen Punkt mit Request-Response berücksichtigen. Man muss sich an einer SSE REST-Schnittstelle orientieren. Ich hoffe ich konnte erläutern, warum. Damit dürfte MQTT für viele Anwendungen in openHAB 3 obsolet sein.