class: title, smokescreen, shelf, bottom, no-footer background-image: url(images/pub-sub.png) # 181U Spring 2020 ### IoT Protocols (part 1) --- layout: true .footer[ - Geoffrey Brown, 2020 - 181U ] <style> h1 { border-bottom: 8px solid rgb(32,67,143); border-radius: 2px; width: 90%; } .smokescreen h1 { border-bottom: none; } .small.remark-slide-content.compact {font-size:1.2rem} .smaller.remark-slide-content.compact {font-size:1.1rem} .small-code.remark-slide-content.compact code {font-size:1.0rem} .very-small-code.remark-slide-content.compact code {font-size:0.9rem} .line-numbers{ /* Set "line-numbers-counter" to 0 */ counter-reset: line-numbers-counter; } .line-numbers .remark-code-line::before { /* Increment "line-numbers-counter" by 1 */ counter-increment: line-numbers-counter; content: counter(line-numbers-counter); text-align: right; width: 20px; border-right: 1px solid #aaa; display: inline-block; margin-right: 10px; padding: 0 5px; } </style> --- class: compact # Agenda * MQTT * Publish-Subscribe Model * MQTT Example * Protocol * Brokers * JSON * COAP --- class: compact # The problem data:image/s3,"s3://crabby-images/b15b1/b15b11a4f072858b5a49044a40d7c899c92e4663" alt="" * "Traditional" applications based upon reliable byte streams between endpoints * Not a good fit for many-to many communication (e.g. Twitter, Snapchat, etc) Image from Wikimedia commons --- class: compact # Ways to exchange messages data:image/s3,"s3://crabby-images/3381c/3381c5d68d92935bc3eab5cd22f652c7d8b6d3b4" alt="" data:image/s3,"s3://crabby-images/84921/8492121e109d9bfee09aa89a42f89f3605283b4b" alt="" --- class:compact # Request/Response * **REST** : Representational State Transfer * Widely used; often based upon HTTP * Later we'll discuss a lighter approach **CoAP** for IoT data:image/s3,"s3://crabby-images/fb0e2/fb0e221f09080bf1e60c4b55b49428f913150674" alt="" data:image/s3,"s3://crabby-images/48da5/48da5bd8a71dba1626c41b10d6ab7b0e1423b724" alt="" --- class: compact # Why isn't HTTP enough ? * The HTTP standard revolutionized how *people* consume data * A single simple model: Send a request, read the response * Available via any tablet, laptop, phone, etc. * The Internet of Things has fundamentally different challenges * HTTP remains a good way to request data from a known source * But we need an event-oriented paradigm: * Emitting information: *one to many* * Listening for events *whenever they happen* * Distrubute huge volumes of small data packets * *Push* information over *unreliable networks* --- class: compact # Pub/Sub Approach data:image/s3,"s3://crabby-images/84578/845784258ed548611de6f9a53c0d6fb7bb6ccdc9" alt="" * Publish/Subscribe * producer/consumer * Various protocols * MQTT, AMQP, XMPP * Pub/sub separates **publisher** who sends messages about specific **topics** from **subscriber** who receives messages about specific **topics** * The protocol depends upon a **broker** known by both **publishers** and **subscribers** which filters incoming messages and distributes them. --- class: compact # MQTT > MQTT is a machine-to-machine (M2M)/Internet of Things connectivity protocol. > It was designed as an extremely lightweight publish/subscribe messaging transport. It is useful for connections with remote > locations where a small code footprint is required and/or network bandwidth is at a premium. (http://mqtt.org) data:image/s3,"s3://crabby-images/fb0e2/fb0e221f09080bf1e60c4b55b49428f913150674" alt="" data:image/s3,"s3://crabby-images/bb9f4/bb9f4b9bcfcd9752c4594e2a72080849b4cefac6" alt="" Figure:http://istsos.org/en/trunk/doc/ws_mqtt.html --- class: compact # The Realm of MQTT data:image/s3,"s3://crabby-images/3759f/3759f5baba3ebea02b4067ed6d34e86a1a0937ed" alt="" data:image/s3,"s3://crabby-images/f1734/f1734b72ad709d2115dd6f21544e10574c6264e0" alt="" --- class: compact # Background/Goals of MQTT * Make it simple to connect physical world to traditional IT world * Expect and cater to frequent network disruption -- built for *low bandwidth*, *high-latency*, *unreliable*, *high-cost* networks (per byte) * Expect client applications may have very limited resources (8-bit, small memory) * Provide loose coupling to support dynamic system environments * Provide multiple deterministic message delivery quialities * Capable of supporting very large numbers of devices * Simple for developers to implement protocol --- class: compact,small # MQTT works on top of ... * Mainly TCP * There is a closely related protocol MQTT-SN (MQTT for sensor networks) that runs on UDP, but COAP is a UDP protocol that is used more which we will look at. * Websockets can also be used. * Enables embedding in web pages. data:image/s3,"s3://crabby-images/3759f/3759f5baba3ebea02b4067ed6d34e86a1a0937ed" alt="" data:image/s3,"s3://crabby-images/f5b7f/f5b7ff75ef7e428a4a5ad211f937788c257afa93" alt="" * Both TCP and websockets can operate on top of "transport layer security" (TLS) and SSL. --- class: compact # MQTT Example data:image/s3,"s3://crabby-images/de62a/de62a8c071974f1225e742d5abd709f4b7654213" alt="" * Clients publish to a topic based upon their location **sensors/temperature/{country}/{city}/{street}** * Sensor on Baker Street in London would publish to **sensors/temperature/uk/london/baker_street** with a message containing the current temperature * A subscriber might collect all temperature data in london by subscribing to **sensors/temperature/uk/london/#** * Note the use of '/' to build topic hierarchy. * **#** is a wildcard matching any number of levels, **+** is a wildcard matching a single level. https://zoetrope.io/tech-blog/brief-practical-introduction-mqtt-protocol-and-its-application-iot/ --- class: compact # MQTT Protocol data:image/s3,"s3://crabby-images/6227d/6227d7d8140fd13c88a7d5755772cac2f45f5c28" alt="" MQTT is connection based. Publishers and Subscribers connect to a broker (using TCP protocol). * In this example, a publisher (client B) regularly updates the current temperature. A subscriber (client A) connects and receives the temperature. * The example illustrates the use of topics -- **temperature/roof** and an optional feature -- retained messages. Simon A. Eugster [<a href="https://creativecommons.org/licenses/by-sa/4.0">CC BY-SA 4.0</a>], <a href="https://commons.wikimedia.org/wiki/File:MQTT_protocol_example_without_QoS.svg">via Wikimedia Commons</a> --- class:compact # Publish/Subscribe Protocol Sequence data:image/s3,"s3://crabby-images/ba58d/ba58dfd4e9a8ec96a0f88478ce3459883b1d738b" alt="" data:image/s3,"s3://crabby-images/0bdbe/0bdbe8c8ba8d75f6f0a695f7de91d223c9880031" alt="" --- class: compact # Topics * MQTT Topics are structured in a hierarchy similar to folders and files in a files system (and URLs) using forward slash (/) as a delimiter data:image/s3,"s3://crabby-images/5db44/5db44f16e9f488cc5d3a38a0d7a8d3e568e69b60" alt="" * Allow creation of user friendly and self-sescriptive naming structure * Topic names are: * Case sensitive * use UTF-8 strings * Must have at least one character data:image/s3,"s3://crabby-images/0701d/0701df8de23ccc26e31140de2414e9285e3f5e4d" alt="" * $SYS is the only **default** or **standard** topic --- class: compact,small # Topics -- wildcards * Topic subscriptions can have wildcards. These enable nodes to subscribe to groups of topics (that may not yet exist !) * '+' matches anything at a given tree level * '#' matches a while sub-tree * Examples: * Subscribing to topic **house/\#** covers: * house/room1/main-light * house/room1/alarm * house/garage/main-light * house/main-door * Subscribing to **house/+/main-light** covers: * house/room1/main-light * house/room2/main-light * house/garage/main-light * but doesn't cover * house/room1/alarm --- class:compact # Quality of Service (QoS) * Messages are published with a **Quality of Service** (QoS) * QoS 0 ("at most once") -- messages are fire-and-forget. * For example, a doorbell notification might only matter if delivered immediately * QoS 1 ("at least once") -- stored by the broker until clients acknowledge delivery * May be delivered with duplicates * QoS 2 ("exactly once") -- messages have a second acknowledgement round-trip to ensure that **non-idempotent** messages are not duplicated. - --- class:compact #MQTT Publish/Subscribe Messaging * A subscription can be durable or non-durable * Durable - Once a subscription is in place a broker will forward matching messages to a subscriber: * Immediately if the subscriber is connects * Store them until the subscriber connects * Non-durable - The subscription lifetime is the period the subscriber is connected to the broker * A publication may be retained * A publisher can mark a publication as retained - The broker remembers the last known good message for a topic - When a subscriber connects, the broker gives the retained message for the topic. --- class: compact # Use on Constrained Devices * Designed for constrained networks: * Protocol compressed into bit-wise headers and variable length fields * Smallest possible packet is 2 bytes * Asynchronous bidirectional **push** delivery of messages to applications * Supports always-connected and sometimes-connected models * Provides session awareness - "last will and testament" enable applications to know when a client goes offline abnormally * Typically uses TCP * Provides multiple deterministic message deliveries quality of service (QOS) * 0 : message delivered at most once * 1 : messsage will be delivered but may be duplicated * 2 : once and only once delivery --- class: compact # Constrained Device * Designed for constrained devices * Suited for applications/devices with limited resources * 8-bit controllers * battery power * Multiple MQTT client implementations in different langauges * Tiny MQTT client -- 30KB C library, 64KB Java library --- class: compact # Use-Cases: Facebook Messenger data:image/s3,"s3://crabby-images/3381c/3381c5d68d92935bc3eab5cd22f652c7d8b6d3b4" alt="" data:image/s3,"s3://crabby-images/437de/437deae8da70155ed356401f5b3bf381c6ebc617" alt="" --- class: compact # Use-Cases: Home pacemaker monitor data:image/s3,"s3://crabby-images/8058f/8058f94127d7f85b3c6634dfaf321a30822ee759" alt="" --- class: compact # Use-Cases: Physical to Enterprise data:image/s3,"s3://crabby-images/3381c/3381c5d68d92935bc3eab5cd22f652c7d8b6d3b4" alt="" data:image/s3,"s3://crabby-images/91ef6/91ef64581f4dfee5bbec7c7675e9690bf8965be9" alt="" --- class: compact # Areas where MQTT has been used data:image/s3,"s3://crabby-images/3759f/3759f5baba3ebea02b4067ed6d34e86a1a0937ed" alt="" data:image/s3,"s3://crabby-images/17fa4/17fa4ff7dc5d970ffb947394e650527206dc9527" alt="" --- class:compact # MQTT Brokers * Widely Used * http://mosquitto.org (we'll use this in lab) * man page: man page: https://mosquitto.org/man/mosquitto-8.html * http://www.hivemq.com data:image/s3,"s3://crabby-images/6f906/6f9065cf6fa85d21ef09bfbea5d9bb1366aea02a" alt="" data:image/s3,"s3://crabby-images/14f49/14f491a889eaf3b103364200e92840a15b81f85b" alt="" --- class:compact,small # Cloud Based MQTT Brokers * https://www.cloudmqtt.com (has a free tier) * https://flespi.com/mqtt-broker (has a free tier) * AWS IoT Core : https://aws.amazon.com/iot-core/ * AZURE IoT Hub : https://azure.microsoft.com/en-us/services/iot-hub/ * Google MQTT-bridge : https://cloud.google.com/iot/docs/how-tos/mqtt-bridge data:image/s3,"s3://crabby-images/156f9/156f9d39a65403c66e52486440a421464f03029a" alt="" data:image/s3,"s3://crabby-images/79e89/79e891b40e1ffc168b0e768ca3b3024d29039f89" alt="" --- class:compact # flespi.com data:image/s3,"s3://crabby-images/7e170/7e170f39e596227db7d16771604d782c78d8558f" alt="" --- class:compact,small # Open Brokers (sandboxes) * TCP Based * https://iot.eclipse.org/getting-started/#sandboxes * Hostname: iot.eclipse.org * http://test.mosquitto.org/ * Hostname: test.mosquitto.org * https://www.hivemq.com/public-mqtt-broker/ * Hostname: broker.hivemq.com * http://www.mqtt-dashboard.com/ * Ports: * standard: 1883 * encrypted: 8883 (TLS v1.2, v1.1 or v1.0 with x509 certificates) * Websockets based: * broker.mqttdashboard.com * test.mosquitto.org * broker.hivemq.com --- # Clients for testing * Mosquitto broker comes with command-line tools for pub/sub * `mosquitto_sub -h HOSTNAME -t TOPIC [-v]` * `mosquitto_pub -h HOSTNAME -t TOPIC -m MESSAGE` * Paho C client library also includes * `paho_c_sub` * `paho_c_pub` --- class: compact,small # MQTT Protocol Example data:image/s3,"s3://crabby-images/49d39/49d39d3c11f6f4d9da9c4545f664b958812cbb6d" alt="" data:image/s3,"s3://crabby-images/347c4/347c43c73d87b3d0f0b2f64d06546289f5c86e82" alt="" An example using the tools **mosquitto-sub** and **mosquitto-pub** to send/receive topics from a local broker (localhost). * -h broker name * -t topic string (on subscriber side this can include wildcards) * -m message string (publish only) * -v (verbose subscription) --- class: compact # MQTT Clients : IOS data:image/s3,"s3://crabby-images/3759f/3759f5baba3ebea02b4067ed6d34e86a1a0937ed" alt="" data:image/s3,"s3://crabby-images/3c233/3c233ab8853f75d31bed4c01c6044a04f6cd941b" alt="" --- class: compact # MQTT Clients : Android data:image/s3,"s3://crabby-images/3759f/3759f5baba3ebea02b4067ed6d34e86a1a0937ed" alt="" data:image/s3,"s3://crabby-images/67374/6737464f1d964a982371bf2b97ea744601be590b" alt="" --- class: compact # MQTT Client: Node-Red data:image/s3,"s3://crabby-images/fb0e2/fb0e221f09080bf1e60c4b55b49428f913150674" alt="" data:image/s3,"s3://crabby-images/a4dfc/a4dfc3ba9438e49f5cfcf24a4be3165b860b50cd" alt="" --- class: compact,very-small-code,hljs-tomorrow-night-eighties,col-2 # MQTT Client Library : Mosquitto (C only) ```c void on_message(struct mosquitto *m, void *user_data, const struct mosquitto_message *msg) { fprintf(stderr, "lights at %s are %s\n", msg->topic, (char*)msg->payload); } int main(int argc, char **argv) { struct mosquitto *client; int ret; mosquitto_lib_init(); client = mosquitto_new("client-id", true, NULL); if (!client) err(1, "mosquitto failed"); ret = mosquitto_connect(client, "127.0.0.1", 1883, 60); if (ret != MOSQ_ERR_SUCCESS) err(1, "mosquitto failed"); ret = mosquitto_subscribe(client, NULL, "switches/+/status", 0); if (ret != MOSQ_ERR_SUCCESS) err(1, "mosquitto failed"); mosquitto_message_callback_set(client, on_message); while (MOSQ_ERR_SUCCESS == mosquitto_loop(client, -1, 1)); return 0; } ``` --- class: compact # MQTT Client Library : Paho * Open source MQTT client library https://www.eclipse.org/paho/ data:image/s3,"s3://crabby-images/3759f/3759f5baba3ebea02b4067ed6d34e86a1a0937ed" alt="" data:image/s3,"s3://crabby-images/d7bd6/d7bd6a331ad1bb66eafe26142e0ea3cb11f0158d" alt="" --- class: compact,very-small-code,hljs-tomorrow-night-eighties,col-2 # Paho Python Example ```Python import paho.mqtt.client as mqtt # The callback for when the client receives a CONNACK # response from the server. def on_connect(client, userdata, flags, rc): print("Connected with result code "+str(rc)) # Subscribing in on_connect() means that if we # lose the connection and # reconnect then subscriptions will be renewed. client.subscribe("$SYS/#") # The callback for when a PUBLISH message is # received from the server. def on_message(client, userdata, msg): print(msg.topic+" "+str(msg.payload)) client = mqtt.Client() client.on_connect = on_connect client.on_message = on_message client.connect("localhost", 1883, 60) # Blocking call that processes network traffic, # dispatches callbacks and handles reconnecting. # Other loop*() functions are available that give a # threaded interface and a manual interface. client.loop_forever() ``` --- class: compact # Comments on Paho Python * Note the use of "callbacks" -- this is a common structuring mechanism for servers * Also supported in Paho asynchronous C library * There are many more options * SSL/TLS support for security * Protocol version (we're using 3.11 in this class) * Support for error logging * Threaded architecture for main loop -- enables main loop to do other things https://www.eclipse.org/paho/clients/python/docs/ --- class: compact # MQTT Protocol Specification ([link](http://docs.oasis-open.org/mqtt/mqtt/v3.1.1/os/mqtt-v3.1.1-os.pdf)) data:image/s3,"s3://crabby-images/fb0e2/fb0e221f09080bf1e60c4b55b49428f913150674" alt="" data:image/s3,"s3://crabby-images/5029c/5029c127e344cd395e7621a1584a08113cb01cf6" alt="" --- class: compact # MQTT Protocol (from specification) * Data representation * Bits in a byte numbered 7 down to 0. 7 is most significant * Integers 16-bit, "big endian" * UTF-8 encoded strings with example (A𪛔) (41 F0 AA 9B 94) data:image/s3,"s3://crabby-images/2014e/2014eec921035384a108a4ce7890c5ac9eacdd75" alt="" data:image/s3,"s3://crabby-images/3381c/3381c5d68d92935bc3eab5cd22f652c7d8b6d3b4" alt="" data:image/s3,"s3://crabby-images/fb70a/fb70a691d5f6e3eafec3e17582104e29d5d50733" alt="" --- class: compact # MQTT Malformed Data data:image/s3,"s3://crabby-images/fb0e2/fb0e221f09080bf1e60c4b55b49428f913150674" alt="" data:image/s3,"s3://crabby-images/7e646/7e6468bb177a1eed04dcd0131a7e438f604a3998" alt="" --- class: compact,small # MQTT Protocol (control packet) data:image/s3,"s3://crabby-images/fb0e2/fb0e221f09080bf1e60c4b55b49428f913150674" alt="" data:image/s3,"s3://crabby-images/a89f9/a89f933f0213979778df4d2d2bc039d9c41e81b7" alt="" data:image/s3,"s3://crabby-images/fb0e2/fb0e221f09080bf1e60c4b55b49428f913150674" alt="" data:image/s3,"s3://crabby-images/8780f/8780fce67ff6ba3ca2fda27d1f6c77948e7e6cce" alt="" | Name | Value | Direction | Description | | ---- | ----- | ---------- | ---------- | | CONNECT | 1 | Client to Server | Connect | | CONNACT | 2 | Server to Client | Ack Conn| | PUBLISH | 3 | Client to Server | Publish message | | PUBACK | 4 | Both | Ack Pub | | ... | ... | ... | ... | | SUBSCRIBE | 8 | Client to Server | Subscribe | | SUBACK | 9 | Server to Client | Ack Sub | | ... | ... | ... | ... | | DISCONNECT | 14 | Client to Server | Disconnect | --- class: compact # MQTT Protocol Publish Message data:image/s3,"s3://crabby-images/fb0e2/fb0e221f09080bf1e60c4b55b49428f913150674" alt="" data:image/s3,"s3://crabby-images/e9f36/e9f362e9f9cb9c55a8e65b123ebb89ca65842f11" alt="" * QoS, retain encoded in flags bits * Remaining length -- length of a variable (topic) header + payload * Topic -- utf-8 encoded string (length + utf-8), 16-bit packet identifier * Payload -- uninterpreted bytes. * Topic string limited to 65536 bytes (16-bit length) * Payload is limited to 268,535,456 bytes --- class: compact # MQTT Publish Packet Variable Header (Topic) data:image/s3,"s3://crabby-images/fb0e2/fb0e221f09080bf1e60c4b55b49428f913150674" alt="" data:image/s3,"s3://crabby-images/fffa5/fffa540029364f2d2f0d00f1a0524e6a02c50787" alt="" --- class:compact,small-code,hljs-tomorrow-night-eighties, # MQTT Remaining Length Field Remaining length field is variable length encoded - bit 7 used a "continuation" bit data:image/s3,"s3://crabby-images/702b1/702b16dd9f9efa29ae3a974065230419ea064900" alt="" ```c do { encodedByte = X MOD 128 X = X DIV 128 // if there are more data to encode, //set the top bit of this byte if ( X > 0 ) { encodedByte = encodedByte OR 128 } 'output' encodedByte } while ( X > 0 ) ``` --- class: compact # MQTT Operational Behavior * Storing State -- client and server must store session state for duration of session * QoS 1 (At least once delivery) -- receiver must respond with PUBACK packet, packet identifier ignored * QoS 2 (Exactly once) -- receiver responds and treats new packets with same identifier as duplicates --- class: compact # MQTT Topics (partial) data:image/s3,"s3://crabby-images/fb0e2/fb0e221f09080bf1e60c4b55b49428f913150674" alt="" data:image/s3,"s3://crabby-images/1296f/1296fc4e22226bbc34c3db001a864d82f1384673" alt="" --- class: compact # Designing Topics The MQTT protocol leaves important issues undefined * Topics -- The protocol defines topics as a hierarchy of names, but gives no guidance about choosing topics. For a small network this might not matter, but it doesn't scale well * Message Bodies -- these are just byte sequences with *no* syntax or semantics. * syntax -- json ?, protobuf ? + schemas * semantics -- left as exercise to user https://d1.awsstatic.com/whitepapers/Designing_MQTT_Topics_for_AWS_IoT_Core.pdf --- class: compact # MQTT Message Patterns : Point-to-Point data:image/s3,"s3://crabby-images/fb0e2/fb0e221f09080bf1e60c4b55b49428f913150674" alt="" data:image/s3,"s3://crabby-images/119b6/119b69d43b263cd417a502ae27f58eb01f592eb0" alt="" * Basic building block -- two things use a single topic as "point-to-point" channel --- class: compact # MQTT Message Patterns: One to many with point-to-point messages Point-to-point messages be used a one-to-many by using a (slightly) different topic for each endpoint data:image/s3,"s3://crabby-images/fb0e2/fb0e221f09080bf1e60c4b55b49428f913150674" alt="" data:image/s3,"s3://crabby-images/92824/92824e46fffe03a66a79bf4c6789437f7eb500d0" alt="" --- class: compact # MQTT Message Patterns: broadcast data:image/s3,"s3://crabby-images/3381c/3381c5d68d92935bc3eab5cd22f652c7d8b6d3b4" alt="" data:image/s3,"s3://crabby-images/1c9eb/1c9eb03696a36c3a19e4c3f8af49f8c18fbfec81" alt="" --- class: compact # MQTT Message Patterns: Many to one data:image/s3,"s3://crabby-images/3381c/3381c5d68d92935bc3eab5cd22f652c7d8b6d3b4" alt="" data:image/s3,"s3://crabby-images/a53c7/a53c79b316cbe4fcd1b716519d2cf56ffe557c50" alt="" --- class: compact # MQTT Topic Best Practices * Use only lowercase letters, numbers, dashes: MQTT topics are case sensitive -- mixing cases can lead to issues * Topic level structures should follow general to specific pattern * Platform/building/location/thing data:image/s3,"s3://crabby-images/6f906/6f9065cf6fa85d21ef09bfbea5d9bb1366aea02a" alt="" data:image/s3,"s3://crabby-images/957c7/957c756b13e5566f7463c7df20dad97183004ff0" alt="" * Include relevant routing information in MQTT topic * Prefix topics to distinguish data and control --- class: compact # MQTT Topics for data Data * prefix (dt) * IoT application * context (one or more layers) * name of thing transmitting data * type of data dt/<application>/<context>/<thing-name>/<dt-type> --- class: compact # MQTT Topics for Control Control * prefix (cmd) * IoT application * context * destination thing * request type; request parameters should go in message cmd/<application>/<context>/<destination-id>/<req-type> Typically, there will be a corresponding response cmd/<application>/<context>/<destination-id>/<res-type> --- class: compact # MQTT Documentation * MQTT Community Wiki * https://github.com/mqtt/mqtt.github.io/wiki * MQTT Tutorial * https://www.hivemq.com/mqtt-essentials/ * MQTT 3.1.1 Specification * http://docs.oasis-open.org/mqtt/mqtt/v3.1.1/mqtt-v3.1.1.html * MQTT Standards body * https://www.oasis-open.org/committees/tc_home.php?wg_abbrev=mqtt --- class: compact # Summary * Cover Photo: Mentor Automotive * Acknowledgements -- slide content and images drawn from: * https://github.com/moleike/mqtt-slides * https://www.oasis-open.org/committees/download.php/49205/MQTT-OASIS-Webinar.pdf * http://wireless.ictp.it/school_2019/slides/MQTT_v2.pdf Pietro Manzoni Intro * https://d1.awsstatic.com/whitepapers/Designing_MQTT_Topics_for_AWS_IoT_Core.pdf