Compare commits
88 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
84d30b6343 | ||
|
|
f38cc536b8 | ||
|
|
f498995aee | ||
|
|
8e44e610b3 | ||
|
|
e66eb7aa6a | ||
|
|
4b4e72905e | ||
|
|
a717ecd66d | ||
|
|
aa62bcaf02 | ||
|
|
90435b1260 | ||
|
|
ac391a49a4 | ||
|
|
30e75b82e0 | ||
|
|
65e22c85c6 | ||
|
|
bd566945cc | ||
|
|
66cb80ff3f | ||
|
|
ef4a0263a6 | ||
|
|
3135d9a28c | ||
|
|
6b9d764c23 | ||
|
|
42d89cdf06 | ||
|
|
67ce97e523 | ||
|
|
25602ec2f5 | ||
|
|
fd305d7b48 | ||
|
|
6339be8044 | ||
|
|
d3bf379d19 | ||
|
|
3c77f7cafd | ||
|
|
ff3ff6e80e | ||
|
|
c4cc36a88f | ||
|
|
14882fed22 | ||
|
|
ead3ae90e9 | ||
|
|
5d07294fde | ||
|
|
088071d17f | ||
|
|
e41452edf0 | ||
|
|
f8a2e35dd9 | ||
|
|
43dbea1f17 | ||
|
|
02496bef73 | ||
|
|
e4ad27c805 | ||
|
|
37fb46ec3b | ||
|
|
245e74666e | ||
|
|
294657f2ca | ||
|
|
bf84e29831 | ||
|
|
0c7c830a26 | ||
|
|
6e601228e6 | ||
|
|
46798ff0de | ||
|
|
45fedf84c9 | ||
|
|
f9c8dca1e5 | ||
|
|
7e1586c0b5 | ||
|
|
123c5a8fa5 | ||
|
|
21fb01848d | ||
|
|
66b1e71ee2 | ||
|
|
e5115087ea | ||
|
|
cc004875b5 | ||
|
|
58ae2436d3 | ||
|
|
cda94368a7 | ||
|
|
0126a39327 | ||
|
|
2587371457 | ||
|
|
a880a1ff2f | ||
|
|
143d57db2d | ||
|
|
2a8dbd09c5 | ||
|
|
00333ed805 | ||
|
|
9228408f57 | ||
|
|
1653b2e386 | ||
|
|
775fbc14ee | ||
|
|
c21b7b63fb | ||
|
|
e45af112c2 | ||
|
|
f9c539ff6a | ||
|
|
7dc23d322c | ||
|
|
4fb632ce3d | ||
|
|
c9130c7a24 | ||
|
|
31c51aeaff | ||
|
|
2eeda4ecab | ||
|
|
787cb77a26 | ||
|
|
d324a913ec | ||
|
|
ac1eedd72a | ||
|
|
0900e799a9 | ||
|
|
2086c7f0e7 | ||
|
|
72382bf351 | ||
|
|
3fec7b87a8 | ||
|
|
8641627742 | ||
|
|
3b2460572b | ||
|
|
a0435b2cfb | ||
|
|
bda041417d | ||
|
|
baffda8a6d | ||
|
|
09e3a3e45f | ||
|
|
f17ece3376 | ||
|
|
0db07df27b | ||
|
|
292592c3dd | ||
|
|
1f267c135b | ||
|
|
2b92833ea5 | ||
|
|
42fc054c94 |
6
.github/workflows/aunit.yml
vendored
6
.github/workflows/aunit.yml
vendored
@@ -9,7 +9,7 @@ on: [push]
|
|||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
|
|
||||||
runs-on: ubuntu-18.04
|
runs-on: ubuntu-20.04
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v2
|
- uses: actions/checkout@v2
|
||||||
@@ -18,9 +18,9 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
cd ..
|
cd ..
|
||||||
git clone https://github.com/hsaturn/TinyConsole
|
git clone https://github.com/hsaturn/TinyConsole
|
||||||
git clone https://github.com/bxparks/EpoxyDuino
|
git clone https://github.com/hsaturn/EpoxyDuino
|
||||||
git clone https://github.com/bxparks/AceRoutine
|
git clone https://github.com/bxparks/AceRoutine
|
||||||
git clone https://github.com/bxparks/AUnit
|
git clone https://github.com/hsaturn/AUnit
|
||||||
git clone https://github.com/bxparks/AceCommon
|
git clone https://github.com/bxparks/AceCommon
|
||||||
git clone https://github.com/hsaturn/EspMock
|
git clone https://github.com/hsaturn/EspMock
|
||||||
- name: Verify tests
|
- name: Verify tests
|
||||||
|
|||||||
2
.github/workflows/ci.yml
vendored
2
.github/workflows/ci.yml
vendored
@@ -1,6 +1,6 @@
|
|||||||
name: "CI"
|
name: "CI"
|
||||||
on:
|
on:
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
ci:
|
ci:
|
||||||
runs-on: ubuntu-20.04
|
runs-on: ubuntu-20.04
|
||||||
|
|||||||
21
README.md
21
README.md
@@ -10,8 +10,22 @@
|
|||||||
|
|
||||||
TinyMqtt is a small, fast and capable Mqtt Broker and Client for Esp8266 / Esp32 / Esp WROOM
|
TinyMqtt is a small, fast and capable Mqtt Broker and Client for Esp8266 / Esp32 / Esp WROOM
|
||||||
|
|
||||||
|
### Statuses of all unit tests of TinyMqtt and its dependencies
|
||||||
|
|
||||||
|
| Project | Unit tests result |
|
||||||
|
| ----------- | ------------ |
|
||||||
|
| TinyMqtt | [](https://github.com/hsaturn/TinyMqtt/actions/workflows/aunit.yml) |
|
||||||
|
| Dependencies ||
|
||||||
|
| TinyConsole | [](https://github.com/hsaturn/TinyConsole/actions/workflows/aunit.yml) |
|
||||||
|
| EpoxyDuino | [](https://github.com/hsaturn/EpoxyDuino/actions/workflows/aunit_tests.yml) |
|
||||||
|
| EspMock | [](https://github.com/hsaturn/EspMock/actions/workflows/aunit.yml) |
|
||||||
|
| AUnit | [](https://github.com/hsaturn/AUnit/actions/workflows/aunit_tests.yml) |
|
||||||
|
| AceRoutine | [](https://github.com/bxparks/AceRoutine/actions/workflows/aunit_tests.yml) |
|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
|
|
||||||
|
- Supports retained messages (not activated by default)
|
||||||
|
- Async Wifi compatible (me-no-dev/ESPAsyncTCP@^1.2.2)
|
||||||
- Very fast broker I saw it re-sent 1000 topics per second for two
|
- Very fast broker I saw it re-sent 1000 topics per second for two
|
||||||
clients that had subscribed (payload ~15 bytes ESP8266). No topic lost.
|
clients that had subscribed (payload ~15 bytes ESP8266). No topic lost.
|
||||||
The max I've seen was 2k msg/s (1 client 1 subscription)
|
The max I've seen was 2k msg/s (1 client 1 subscription)
|
||||||
@@ -51,6 +65,13 @@ TinyMqtt is a small, fast and capable Mqtt Broker and Client for Esp8266 / Esp32
|
|||||||
- tinymqtt-test : This is a complex sketch with a terminal console
|
- tinymqtt-test : This is a complex sketch with a terminal console
|
||||||
that allows to add clients publish, connect etc with interpreted commands.
|
that allows to add clients publish, connect etc with interpreted commands.
|
||||||
|
|
||||||
|
## Retained messages
|
||||||
|
|
||||||
|
Qos 1 is not supported, but retained messages are. So a new subscription is able to send old messages.
|
||||||
|
This feature is disabled by default.
|
||||||
|
The default retain parameter of MqttBroker::MqttBroker takes an optional (0 by default) number of retained messages.
|
||||||
|
MqttBroker::retain(n) will also make the broker store n messages at max.
|
||||||
|
|
||||||
## Standalone mode (zeroconf)
|
## Standalone mode (zeroconf)
|
||||||
-> The zeroconf mode is not yet implemented
|
-> The zeroconf mode is not yet implemented
|
||||||
zeroconf clients to connect to broker on local network.
|
zeroconf clients to connect to broker on local network.
|
||||||
|
|||||||
74
bump_version.sh
Executable file
74
bump_version.sh
Executable file
@@ -0,0 +1,74 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
current_version=$(git describe --tags --abbrev=0)
|
||||||
|
|
||||||
|
function error
|
||||||
|
{
|
||||||
|
echo
|
||||||
|
echo "ERROR *** $1"
|
||||||
|
}
|
||||||
|
if [ "$1" == "-d" ]; then
|
||||||
|
do=0
|
||||||
|
shift
|
||||||
|
else
|
||||||
|
do=1
|
||||||
|
fi
|
||||||
|
if [ "$1" == "" ]; then
|
||||||
|
echo
|
||||||
|
echo "Syntax: $0 [-d] {new_version} [commit message]"
|
||||||
|
echo
|
||||||
|
echo " -d : dry run, generate json and update properties but do not run git commands"
|
||||||
|
echo ""
|
||||||
|
echo " Current version: $current_version"
|
||||||
|
echo
|
||||||
|
else
|
||||||
|
tm=$(git status --porcelain -- src/TinyMqtt.h | wc -l)
|
||||||
|
echo "Current version: ($current_version)"
|
||||||
|
echo "New version : ($1)"
|
||||||
|
echo "Take info from : library.properties"
|
||||||
|
if [ "$tm" == "1" ]; then
|
||||||
|
error "You cannot bump version if TinyMqtt.h is modified"
|
||||||
|
exit
|
||||||
|
fi
|
||||||
|
echo -n "Do you want to proceed ? "
|
||||||
|
read a
|
||||||
|
if [ "$a" == "y" ]; then
|
||||||
|
echo "Doing this..."
|
||||||
|
grep $current_version library.properties
|
||||||
|
if [ "$?" == "0" ]; then
|
||||||
|
sed -i "s/$current_version/$1/" library.properties
|
||||||
|
sed -i "s/#define TINY_MQTT_REVISION/#define TINY_MQTT_REVISION \"$1\"/" src/TinyMqtt.h
|
||||||
|
|
||||||
|
cp library.json.skeleton library.json
|
||||||
|
while ifs= read -r line; do
|
||||||
|
name=$(echo "$line" | sed "s/=.*//g")
|
||||||
|
value=$(echo "$line" | cut -d= -f 2 | sed 's/"//g')
|
||||||
|
echo " Replacing $name in json"
|
||||||
|
if [ "$name" == "depends" ]; then
|
||||||
|
depends=$(echo "$value" | sed "s/,/ /g")
|
||||||
|
echo " Depends=$depends"
|
||||||
|
fi
|
||||||
|
sed -i "s@#$name@$value@g" library.json
|
||||||
|
done < library.properties
|
||||||
|
deps=""
|
||||||
|
for depend in $depends; do
|
||||||
|
if [ "$deps" != "" ]; then
|
||||||
|
deps="$deps, "
|
||||||
|
fi
|
||||||
|
deps="$deps'$depend' : '*'"
|
||||||
|
done
|
||||||
|
sed -i "s@#dependencies@$deps@g" library.json
|
||||||
|
sed -i "s/'/\"/g" library.json
|
||||||
|
if [ "$do" == "1" ]; then
|
||||||
|
echo "Pushing all"
|
||||||
|
git add library.properties
|
||||||
|
git add library.json
|
||||||
|
git commit -m "Release $1 $2"
|
||||||
|
git tag $1
|
||||||
|
git push
|
||||||
|
git push --tags
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
error "Current version does not match library.property version, aborting"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
fi
|
||||||
@@ -0,0 +1,111 @@
|
|||||||
|
#include <SPI.h>
|
||||||
|
#include <Ethernet.h>
|
||||||
|
|
||||||
|
#include "TinyMqtt.h" // https://github.com/hsaturn/TinyMqtt
|
||||||
|
|
||||||
|
#define WIZRST D3 //Specify pin to use for reseting W5500
|
||||||
|
#define WIZCS D1 //Specify the pin for SPI CS
|
||||||
|
|
||||||
|
#define PORT 1883
|
||||||
|
MqttBroker broker(PORT);
|
||||||
|
unsigned long Time;
|
||||||
|
unsigned long freeRam;
|
||||||
|
|
||||||
|
/** Basic Mqtt Broker
|
||||||
|
*
|
||||||
|
* +-----------------------------+
|
||||||
|
* | ESP |
|
||||||
|
* | +--------+ |
|
||||||
|
* | | broker | | 1883 <--- External client/s
|
||||||
|
* | +--------+ |
|
||||||
|
* | |
|
||||||
|
* +-----------------------------+
|
||||||
|
*
|
||||||
|
* Your ESP will become a MqttBroker.
|
||||||
|
* You can test it with any client such as mqtt-spy for example
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#if defined(USE_ETHERNET)
|
||||||
|
|
||||||
|
byte mac[] = { 0x00, 0xAA, 0xBB, 0xE0, 0x01, 0x25 }; //MAC Address
|
||||||
|
//IPAddress ip = (192,168,0,88); //Fixed IP Address
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
const char* ssid = "xxxxxxx";
|
||||||
|
const char* password = "xxxxxxxx";
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
void WizReset() {
|
||||||
|
Serial.print("Resetting Wiz W5500 Ethernet Board... ");
|
||||||
|
pinMode(WIZRST, OUTPUT);
|
||||||
|
digitalWrite(WIZRST, HIGH);
|
||||||
|
delay(250);
|
||||||
|
digitalWrite(WIZRST, LOW);
|
||||||
|
delay(50);
|
||||||
|
digitalWrite(WIZRST, HIGH);
|
||||||
|
delay(350);
|
||||||
|
Serial.println("Done.");
|
||||||
|
}
|
||||||
|
|
||||||
|
void setup()
|
||||||
|
{
|
||||||
|
Serial.begin(115200);
|
||||||
|
|
||||||
|
#if defined(USE_ETHERNET)
|
||||||
|
|
||||||
|
Ethernet.init(WIZCS); // SPI CS Pin
|
||||||
|
SPI.begin();
|
||||||
|
WizReset();
|
||||||
|
Console << TinyConsole::green << "Starting Ethernet...." << endl;
|
||||||
|
Ethernet.begin(mac); //Connect using DHCP
|
||||||
|
//Ethernet.begin(mac,ip); //Connect using Fixed IP
|
||||||
|
if (Ethernet.hardwareStatus() == EthernetNoHardware) {
|
||||||
|
Console << TinyConsole::red << "Ethernet shield was not found. Sorry, can't run without hardware. :(" << endl;
|
||||||
|
} else
|
||||||
|
if (Ethernet.linkStatus() == LinkOFF) {
|
||||||
|
Console << TinyConsole::red << "Ethernet cable is not connected." << endl;
|
||||||
|
}
|
||||||
|
Console << TinyConsole::green << "Local Ethernet IP address: " << Ethernet.localIP() << endl;
|
||||||
|
broker.begin();
|
||||||
|
Console << "Broker ready (eth) : " << Ethernet.localIP() << " on port " << PORT << endl;
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
if (strlen(ssid)==0)
|
||||||
|
Console << TinyConsole::red << "****** PLEASE MODIFY ssid/password *************" << endl;
|
||||||
|
WiFi.disconnect(); //Remove previous SSID & Password
|
||||||
|
WiFi.begin(ssid, password); // Connect to the network
|
||||||
|
Console << TinyConsole::white << "WiFi Connecting to " << ssid << " ...";
|
||||||
|
int i = 0;
|
||||||
|
while (WiFi.status() != WL_CONNECTED) { // Wait for the Wi-Fi to connect
|
||||||
|
delay(1000);
|
||||||
|
Console << TinyConsole::white << ".";
|
||||||
|
i++;
|
||||||
|
if (i > 20) break;
|
||||||
|
}
|
||||||
|
Console << endl << endl;
|
||||||
|
Console << TinyConsole::green << "Connected to " << ssid << " IP address: " << WiFi.localIP() << endl;
|
||||||
|
broker.begin();
|
||||||
|
Console << "Broker ready (wifi) : " << WiFi.localIP() << " on port " << PORT << endl;
|
||||||
|
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void loop()
|
||||||
|
{
|
||||||
|
broker.loop();
|
||||||
|
if(millis()-Time>10000) {
|
||||||
|
Time=millis();
|
||||||
|
if(ESP.getFreeHeap()!=freeRam)
|
||||||
|
{
|
||||||
|
freeRam=ESP.getFreeHeap();
|
||||||
|
Serial.print("RAM:");
|
||||||
|
Serial.println(freeRam);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
@@ -0,0 +1,117 @@
|
|||||||
|
#include <SPI.h>
|
||||||
|
#include <Ethernet.h>
|
||||||
|
|
||||||
|
#include "TinyMqtt.h" // https://github.com/hsaturn/TinyMqtt
|
||||||
|
|
||||||
|
#define WIZRST 20 //Specify pin to use for reseting W5500 = GPIO20
|
||||||
|
#define WIZCS 17 //Specify the pin for SPI CS = GPIO17 = D5
|
||||||
|
|
||||||
|
#define PORT 1883
|
||||||
|
MqttBroker broker(PORT);
|
||||||
|
unsigned long Time;
|
||||||
|
unsigned long freeRam;
|
||||||
|
|
||||||
|
/** Basic Mqtt Broker
|
||||||
|
*
|
||||||
|
* +-----------------------------+
|
||||||
|
* | ESP |
|
||||||
|
* | +--------+ |
|
||||||
|
* | | broker | | 1883 <--- External client/s
|
||||||
|
* | +--------+ |
|
||||||
|
* | |
|
||||||
|
* +-----------------------------+
|
||||||
|
*
|
||||||
|
* Your ESP will become a MqttBroker.
|
||||||
|
* You can test it with any client such as mqtt-spy for example
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
#if defined(USE_ETHERNET)
|
||||||
|
|
||||||
|
byte mac[] = { 0x00, 0xAA, 0xBB, 0xE0, 0x01, 0x25 }; //MAC Address
|
||||||
|
//IPAddress ip = (192,168,0,88); //Fixed IP Address
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
const char* ssid = "xxxxxxx";
|
||||||
|
const char* password = "xxxxxxxx";
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
void WizReset() {
|
||||||
|
Serial.print("Resetting Wiz W5500 Ethernet Board... ");
|
||||||
|
pinMode(WIZRST, OUTPUT);
|
||||||
|
digitalWrite(WIZRST, HIGH);
|
||||||
|
delay(250);
|
||||||
|
digitalWrite(WIZRST, LOW);
|
||||||
|
delay(50);
|
||||||
|
digitalWrite(WIZRST, HIGH);
|
||||||
|
delay(350);
|
||||||
|
Serial.println("Done.");
|
||||||
|
}
|
||||||
|
|
||||||
|
void setup()
|
||||||
|
{
|
||||||
|
Serial.begin(115200);
|
||||||
|
|
||||||
|
#if defined(USE_ETHERNET)
|
||||||
|
|
||||||
|
Ethernet.init(WIZCS);
|
||||||
|
SPI.begin();
|
||||||
|
WizReset();
|
||||||
|
Console << TinyConsole::green << "Starting Ethernet...." << endl;
|
||||||
|
Ethernet.begin(mac); //Connect using DHCP
|
||||||
|
//Ethernet.begin(mac,ip); //Connect using Fixed IP
|
||||||
|
if (Ethernet.hardwareStatus() == EthernetNoHardware) {
|
||||||
|
Console << TinyConsole::red << "Ethernet shield was not found. Sorry, can't run without hardware. :(" << endl;
|
||||||
|
} else
|
||||||
|
if (Ethernet.linkStatus() == LinkOFF) {
|
||||||
|
Console << TinyConsole::red << "Ethernet cable is not connected." << endl;
|
||||||
|
}
|
||||||
|
Console << TinyConsole::green << "Local Ethernet IP address: " << Ethernet.localIP() << endl;
|
||||||
|
broker.begin();
|
||||||
|
Console << "Broker ready (eth) : " << Ethernet.localIP() << " on port " << PORT << endl;
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
if (strlen(ssid)==0)
|
||||||
|
Console << TinyConsole::red << "****** PLEASE MODIFY ssid/password *************" << endl;
|
||||||
|
WiFi.disconnect(); //Remove previous SSID & Password
|
||||||
|
WiFi.begin(ssid, password); // Connect to the network
|
||||||
|
Console << TinyConsole::white << "WiFi Connecting to " << ssid << " ...";
|
||||||
|
int i = 0;
|
||||||
|
while (WiFi.status() != WL_CONNECTED) { // Wait for the Wi-Fi to connect
|
||||||
|
delay(1000);
|
||||||
|
Console << TinyConsole::white << ".";
|
||||||
|
i++;
|
||||||
|
if (i > 20) break;
|
||||||
|
}
|
||||||
|
Console << endl << endl;
|
||||||
|
Console << TinyConsole::green << "Connected to " << ssid << " IP address: " << WiFi.localIP() << endl;
|
||||||
|
broker.begin();
|
||||||
|
Console << "Broker ready (wifi) : " << WiFi.localIP() << " on port " << PORT << endl;
|
||||||
|
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void loop()
|
||||||
|
{
|
||||||
|
broker.loop();
|
||||||
|
if(millis()-Time>10000) {
|
||||||
|
Time=millis();
|
||||||
|
#ifdef ARDUINO_ARCH_RP2040
|
||||||
|
if(rp2040.getFreeHeap()!=freeRam) {
|
||||||
|
freeRam=rp2040.getFreeHeap();
|
||||||
|
#else
|
||||||
|
if(ESP.getFreeHeap()!=freeRam) {
|
||||||
|
freeRam=ESP.getFreeHeap();
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
Serial.print("RAM:");
|
||||||
|
Serial.println(freeRam);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
5
examples/W5500/client-with-W5500-RP2040/readme
Normal file
5
examples/W5500/client-with-W5500-RP2040/readme
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
In TinyString.h the operator
|
||||||
|
|
||||||
|
TinyString& operator +=(int);
|
||||||
|
|
||||||
|
may need to be added.
|
||||||
@@ -1,12 +1,12 @@
|
|||||||
#include <TinyMqtt.h> // https://github.com/hsaturn/TinyMqtt
|
#include <TinyMqtt.h> // https://github.com/hsaturn/TinyMqtt
|
||||||
#include <MqttClassBinder.h>
|
#include <MqttClassBinder.h>
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Example on how to bind a class:onPublish function
|
* Example on how to bind a class:onPublish function
|
||||||
*
|
*
|
||||||
* Local broker that accept connections and two local clients
|
* Local broker that accept connections and two local clients
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
* +-----------------------------+
|
* +-----------------------------+
|
||||||
* | ESP |
|
* | ESP |
|
||||||
* | +--------+ | 1883 <--- External client/s
|
* | +--------+ | 1883 <--- External client/s
|
||||||
@@ -15,14 +15,14 @@
|
|||||||
* | | ^ |
|
* | | ^ |
|
||||||
* | | | |
|
* | | | |
|
||||||
* | | | | -----
|
* | | | | -----
|
||||||
* | v v | ---
|
* | v v | ---
|
||||||
* | +----------+ +----------+ | -
|
* | +----------+ +----------+ | -
|
||||||
* | | internal | | internal | +-------* Wifi
|
* | | internal | | internal | +-------* Wifi
|
||||||
* | | client | | client | |
|
* | | client | | client | |
|
||||||
* | +----------+ +----------+ |
|
* | +----------+ +----------+ |
|
||||||
* | |
|
* | |
|
||||||
* +-----------------------------+
|
* +-----------------------------+
|
||||||
*
|
*
|
||||||
* pros - Reduces internal latency (when publish is received by the same ESP)
|
* pros - Reduces internal latency (when publish is received by the same ESP)
|
||||||
* - Reduces wifi traffic
|
* - Reduces wifi traffic
|
||||||
* - No need to have an external broker
|
* - No need to have an external broker
|
||||||
@@ -30,7 +30,7 @@
|
|||||||
* - accepts external clients
|
* - accepts external clients
|
||||||
* - MqttClassBinder allows to mix together many mqtt sources
|
* - MqttClassBinder allows to mix together many mqtt sources
|
||||||
*
|
*
|
||||||
* cons - Takes more memory (48 more bytes for the one MqttClassBinder<Class>
|
* cons - Takes more memory (24 more bytes for the one MqttClassBinder<Class>
|
||||||
* - a bit hard to understand
|
* - a bit hard to understand
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
@@ -50,7 +50,7 @@ MqttClient mqtt_sender(&broker);
|
|||||||
class MqttReceiver: public MqttClassBinder<MqttReceiver>
|
class MqttReceiver: public MqttClassBinder<MqttReceiver>
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
|
||||||
void onPublish(const MqttClient* source, const Topic& topic, const char* payload, size_t /* length */)
|
void onPublish(const MqttClient* source, const Topic& topic, const char* payload, size_t /* length */)
|
||||||
{
|
{
|
||||||
Serial
|
Serial
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
#include <TinyMqtt.h> // https://github.com/hsaturn/TinyMqtt
|
#include <TinyMqtt.h> // https://github.com/hsaturn/TinyMqtt
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Local broker that accept connections and two local clients
|
* Local broker that accept connections and two local clients
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
* +-----------------------------+
|
* +-----------------------------+
|
||||||
* | ESP |
|
* | ESP |
|
||||||
* | +--------+ | 1883 <--- External client/s
|
* | +--------+ | 1883 <--- External client/s
|
||||||
@@ -12,14 +12,14 @@
|
|||||||
* | | ^ |
|
* | | ^ |
|
||||||
* | | | |
|
* | | | |
|
||||||
* | | | | -----
|
* | | | | -----
|
||||||
* | v v | ---
|
* | v v | ---
|
||||||
* | +----------+ +----------+ | -
|
* | +----------+ +----------+ | -
|
||||||
* | | internal | | internal | +-------* Wifi
|
* | | internal | | internal | +-------* Wifi
|
||||||
* | | client | | client | |
|
* | | client | | client | |
|
||||||
* | +----------+ +----------+ |
|
* | +----------+ +----------+ |
|
||||||
* | |
|
* | |
|
||||||
* +-----------------------------+
|
* +-----------------------------+
|
||||||
*
|
*
|
||||||
* pros - Reduces internal latency (when publish is received by the same ESP)
|
* pros - Reduces internal latency (when publish is received by the same ESP)
|
||||||
* - Reduces wifi traffic
|
* - Reduces wifi traffic
|
||||||
* - No need to have an external broker
|
* - No need to have an external broker
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
#include <TinyMqtt.h> // https://github.com/hsaturn/TinyMqtt
|
#include <TinyMqtt.h> // https://github.com/hsaturn/TinyMqtt
|
||||||
|
|
||||||
/** TinyMQTT allows a disconnected mode:
|
/** TinyMQTT allows a disconnected mode:
|
||||||
*
|
*
|
||||||
* +-----------------------------+
|
* +-----------------------------+
|
||||||
* | ESP |
|
* | ESP |
|
||||||
* | +--------+ |
|
* | +--------+ |
|
||||||
@@ -12,8 +12,8 @@
|
|||||||
* | v v |
|
* | v v |
|
||||||
* | +----------+ +----------+ |
|
* | +----------+ +----------+ |
|
||||||
* | | internal | | internal | |
|
* | | internal | | internal | |
|
||||||
* | | client | | client | |
|
* | | client | | client | |
|
||||||
* | +----------+ +----------+ |
|
* | +----------+ +----------+ |
|
||||||
* | |
|
* | |
|
||||||
* +-----------------------------+
|
* +-----------------------------+
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
#include "TinyMqtt.h" // https://github.com/hsaturn/TinyMqtt
|
#include "TinyMqtt.h" // https://github.com/hsaturn/TinyMqtt
|
||||||
|
|
||||||
#define PORT 1883
|
const uint16_t PORT 1883;
|
||||||
MqttBroker broker(PORT);
|
const uint8_t RETAIN = 10; // Max retained messages
|
||||||
|
|
||||||
|
MqttBroker broker(PORT, RETAIN);
|
||||||
|
|
||||||
/** Basic Mqtt Broker
|
/** Basic Mqtt Broker
|
||||||
*
|
*
|
||||||
@@ -16,6 +18,8 @@ MqttBroker broker(PORT);
|
|||||||
* Your ESP will become a MqttBroker.
|
* Your ESP will become a MqttBroker.
|
||||||
* You can test it with any client such as mqtt-spy for example
|
* You can test it with any client such as mqtt-spy for example
|
||||||
*
|
*
|
||||||
|
* Messages are retained *only* if retain > 0
|
||||||
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
const char* ssid = "";
|
const char* ssid = "";
|
||||||
|
|||||||
@@ -1,22 +1,22 @@
|
|||||||
#include "TinyMqtt.h" // https://github.com/hsaturn/TinyMqtt
|
#include "TinyMqtt.h" // https://github.com/hsaturn/TinyMqtt
|
||||||
#include "TinyStreaming.h" // https://github.com/hsaturn/TinyConsole
|
#include "TinyStreaming.h" // https://github.com/hsaturn/TinyConsole
|
||||||
|
|
||||||
/** Simple Client (The simplest configuration)
|
/** Simple Client (The simplest configuration, client only sends topics)
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
* +--------+
|
* +--------+
|
||||||
* +------>| broker |<--- < Other client
|
* +------>| broker |<--- < Other client
|
||||||
* | +--------+
|
* | +--------+
|
||||||
* |
|
* |
|
||||||
* +-----------------+
|
* +-----------------+
|
||||||
* | ESP | |
|
* | ESP | |
|
||||||
* | +----------+ |
|
* | +----------+ |
|
||||||
* | | internal | |
|
* | | internal | |
|
||||||
* | | client | |
|
* | | client | |
|
||||||
* | +----------+ |
|
* | +----------+ |
|
||||||
* | |
|
* | |
|
||||||
* +-----------------+
|
* +-----------------+
|
||||||
*
|
*
|
||||||
* 1 - change the ssid/password
|
* 1 - change the ssid/password
|
||||||
* 2 - change BROKER values (or keep emqx.io test broker)
|
* 2 - change BROKER values (or keep emqx.io test broker)
|
||||||
* 3 - you can use mqtt-spy to connect to the same broker and
|
* 3 - you can use mqtt-spy to connect to the same broker and
|
||||||
@@ -39,7 +39,7 @@ const char* password = "";
|
|||||||
static float temp=19;
|
static float temp=19;
|
||||||
static MqttClient client;
|
static MqttClient client;
|
||||||
|
|
||||||
void setup()
|
void setup()
|
||||||
{
|
{
|
||||||
Serial.begin(115200);
|
Serial.begin(115200);
|
||||||
delay(500);
|
delay(500);
|
||||||
@@ -50,11 +50,11 @@ void setup()
|
|||||||
|
|
||||||
WiFi.mode(WIFI_STA);
|
WiFi.mode(WIFI_STA);
|
||||||
WiFi.begin(ssid, password);
|
WiFi.begin(ssid, password);
|
||||||
|
|
||||||
while (WiFi.status() != WL_CONNECTED)
|
while (WiFi.status() != WL_CONNECTED)
|
||||||
{ delay(500); Serial << '.'; }
|
{ delay(500); Serial << '.'; }
|
||||||
|
|
||||||
Serial << "Connected to " << ssid << "IP address: " << WiFi.localIP() << endl;
|
Serial << "Connected to " << ssid << "IP address: " << WiFi.localIP() << endl;
|
||||||
|
|
||||||
client.connect(BROKER, BROKER_PORT);
|
client.connect(BROKER, BROKER_PORT);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,14 +1,14 @@
|
|||||||
// vim: ts=2 sw=2 expandtab smartindent
|
// vim: ts=2 sw=2 expandtab smartindent
|
||||||
#include <TinyConsole.h>
|
#include <TinyConsole.h>
|
||||||
#include <TinyMqtt.h> // https://github.com/hsaturn/TinyMqtt
|
#include <TinyMqtt.h> // https://github.com/hsaturn/TinyMqtt
|
||||||
#include <TinyStreaming.h>
|
#include <TinyStreaming.h>
|
||||||
#if defined(ESP8266)
|
#if defined(ESP8266)
|
||||||
#include <ESP8266mDNS.h>
|
#include <ESP8266mDNS.h>
|
||||||
#elif defined(ESP32)
|
#elif defined(ESP32)
|
||||||
#include <WiFi.h>
|
#include <WiFi.h>
|
||||||
#include <ESPmDNS.h>
|
#include <ESPmDNS.h>
|
||||||
#else
|
#else
|
||||||
#error Unsupported platform
|
#error Unsupported platform
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
@@ -26,45 +26,47 @@ auto save_cursor = TinyConsole::save_cursor;
|
|||||||
auto restore_cursor = TinyConsole::restore_cursor;
|
auto restore_cursor = TinyConsole::restore_cursor;
|
||||||
auto erase_to_end = TinyConsole::erase_to_end;
|
auto erase_to_end = TinyConsole::erase_to_end;
|
||||||
|
|
||||||
const char* ssid = "";
|
const char *ssid = "Freebox-786A2F";
|
||||||
const char* password = "";
|
const char *password = "usurpavi8dalum64lumine?";
|
||||||
|
|
||||||
|
void onCommand(const string &command);
|
||||||
|
void eval(string &cmd);
|
||||||
|
|
||||||
struct free_broker
|
struct free_broker
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
free_broker(const char* s, uint16_t p, const char* comment) : url(s), port(p) {}
|
free_broker(const char *s, uint16_t p, const char * /* comment */) : url(s), port(p) {}
|
||||||
|
|
||||||
std::string url;
|
string url;
|
||||||
uint16_t port;
|
uint16_t port;
|
||||||
};
|
};
|
||||||
|
|
||||||
const std::map<std::string, free_broker> list =
|
const std::map<string, free_broker> list =
|
||||||
{
|
{
|
||||||
{ "mqtthq", { "public.mqtthq.com" , 8083, "publish/subscribe" }},
|
{"mqtthq", {"public.mqtthq.com", 8083, "publish/subscribe"}},
|
||||||
{ "hivemq", { "broker.hivemq.com", 1883, "" }}
|
{"hivemq", {"broker.hivemq.com", 1883, ""}}};
|
||||||
};
|
|
||||||
|
|
||||||
/** Very complex example
|
/** Very complex example
|
||||||
* Console allowing to make any kind of test,
|
* Console allowing to make any kind of test,
|
||||||
* even some stress tests.
|
* even some stress tests.
|
||||||
*
|
*
|
||||||
* Upload the sketch, the use the terminal.
|
* Upload the sketch, the use the terminal.
|
||||||
* Press H for mini help.
|
* Press H for mini help.
|
||||||
*
|
*
|
||||||
* tested with mqtt-spy-0.5.4
|
* tested with mqtt-spy-0.5.4
|
||||||
* TODO examples of scripts
|
* TODO examples of scripts
|
||||||
*/
|
*/
|
||||||
|
|
||||||
void onPublish(const MqttClient* srce, const Topic& topic, const char* payload, size_t length)
|
void onPublish(const MqttClient *srce, const Topic &topic, const char *payload, size_t length)
|
||||||
{
|
{
|
||||||
Console << cyan << "--> " << srce->id().c_str() << ": received " << topic.c_str() << white;
|
Console << cyan << "--> " << srce->id().c_str() << ": received " << topic.c_str() << white;
|
||||||
if (payload)
|
if (payload)
|
||||||
{
|
{
|
||||||
Console << ", payload[" << length << "]=[";
|
Console << ", payload[" << length << "]=[";
|
||||||
while(length--)
|
while (length--)
|
||||||
{
|
{
|
||||||
const char c=*payload++;
|
const char c = *payload++;
|
||||||
if (c<32)
|
if (c < 32)
|
||||||
Console << '?';
|
Console << '?';
|
||||||
else
|
else
|
||||||
Console << c;
|
Console << c;
|
||||||
@@ -73,8 +75,8 @@ void onPublish(const MqttClient* srce, const Topic& topic, const char* payload,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
std::map<std::string, MqttClient*> clients;
|
std::map<string, MqttClient *> clients;
|
||||||
std::map<std::string, MqttBroker*> brokers;
|
std::map<string, MqttBroker *> brokers;
|
||||||
|
|
||||||
void setup()
|
void setup()
|
||||||
{
|
{
|
||||||
@@ -86,7 +88,8 @@ void setup()
|
|||||||
delay(500);
|
delay(500);
|
||||||
|
|
||||||
Console.cls();
|
Console.cls();
|
||||||
Console << endl << endl;
|
Console << endl
|
||||||
|
<< endl;
|
||||||
Console << yellow
|
Console << yellow
|
||||||
<< "***************************************************************" << endl;
|
<< "***************************************************************" << endl;
|
||||||
Console << "* Welcome to the TinyMqtt console" << endl;
|
Console << "* Welcome to the TinyMqtt console" << endl;
|
||||||
@@ -94,10 +97,11 @@ void setup()
|
|||||||
Console << "* Enter help to view the list of commands." << endl;
|
Console << "* Enter help to view the list of commands." << endl;
|
||||||
Console << "***************************************************************" << endl;
|
Console << "***************************************************************" << endl;
|
||||||
Console << endl;
|
Console << endl;
|
||||||
if (strlen(ssid)==0)
|
if (strlen(ssid) == 0)
|
||||||
Console << red << "* ERROR: You must modify ssid/password in order" << endl
|
Console << red << "* ERROR: You must modify ssid/password in order" << endl
|
||||||
<< " to be able to connect to your Wifi network." << endl;
|
<< " to be able to connect to your Wifi network." << endl;
|
||||||
Console << endl << white;
|
Console << endl
|
||||||
|
<< white;
|
||||||
|
|
||||||
Console << "Connecting to '" << ssid << "' ";
|
Console << "Connecting to '" << ssid << "' ";
|
||||||
|
|
||||||
@@ -107,215 +111,234 @@ void setup()
|
|||||||
WiFi.begin(ssid, password);
|
WiFi.begin(ssid, password);
|
||||||
|
|
||||||
while (WiFi.status() != WL_CONNECTED)
|
while (WiFi.status() != WL_CONNECTED)
|
||||||
{ Console << '-'; delay(500); }
|
{
|
||||||
|
Console << '-';
|
||||||
|
delay(500);
|
||||||
|
}
|
||||||
|
|
||||||
Console << endl << "Connected to " << ssid << "IP address: " << WiFi.localIP() << endl;
|
Console << endl
|
||||||
|
<< "Connected to " << ssid << "IP address: " << WiFi.localIP() << endl;
|
||||||
|
|
||||||
const char* name="tinytest";
|
const char *name = "tinytest";
|
||||||
Console << "Starting MDNS, name= " << name;
|
Console << "Starting MDNS, name= " << name;
|
||||||
if (!MDNS.begin(name))
|
if (!MDNS.begin(name))
|
||||||
Console << " error, not available." << endl;
|
Console << " error, not available." << endl;
|
||||||
else
|
else
|
||||||
Console << " ok." << endl;
|
Console << " ok." << endl;
|
||||||
|
|
||||||
|
MqttBroker *broker = new MqttBroker(1883);
|
||||||
MqttBroker* broker = new MqttBroker(1883);
|
|
||||||
broker->begin();
|
broker->begin();
|
||||||
brokers["broker"] = broker;
|
brokers["broker"] = broker;
|
||||||
|
|
||||||
if (Console.isTerm()) onCommand("every 333 view");
|
if (Console.isTerm())
|
||||||
|
onCommand("every 333 view");
|
||||||
|
|
||||||
|
Console.prompt();
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string getword(std::string& str, const char* if_empty=nullptr, char sep=' ');
|
string getword(string &str, const char *if_empty = nullptr, char sep = ' ');
|
||||||
|
|
||||||
int getint(std::string& str, const int if_empty=0)
|
int getint(string &str, const int if_empty = 0)
|
||||||
{
|
{
|
||||||
std::string str2=str;
|
string str2 = str;
|
||||||
std::string sword = getword(str);
|
string sword = getword(str);
|
||||||
if (sword[0] and isdigit(sword[0]))
|
if (sword[0] and isdigit(sword[0]))
|
||||||
{
|
{
|
||||||
int ret=atoi(sword.c_str());
|
int ret = atoi(sword.c_str());
|
||||||
while(isdigit(sword[0]) or sword[0]==' ') sword.erase(0,1);
|
while (isdigit(sword[0]) or sword[0] == ' ')
|
||||||
if (sword.length()) str = sword+' '+str;
|
sword.erase(0, 1);
|
||||||
|
if (sword.length())
|
||||||
|
str = sword + ' ' + str;
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
str=str2;
|
str = str2;
|
||||||
return if_empty;
|
return if_empty;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string getword(std::string& str, const char* if_empty/*=nullptr*/, char sep/*=' '*/)
|
string getword(string &str, const char *if_empty /*=nullptr*/, char sep /*=' '*/)
|
||||||
{
|
{
|
||||||
char quote=(str[0]=='"' or str[0]=='\'' ? str[0] : 0);
|
char quote = (str[0] == '"' or str[0] == '\'' ? str[0] : 0);
|
||||||
if (quote) str.erase(0,1);
|
if (quote)
|
||||||
std::string sword;
|
str.erase(0, 1);
|
||||||
while(str.length() and (str[0]!=sep or quote))
|
string sword;
|
||||||
|
while (str.length() and (str[0] != sep or quote))
|
||||||
{
|
{
|
||||||
if (str[0]==quote)
|
if (str[0] == quote)
|
||||||
{
|
{
|
||||||
str.erase(0,1);
|
str.erase(0, 1);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
sword += str[0];
|
sword += str[0];
|
||||||
str.erase(0,1);
|
str.erase(0, 1);
|
||||||
}
|
}
|
||||||
while(str[0]==sep) str.erase(0,1);
|
while (str[0] == sep)
|
||||||
if (if_empty and sword.length()==0) return if_empty;
|
str.erase(0, 1);
|
||||||
if (quote==false and sword.length()>=4 and sword.substr(0,3)=="rnd")
|
if (if_empty and sword.length() == 0)
|
||||||
|
return if_empty;
|
||||||
|
if (quote == false and sword.length() >= 4 and sword.substr(0, 3) == "rnd")
|
||||||
{
|
{
|
||||||
sword.erase(0,3);
|
sword.erase(0, 3);
|
||||||
if (sword[0]=='(')
|
if (sword[0] == '(')
|
||||||
{
|
{
|
||||||
int to = 100;
|
int to = 100;
|
||||||
sword.erase(0,1);
|
sword.erase(0, 1);
|
||||||
int from=getint(sword);
|
int from = getint(sword);
|
||||||
if (sword[0]==',')
|
if (sword[0] == ',')
|
||||||
{
|
{
|
||||||
sword.erase(0,1);
|
sword.erase(0, 1);
|
||||||
to = getint(sword);
|
to = getint(sword);
|
||||||
if (sword[0]!=')') Console << "Missing ')'" << endl;
|
if (sword[0] != ')')
|
||||||
|
Console << "Missing ')'" << endl;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
to=from;
|
to = from;
|
||||||
from=0;
|
from = 0;
|
||||||
}
|
}
|
||||||
return String(random(from,to)).c_str();
|
return String(random(from, to)).c_str();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
Console << "Missing '('" << endl;
|
Console << "Missing '('" << endl;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
while(str[0]==' ') str.erase(0,1);
|
while (str[0] == ' ')
|
||||||
|
str.erase(0, 1);
|
||||||
return sword;
|
return sword;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool isaddr(std::string s)
|
bool isaddr(string s)
|
||||||
{
|
{
|
||||||
if (s.length()==0 or s.length()>3) return false;
|
if (s.length() == 0 or s.length() > 3)
|
||||||
for(char c: s)
|
return false;
|
||||||
if (c<'0' or c>'9') return false;
|
for (char c : s)
|
||||||
|
if (c < '0' or c > '9')
|
||||||
|
return false;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string getip(std::string& str, const char* if_empty=nullptr, char sep=' ')
|
string getip(string &str, const char *if_empty = nullptr, char sep = ' ')
|
||||||
{
|
{
|
||||||
std::string addr=getword(str, if_empty, sep);
|
string addr = getword(str, if_empty, sep);
|
||||||
std::string ip=addr;
|
string ip = addr;
|
||||||
std::vector<std::string> build;
|
std::vector<string> build;
|
||||||
while(ip.length())
|
while (ip.length())
|
||||||
{
|
{
|
||||||
std::string b=getword(ip,nullptr,'.');
|
string b = getword(ip, nullptr, '.');
|
||||||
if (isaddr(b) && build.size()<4)
|
if (isaddr(b) && build.size() < 4)
|
||||||
{
|
{
|
||||||
build.push_back(b);
|
build.push_back(b);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
return addr;
|
return addr;
|
||||||
}
|
}
|
||||||
IPAddress local=WiFi.localIP();
|
IPAddress local = WiFi.localIP();
|
||||||
addr.clear();
|
addr.clear();
|
||||||
while(build.size()!=4)
|
while (build.size() != 4)
|
||||||
{
|
{
|
||||||
std::stringstream b;
|
std::stringstream b;
|
||||||
b << (int)local[3-build.size()];
|
b << (int)local[3 - build.size()];
|
||||||
build.insert(build.begin(), b.str());
|
build.insert(build.begin(), b.str().c_str());
|
||||||
}
|
}
|
||||||
for(std::string s: build)
|
for (string s : build)
|
||||||
{
|
{
|
||||||
if (addr.length()) addr += '.';
|
if (addr.length())
|
||||||
|
addr += '.';
|
||||||
addr += s;
|
addr += s;
|
||||||
}
|
}
|
||||||
Console << "connect address: " << addr << endl;
|
Console << "connect address: " << addr << endl;
|
||||||
return addr;
|
return addr;
|
||||||
}
|
}
|
||||||
|
|
||||||
std::map<std::string, std::string> vars;
|
std::map<string, string> vars;
|
||||||
|
|
||||||
std::set<std::string> commands = {
|
std::set<string> commands = {
|
||||||
"broker", "blink", "client", "connect",
|
"broker", "blink", "client", "connect",
|
||||||
"create", "delete", "debug", "help", "interval",
|
"create", "delete", "debug", "help", "interval",
|
||||||
"list", "ls", "ip", "off", "on", "set",
|
"list", "ls", "ip", "off", "on", "set",
|
||||||
"publish", "reset", "subscribe", "unsubscribe", "view", "echo", "every"
|
"publish", "reset", "subscribe", "unsubscribe", "view", "echo", "every"};
|
||||||
};
|
|
||||||
|
|
||||||
void convertToCommand(std::string& search)
|
void convertToCommand(string &search)
|
||||||
{
|
{
|
||||||
while(search[0]==' ') search.erase(0,1);
|
while (search[0] == ' ')
|
||||||
if (search.length()==0) return;
|
search.erase(0, 1);
|
||||||
std::string matches;
|
if (search.length() == 0)
|
||||||
int count=0;
|
return;
|
||||||
for(std::string cmd: commands)
|
string matches;
|
||||||
|
int count = 0;
|
||||||
|
for (string cmd : commands)
|
||||||
{
|
{
|
||||||
if (cmd.substr(0, search.length()) == search)
|
if (cmd.substr(0, search.length()) == search)
|
||||||
{
|
{
|
||||||
if (count) matches +=", ";
|
if (count)
|
||||||
|
matches += ", ";
|
||||||
count++;
|
count++;
|
||||||
matches += cmd;
|
matches += cmd;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (count==1)
|
if (count == 1)
|
||||||
search = matches;
|
search = matches;
|
||||||
else if (count>1)
|
else if (count > 1)
|
||||||
{
|
{
|
||||||
Console << "Ambiguous command: " << matches << endl;
|
Console << "Ambiguous command: " << matches << endl;
|
||||||
search.clear();
|
search.clear();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void replace(const char* d, std::string& str, std::string srch, std::string to)
|
void replace(const char *d, string &str, string srch, string to)
|
||||||
{
|
{
|
||||||
if (d[0] && d[1])
|
if (d[0] && d[1])
|
||||||
{
|
{
|
||||||
srch=d[0]+srch+d[1];
|
srch = d[0] + srch + d[1];
|
||||||
to=d[0]+to+d[1];
|
to = d[0] + to + d[1];
|
||||||
|
|
||||||
size_t pos = 0;
|
size_t pos = 0;
|
||||||
while((pos=str.find(srch, pos)) != std::string::npos)
|
while ((pos = str.find(srch, pos)) != string::npos)
|
||||||
{
|
{
|
||||||
str.erase(pos, srch.length());
|
str.erase(pos, srch.length());
|
||||||
str.insert(pos, to);
|
str.insert(pos, to);
|
||||||
pos += to.length()-1;
|
pos += to.length() - 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void replaceVars(std::string& cmd)
|
void replaceVars(string &cmd)
|
||||||
{
|
{
|
||||||
cmd = ' '+cmd+' ';
|
cmd = ' ' + cmd + ' ';
|
||||||
|
|
||||||
for(auto it: vars)
|
for (auto it : vars)
|
||||||
{
|
{
|
||||||
replace("..", cmd, it.first, it.second);
|
replace("..", cmd, it.first, it.second);
|
||||||
replace(". ", cmd, it.first, it.second);
|
replace(". ", cmd, it.first, it.second);
|
||||||
replace(" .", cmd, it.first, it.second);
|
replace(" .", cmd, it.first, it.second);
|
||||||
replace(" ", cmd, it.first, it.second);
|
replace(" ", cmd, it.first, it.second);
|
||||||
}
|
}
|
||||||
cmd.erase(0, cmd.find_first_not_of(" "));
|
cmd.erase(0, cmd.find_first_not_of(' '));
|
||||||
cmd.erase(cmd.find_last_not_of(" ")+1);
|
cmd.erase(cmd.find_last_not_of(' ') + 1);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool compare(std::string s, const char* cmd)
|
bool compare(string s, const char *cmd)
|
||||||
{
|
{
|
||||||
uint8_t p=0;
|
uint8_t p = 0;
|
||||||
while(s[p++]==*cmd++)
|
while (s[p++] == *cmd++)
|
||||||
{
|
{
|
||||||
if (*cmd==0 or s[p]==0) return true;
|
if (*cmd == 0 or s[p] == 0)
|
||||||
if (s[p]==' ') return true;
|
return true;
|
||||||
|
if (s[p] == ' ')
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
using ClientFunction = void(*)(std::string& cmd, MqttClient* publish);
|
using ClientFunction = void (*)(string &cmd, MqttClient *publish);
|
||||||
|
|
||||||
struct Every
|
struct Every
|
||||||
{
|
{
|
||||||
std::string cmd;
|
string cmd;
|
||||||
uint32_t ms;
|
uint32_t ms;
|
||||||
uint32_t next;
|
uint32_t next;
|
||||||
uint32_t underrun=0;
|
uint32_t underrun = 0;
|
||||||
bool active=true;
|
bool active = true;
|
||||||
|
|
||||||
void dump()
|
void dump()
|
||||||
{
|
{
|
||||||
@@ -324,12 +347,12 @@ struct Every
|
|||||||
else
|
else
|
||||||
Console << red << "disabled";
|
Console << red << "disabled";
|
||||||
|
|
||||||
auto mill=millis();
|
auto mill = millis();
|
||||||
Console << white << ms << "ms [" << cmd << "] next in ";
|
Console << white << ms << "ms [" << cmd << "] next in ";
|
||||||
if (mill > next)
|
if (mill > next)
|
||||||
Console << "now";
|
Console << "now";
|
||||||
else
|
else
|
||||||
Console << next-mill << "ms";
|
Console << next - mill << "ms";
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -341,49 +364,49 @@ int16_t blink;
|
|||||||
|
|
||||||
std::vector<Every> everies;
|
std::vector<Every> everies;
|
||||||
|
|
||||||
void onCommand(const std::string& command)
|
void onCommand(const string &command)
|
||||||
{
|
{
|
||||||
Console << endl;
|
Console << endl;
|
||||||
std::string cmd=command;
|
string cmd = command;
|
||||||
if (cmd.substr(0,3)!="set") replaceVars(cmd);
|
if (cmd.substr(0, 3) != "set")
|
||||||
|
replaceVars(cmd);
|
||||||
eval(cmd);
|
eval(cmd);
|
||||||
Console << endl;
|
Console << endl;
|
||||||
Console.prompt();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void clientConnect(MqttClient* client, std::string& cmd)
|
void clientConnect(MqttClient *client, string &cmd)
|
||||||
{
|
{
|
||||||
std::string remote = getword(cmd);
|
string remote = getword(cmd);
|
||||||
uint16_t port;
|
uint16_t port;
|
||||||
auto it=list.find(remote);
|
auto it = list.find(remote);
|
||||||
if (it != list.end())
|
if (it != list.end())
|
||||||
{
|
{
|
||||||
Console << "Connecting to free broker: " << remote << endl;
|
Console << "Connecting to free broker: " << remote << endl;
|
||||||
remote = it->second.url;
|
remote = it->second.url;
|
||||||
port=it->second.port;
|
port = it->second.port;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
port=getint(cmd);
|
port = getint(cmd);
|
||||||
|
|
||||||
client->connect(remote.c_str(), port, getint(cmd, 60));
|
client->connect(remote.c_str(), port, getint(cmd, 60));
|
||||||
Console << (client->connected() ? "connected." : "not connected") << endl;
|
Console << (client->connected() ? "connected." : "not connected") << endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
void eval(std::string& cmd)
|
void eval(string &cmd)
|
||||||
{
|
{
|
||||||
while(cmd.length())
|
while (cmd.length())
|
||||||
{
|
{
|
||||||
MqttError retval = MqttOk;
|
MqttError retval = MqttOk;
|
||||||
|
|
||||||
std::string s;
|
string s;
|
||||||
MqttBroker* broker = nullptr;
|
MqttBroker *broker = nullptr;
|
||||||
MqttClient* client = nullptr;
|
MqttClient *client = nullptr;
|
||||||
|
|
||||||
// client.function notation
|
// client.function notation
|
||||||
if (cmd.find('.') != std::string::npos &&
|
if (cmd.find('.') != string::npos &&
|
||||||
cmd.find('.') < cmd.find(' '))
|
cmd.find('.') < cmd.find(' '))
|
||||||
{
|
{
|
||||||
s=getword(cmd, nullptr, '.');
|
s = getword(cmd, nullptr, '.');
|
||||||
|
|
||||||
if (s.length())
|
if (s.length())
|
||||||
{
|
{
|
||||||
@@ -404,28 +427,26 @@ void eval(std::string& cmd)
|
|||||||
}
|
}
|
||||||
|
|
||||||
s = getword(cmd);
|
s = getword(cmd);
|
||||||
if (s.length()) convertToCommand(s);
|
if (s.length())
|
||||||
if (s.length()==0)
|
convertToCommand(s);
|
||||||
{}
|
if (s.length() == 0)
|
||||||
else if (compare(s, "debug"))
|
|
||||||
{
|
{
|
||||||
#if TINY_MQTT_DEBUG
|
|
||||||
TinyMqtt::debug = getint(cmd);
|
|
||||||
#else
|
|
||||||
Console << red << "TinyMqtt not compiled in debug" << endl;
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
else if (compare(s, "list"))
|
else if (compare(s, "list"))
|
||||||
{
|
{
|
||||||
Console << "List of free servers" << endl;
|
Console << "List of free servers" << endl;
|
||||||
for(const auto& fb: list)
|
for (const auto &fb : list)
|
||||||
{
|
{
|
||||||
Console << " " << fb.first << " : " << fb.second.url << ":" << fb.second.port << endl;
|
Console << " " << fb.first << " : " << fb.second.url << ":" << fb.second.port << endl;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else if (compare(s, "free"))
|
||||||
|
{
|
||||||
|
Console << "Free memory: " << ESP.getFreeHeap() << endl;
|
||||||
|
}
|
||||||
else if (compare(s, "delete"))
|
else if (compare(s, "delete"))
|
||||||
{
|
{
|
||||||
if (client==nullptr && broker==nullptr)
|
if (client == nullptr && broker == nullptr)
|
||||||
{
|
{
|
||||||
s = getword(cmd);
|
s = getword(cmd);
|
||||||
if (clients.find(s) != clients.end())
|
if (clients.find(s) != clients.end())
|
||||||
@@ -441,9 +462,10 @@ void eval(std::string& cmd)
|
|||||||
}
|
}
|
||||||
if (client)
|
if (client)
|
||||||
{
|
{
|
||||||
for (auto it: clients)
|
for (auto it : clients)
|
||||||
{
|
{
|
||||||
if (it.second != client) continue;
|
if (it.second != client)
|
||||||
|
continue;
|
||||||
Console << "deleted" << endl;
|
Console << "deleted" << endl;
|
||||||
delete (it.second);
|
delete (it.second);
|
||||||
clients.erase(it.first);
|
clients.erase(it.first);
|
||||||
@@ -453,9 +475,10 @@ void eval(std::string& cmd)
|
|||||||
}
|
}
|
||||||
else if (broker)
|
else if (broker)
|
||||||
{
|
{
|
||||||
for(auto it: brokers)
|
for (auto it : brokers)
|
||||||
{
|
{
|
||||||
if (broker != it.second) continue;
|
if (broker != it.second)
|
||||||
|
continue;
|
||||||
Console << "deleted" << endl;
|
Console << "deleted" << endl;
|
||||||
delete (it.second);
|
delete (it.second);
|
||||||
brokers.erase(it.first);
|
brokers.erase(it.first);
|
||||||
@@ -468,9 +491,24 @@ void eval(std::string& cmd)
|
|||||||
}
|
}
|
||||||
else if (broker)
|
else if (broker)
|
||||||
{
|
{
|
||||||
if (compare(s,"connect"))
|
if (compare(s, "connect"))
|
||||||
{
|
{
|
||||||
Console << "NYI" << endl;
|
string remote = getword(cmd);
|
||||||
|
int port = getint(cmd);
|
||||||
|
if (port == 0) port=1883;
|
||||||
|
broker->connect(remote, port);
|
||||||
|
if (broker->connected())
|
||||||
|
Console << "Broker connected";
|
||||||
|
else
|
||||||
|
Console << red << "Unable to connect";
|
||||||
|
|
||||||
|
Console << " to " << remote << ':' << port << white << endl;
|
||||||
|
}
|
||||||
|
else if (compare(s, "retain"))
|
||||||
|
{
|
||||||
|
if (cmd.size())
|
||||||
|
broker->retain(getint(cmd));
|
||||||
|
Console << "retain=" << broker->retain() << ", retained msg=" << broker->retainCount() << endl;
|
||||||
}
|
}
|
||||||
else if (compare(s, "view"))
|
else if (compare(s, "view"))
|
||||||
{
|
{
|
||||||
@@ -484,15 +522,19 @@ void eval(std::string& cmd)
|
|||||||
}
|
}
|
||||||
else if (client)
|
else if (client)
|
||||||
{
|
{
|
||||||
if (compare(s,"connect"))
|
if (compare(s, "connect"))
|
||||||
{
|
{
|
||||||
clientConnect(client, cmd);
|
clientConnect(client, cmd);
|
||||||
}
|
}
|
||||||
else if (compare(s,"publish"))
|
else if (compare(s, "rpublish"))
|
||||||
|
{
|
||||||
|
retval = client->publish(getword(cmd), getword(cmd), true);
|
||||||
|
}
|
||||||
|
else if (compare(s, "publish"))
|
||||||
{
|
{
|
||||||
retval = client->publish(getword(cmd), getword(cmd));
|
retval = client->publish(getword(cmd), getword(cmd));
|
||||||
}
|
}
|
||||||
else if (compare(s,"subscribe"))
|
else if (compare(s, "subscribe"))
|
||||||
{
|
{
|
||||||
client->subscribe(getword(cmd));
|
client->subscribe(getword(cmd));
|
||||||
}
|
}
|
||||||
@@ -512,27 +554,27 @@ void eval(std::string& cmd)
|
|||||||
}
|
}
|
||||||
else if (compare(s, "on"))
|
else if (compare(s, "on"))
|
||||||
{
|
{
|
||||||
uint8_t pin=getint(cmd, 2);
|
uint8_t pin = getint(cmd, 2);
|
||||||
pinMode(pin, OUTPUT);
|
pinMode(pin, OUTPUT);
|
||||||
digitalWrite(pin, HIGH);
|
digitalWrite(pin, HIGH);
|
||||||
}
|
}
|
||||||
else if (compare(s, "off"))
|
else if (compare(s, "off"))
|
||||||
{
|
{
|
||||||
uint8_t pin=getint(cmd, 2);
|
uint8_t pin = getint(cmd, 2);
|
||||||
pinMode(pin, OUTPUT);
|
pinMode(pin, OUTPUT);
|
||||||
digitalWrite(pin, LOW);
|
digitalWrite(pin, LOW);
|
||||||
}
|
}
|
||||||
else if (compare(s, "echo"))
|
else if (compare(s, "echo"))
|
||||||
{
|
{
|
||||||
s=getword(cmd);
|
s = getword(cmd);
|
||||||
if (s=="on")
|
if (s == "on")
|
||||||
echo_on = true;
|
echo_on = true;
|
||||||
else if (s=="off")
|
else if (s == "off")
|
||||||
echo_on = false;
|
echo_on = false;
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
Console << s << ' ';
|
Console << s << ' ';
|
||||||
while(cmd.length())
|
while (cmd.length())
|
||||||
{
|
{
|
||||||
Console << getword(cmd) << ' ';
|
Console << getword(cmd) << ' ';
|
||||||
}
|
}
|
||||||
@@ -546,9 +588,9 @@ void eval(std::string& cmd)
|
|||||||
if (cmd.length())
|
if (cmd.length())
|
||||||
{
|
{
|
||||||
Every every;
|
Every every;
|
||||||
every.ms=ms;
|
every.ms = ms;
|
||||||
every.cmd=cmd;
|
every.cmd = cmd;
|
||||||
every.next=millis()+ms;
|
every.next = millis() + ms;
|
||||||
everies.push_back(every);
|
everies.push_back(every);
|
||||||
every.dump();
|
every.dump();
|
||||||
Console << endl;
|
Console << endl;
|
||||||
@@ -557,16 +599,16 @@ void eval(std::string& cmd)
|
|||||||
}
|
}
|
||||||
else if (compare(cmd, "off") or compare(cmd, "on"))
|
else if (compare(cmd, "off") or compare(cmd, "on"))
|
||||||
{
|
{
|
||||||
bool active=getword(cmd)=="on";
|
bool active = getword(cmd) == "on";
|
||||||
uint8_t ever=getint(cmd, 100);
|
uint8_t ever = getint(cmd, 100);
|
||||||
uint8_t count=0;
|
uint8_t count = 0;
|
||||||
for(auto& every: everies)
|
for (auto &every : everies)
|
||||||
{
|
{
|
||||||
if (count==ever or (ever==100))
|
if (count == ever or (ever == 100))
|
||||||
{
|
{
|
||||||
if (every.active != active)
|
if (every.active != active)
|
||||||
{
|
{
|
||||||
Console << "every #" << count << (active ? " on" :" off") << endl;
|
Console << "every #" << count << (active ? " on" : " off") << endl;
|
||||||
every.active = active;
|
every.active = active;
|
||||||
every.underrun = 0;
|
every.underrun = 0;
|
||||||
}
|
}
|
||||||
@@ -574,12 +616,12 @@ void eval(std::string& cmd)
|
|||||||
count++;
|
count++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (compare(cmd, "list") or cmd.length()==0)
|
else if (compare(cmd, "list") or cmd.length() == 0)
|
||||||
{
|
{
|
||||||
getword(cmd);
|
getword(cmd);
|
||||||
Console << "List of everies (ms=" << millis() << ")" << endl;
|
Console << "List of everies (ms=" << millis() << ")" << endl;
|
||||||
uint8_t count=0;
|
uint8_t count = 0;
|
||||||
for(auto& every: everies)
|
for (auto &every : everies)
|
||||||
{
|
{
|
||||||
Console << count << ": ";
|
Console << count << ": ";
|
||||||
every.dump();
|
every.dump();
|
||||||
@@ -591,20 +633,20 @@ void eval(std::string& cmd)
|
|||||||
{
|
{
|
||||||
Console << "Removing..." << endl;
|
Console << "Removing..." << endl;
|
||||||
getword(cmd);
|
getword(cmd);
|
||||||
int8_t every=getint(cmd, -1);
|
int8_t every = getint(cmd, -1);
|
||||||
if (every==-1 and compare(cmd, "last") and everies.size())
|
if (every == -1 and compare(cmd, "last") and everies.size())
|
||||||
{
|
{
|
||||||
getword(cmd);
|
getword(cmd);
|
||||||
everies.erase(everies.begin()+everies.size()-1);
|
everies.erase(everies.begin() + everies.size() - 1);
|
||||||
}
|
}
|
||||||
else if (every==-1 and compare(cmd, "all"))
|
else if (every == -1 and compare(cmd, "all"))
|
||||||
{
|
{
|
||||||
getword(cmd);
|
getword(cmd);
|
||||||
everies.clear();
|
everies.clear();
|
||||||
}
|
}
|
||||||
else if (everies.size() > (uint8_t)every)
|
else if (everies.size() > (uint8_t)every)
|
||||||
{
|
{
|
||||||
everies.erase(everies.begin()+every);
|
everies.erase(everies.begin() + every);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
Console << "Bad colmmand" << endl;
|
Console << "Bad colmmand" << endl;
|
||||||
@@ -617,33 +659,33 @@ void eval(std::string& cmd)
|
|||||||
int8_t blink_nr = getint(cmd, -1);
|
int8_t blink_nr = getint(cmd, -1);
|
||||||
if (blink_nr >= 0)
|
if (blink_nr >= 0)
|
||||||
{
|
{
|
||||||
blink_ms_on[blink_nr]=getint(cmd, blink_ms_on[blink_nr]);
|
blink_ms_on[blink_nr] = getint(cmd, blink_ms_on[blink_nr]);
|
||||||
blink_ms_off[blink_nr]=getint(cmd, blink_ms_on[blink_nr]);
|
blink_ms_off[blink_nr] = getint(cmd, blink_ms_on[blink_nr]);
|
||||||
pinMode(blink_nr, OUTPUT);
|
pinMode(blink_nr, OUTPUT);
|
||||||
blink_next[blink_nr] = millis();
|
blink_next[blink_nr] = millis();
|
||||||
Console << "Blink " << blink_nr << ' ' << (blink_ms_on[blink_nr] ? "on" : "off") << endl;
|
Console << "Blink " << blink_nr << ' ' << (blink_ms_on[blink_nr] ? "on" : "off") << endl;
|
||||||
if (blink_ms_on[blink_nr])
|
if (blink_ms_on[blink_nr])
|
||||||
blink |= 1<< blink_nr;
|
blink |= 1 << blink_nr;
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
blink &= ~(1<< blink_nr);
|
blink &= ~(1 << blink_nr);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (compare(s, "broker"))
|
else if (compare(s, "broker"))
|
||||||
{
|
{
|
||||||
std::string id=getword(cmd);
|
string id = getword(cmd);
|
||||||
if (clients.find(id) != clients.end())
|
if (clients.find(id) != clients.end())
|
||||||
{
|
{
|
||||||
Console << "A client already have that name" << endl;
|
Console << "A client already have that name" << endl;
|
||||||
cmd.clear();
|
cmd.clear();
|
||||||
}
|
}
|
||||||
else if (id.length() or brokers.find(id)!=brokers.end())
|
else if (id.length() or brokers.find(id) != brokers.end())
|
||||||
{
|
{
|
||||||
int port=getint(cmd, 0);
|
int port = getint(cmd, 0);
|
||||||
if (port)
|
if (port)
|
||||||
{
|
{
|
||||||
MqttBroker* broker = new MqttBroker(port);
|
MqttBroker *broker = new MqttBroker(port);
|
||||||
broker->begin();
|
broker->begin();
|
||||||
|
|
||||||
brokers[id] = broker;
|
brokers[id] = broker;
|
||||||
@@ -663,25 +705,26 @@ void eval(std::string& cmd)
|
|||||||
}
|
}
|
||||||
else if (compare(s, "client"))
|
else if (compare(s, "client"))
|
||||||
{
|
{
|
||||||
std::string id=getword(cmd);
|
string id = getword(cmd);
|
||||||
if (brokers.find(id) != brokers.end())
|
if (brokers.find(id) != brokers.end())
|
||||||
{
|
{
|
||||||
Console << "A broker have that name" << endl;
|
Console << "A broker have that name" << endl;
|
||||||
cmd.clear();
|
cmd.clear();
|
||||||
}
|
}
|
||||||
else if (id.length() or clients.find(id)!=clients.end())
|
else if (id.length() or clients.find(id) != clients.end())
|
||||||
{
|
{
|
||||||
s=getword(cmd); // broker
|
s = getword(cmd); // broker
|
||||||
if (s=="" or brokers.find(s) != brokers.end() or list.find(s) != list.end())
|
if (s == "" or brokers.find(s) != brokers.end() or list.find(s) != list.end())
|
||||||
{
|
{
|
||||||
MqttBroker* broker = nullptr;
|
MqttBroker *broker = nullptr;
|
||||||
if (s.length()) broker = brokers[s];
|
if (s.length())
|
||||||
MqttClient* client = new MqttClient(broker, id);
|
broker = brokers[s];
|
||||||
clients[id]=client;
|
MqttClient *client = new MqttClient(broker, id);
|
||||||
|
clients[id] = client;
|
||||||
client->setCallback(onPublish);
|
client->setCallback(onPublish);
|
||||||
if (list.find(s) != list.end())
|
if (list.find(s) != list.end())
|
||||||
{
|
{
|
||||||
cmd=s+' '+cmd;
|
cmd = s + ' ' + cmd;
|
||||||
clientConnect(client, cmd);
|
clientConnect(client, cmd);
|
||||||
}
|
}
|
||||||
Console << "new client (" << id.c_str() << ", " << s.c_str() << ')' << endl;
|
Console << "new client (" << id.c_str() << ", " << s.c_str() << ')' << endl;
|
||||||
@@ -700,10 +743,10 @@ void eval(std::string& cmd)
|
|||||||
}
|
}
|
||||||
else if (compare(s, "set"))
|
else if (compare(s, "set"))
|
||||||
{
|
{
|
||||||
std::string name(getword(cmd));
|
string name(getword(cmd));
|
||||||
if (name.length()==0)
|
if (name.length() == 0)
|
||||||
{
|
{
|
||||||
for(auto it: vars)
|
for (auto it : vars)
|
||||||
{
|
{
|
||||||
Console << " " << it.first << " -> " << it.second << endl;
|
Console << " " << it.first << " -> " << it.second << endl;
|
||||||
}
|
}
|
||||||
@@ -730,18 +773,19 @@ void eval(std::string& cmd)
|
|||||||
if (view)
|
if (view)
|
||||||
{
|
{
|
||||||
Console << save_cursor << magenta;
|
Console << save_cursor << magenta;
|
||||||
Console.gotoxy(1,1);
|
Console.gotoxy(1, 1);
|
||||||
}
|
}
|
||||||
Console << "--< " << '/' << clients.size() << " client/s. >--" << erase_to_end << endl;
|
Console << "--< " << '/' << clients.size() << " client/s. >--" << erase_to_end;
|
||||||
for(auto it: clients)
|
Console << " (FreeMem: " << ESP.getFreeHeap() << ')' << endl;
|
||||||
|
for (auto it : clients)
|
||||||
{
|
{
|
||||||
it.second->dump(" ");
|
it.second->dump(" ");
|
||||||
}
|
}
|
||||||
|
|
||||||
Console << "--< " << brokers.size() << " brokers/s. >--" << erase_to_end << endl;
|
Console << "--< " << brokers.size() << " brokers/s. >--" << erase_to_end << endl;
|
||||||
for(auto it: brokers)
|
for (auto it : brokers)
|
||||||
{
|
{
|
||||||
Console << " +-- '" << it.first.c_str() << "' " << it.second->clientsCount() << " client/s."<< erase_to_end << endl;
|
Console << " +-- '" << it.first.c_str() << "' " << it.second->clientsCount() << " client/s." << erase_to_end << endl;
|
||||||
it.second->dump(" ");
|
it.second->dump(" ");
|
||||||
}
|
}
|
||||||
if (view)
|
if (view)
|
||||||
@@ -754,23 +798,26 @@ void eval(std::string& cmd)
|
|||||||
ESP.restart();
|
ESP.restart();
|
||||||
else if (compare(s, "ip"))
|
else if (compare(s, "ip"))
|
||||||
Console << "IP: " << WiFi.localIP() << endl;
|
Console << "IP: " << WiFi.localIP() << endl;
|
||||||
else if (compare(s,"help"))
|
else if (compare(s, "help"))
|
||||||
{
|
{
|
||||||
Console << "syntax:" << endl;
|
Console << "syntax:" << endl;
|
||||||
Console << " MqttBroker:" << endl;
|
Console << " MqttBroker:" << endl;
|
||||||
Console << " broker {broker_name} {port} : create a new broker" << endl;
|
Console << " broker {name} {port} : create a new broker" << endl;
|
||||||
Console << " broker_name can be one of 'list'" << endl;
|
Console << " name can be one of 'list'" << endl;
|
||||||
Console << " broker_name.delete : delete a broker (buggy)" << endl;
|
Console << " name.delete : delete a broker (buggy)" << endl;
|
||||||
Console << " broker_name.view : dump a broker" << endl;
|
Console << " name.retain [#] : show/set retain value" << endl;
|
||||||
|
Console << " name.view : dump a broker" << endl;
|
||||||
Console << endl;
|
Console << endl;
|
||||||
Console << " MqttClient:" << endl;
|
Console << " MqttClient:" << endl;
|
||||||
Console << " client {name} {broker} : create a client then" << endl;
|
Console << " client {name} {broker} : create a client then" << endl;
|
||||||
Console << " name.connect [ip] [port] [alive]" << endl;
|
Console << " name.connect [ip] [port] [alive]" << endl;
|
||||||
Console << " name.[un]subscribe topic" << endl;
|
Console << " name.[un]subscribe topic" << endl;
|
||||||
Console << " name.publish topic [payload]" << endl;
|
Console << " name.publish topic [payload]" << endl;
|
||||||
|
Console << " name.rpublish topic [payload] : publish a retained message" << endl;
|
||||||
Console << " name.view" << endl;
|
Console << " name.view" << endl;
|
||||||
Console << " name.delete" << endl;
|
Console << " name.delete" << endl;
|
||||||
Console << endl;
|
Console << endl;
|
||||||
|
Console << " free : view free mem" << endl;
|
||||||
Console << " list : list of free brokers (debug 1 advised)" << endl;
|
Console << " list : list of free brokers (debug 1 advised)" << endl;
|
||||||
Console << " debug #" << endl;
|
Console << " debug #" << endl;
|
||||||
Console << " list : get list of free brokers" << endl;
|
Console << " list : get list of free brokers" << endl;
|
||||||
@@ -788,7 +835,8 @@ void eval(std::string& cmd)
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
while(s[0]==' ') s.erase(0,1);
|
while (s[0] == ' ')
|
||||||
|
s.erase(0, 1);
|
||||||
if (s.length())
|
if (s.length())
|
||||||
Console << "Unknown command (" << s.c_str() << ")" << endl;
|
Console << "Unknown command (" << s.c_str() << ")" << endl;
|
||||||
}
|
}
|
||||||
@@ -802,30 +850,32 @@ void eval(std::string& cmd)
|
|||||||
|
|
||||||
void loop()
|
void loop()
|
||||||
{
|
{
|
||||||
auto ms=millis();
|
auto ms = millis();
|
||||||
int8_t out=0;
|
int8_t out = 0;
|
||||||
int16_t blink_bits = blink;
|
int16_t blink_bits = blink;
|
||||||
uint8_t e=0;
|
uint8_t e = 0;
|
||||||
|
|
||||||
for(auto& every: everies)
|
for (auto &every : everies)
|
||||||
{
|
{
|
||||||
if (not every.active) continue;
|
if (not every.active)
|
||||||
|
continue;
|
||||||
if (every.ms && every.cmd.length() && ms > every.next)
|
if (every.ms && every.cmd.length() && ms > every.next)
|
||||||
{
|
{
|
||||||
std::string cmd(every.cmd);
|
string cmd(every.cmd);
|
||||||
eval(cmd);
|
eval(cmd);
|
||||||
every.next += every.ms;
|
every.next += every.ms;
|
||||||
if (ms > every.next and ms > every.underrun)
|
if (ms > every.next and ms > every.underrun)
|
||||||
{
|
{
|
||||||
every.next += every.ms;
|
every.next += every.ms;
|
||||||
Console << yellow << "Underrun every #" << e << ", " << (ms - every.next) << "ms late" << endl;
|
Console << yellow << "Underrun every #" << e << ", " << (ms - every.next) << "ms late" << endl;
|
||||||
every.underrun = ms+5000;
|
Console.fg(white);
|
||||||
|
every.underrun = ms + 5000;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
e++;
|
e++;
|
||||||
}
|
}
|
||||||
|
|
||||||
while(blink_bits)
|
while (blink_bits)
|
||||||
{
|
{
|
||||||
if (blink_ms_on[out] and ms > blink_next[out])
|
if (blink_ms_on[out] and ms > blink_next[out])
|
||||||
{
|
{
|
||||||
@@ -841,19 +891,18 @@ void loop()
|
|||||||
}
|
}
|
||||||
blink_state[out] = not blink_state[out];
|
blink_state[out] = not blink_state[out];
|
||||||
}
|
}
|
||||||
blink_bits >>=1;
|
blink_bits >>= 1;
|
||||||
out++;
|
out++;
|
||||||
}
|
}
|
||||||
|
|
||||||
static long count;
|
|
||||||
#if defined(ESP9266)
|
#if defined(ESP9266)
|
||||||
MDNS.update();
|
MDNS.update();
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
for(auto it: brokers)
|
for (auto it : brokers)
|
||||||
it.second->loop();
|
it.second->loop();
|
||||||
|
|
||||||
for(auto it: clients)
|
for (auto it : clients)
|
||||||
it.second->loop();
|
it.second->loop();
|
||||||
|
|
||||||
Console.loop();
|
Console.loop();
|
||||||
|
|||||||
20
library.json
Normal file
20
library.json
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
{
|
||||||
|
"name": "TinyMqtt",
|
||||||
|
"keywords": [ "ethernet, mqtt, m2m, iot" ],
|
||||||
|
"description": "A lightweight MQTT library for ESP8266 and ESP32, supporting MQTT 3.1.1 with QoS and allowing to create a Mqtt broker",
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/hsaturn/TinyMqtt.git"
|
||||||
|
},
|
||||||
|
"dependencies":
|
||||||
|
{ "hsaturn/TinyConsole" : "*" },
|
||||||
|
"version": "1.1.4",
|
||||||
|
"exclude": "",
|
||||||
|
"examples": "examples/*/*.ino",
|
||||||
|
"frameworks": "arduino",
|
||||||
|
"platforms": [
|
||||||
|
"atmelavr",
|
||||||
|
"espressif8266",
|
||||||
|
"espressif32"
|
||||||
|
]
|
||||||
|
}
|
||||||
20
library.json.skeleton
Normal file
20
library.json.skeleton
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
{
|
||||||
|
"name": "#name",
|
||||||
|
"keywords": [ "ethernet, mqtt, m2m, iot" ],
|
||||||
|
"description": "#paragraph",
|
||||||
|
"repository": {
|
||||||
|
"type": "git",
|
||||||
|
"url": "https://github.com/hsaturn/TinyMqtt.git"
|
||||||
|
},
|
||||||
|
"dependencies":
|
||||||
|
{ #dependencies },
|
||||||
|
"version": "#version",
|
||||||
|
"exclude": "",
|
||||||
|
"examples": "examples/*/*.ino",
|
||||||
|
"frameworks": "arduino",
|
||||||
|
"platforms": [
|
||||||
|
"atmelavr",
|
||||||
|
"espressif8266",
|
||||||
|
"espressif32"
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -1,11 +1,11 @@
|
|||||||
name=TinyMqtt
|
name=TinyMqtt
|
||||||
version=0.9.10
|
version=1.1.4
|
||||||
author=Francois BIOT, HSaturn, <hsaturn@gmail.com>
|
author=Francois BIOT, HSaturn, <hsaturn@gmail.com>
|
||||||
maintainer=Francois BIOT, HSaturn, <hsaturn@gmail.com>
|
maintainer=Francois BIOT <hsaturn@gmail.com>
|
||||||
sentence=A tiny broker and client library for MQTT messaging.
|
sentence=A tiny broker and client library for MQTT messaging.
|
||||||
paragraph=MQTT is a lightweight messaging protocol ideal for small devices. This library allows to send and receive MQTT messages and to host a broker in your ESP 8266 and 32 WROOM. It does support MQTT 3.1.1 with QoS=0.
|
paragraph=A lightweight MQTT library for ESP8266 and ESP32, supporting MQTT 3.1.1 with QoS and allowing to create a Mqtt broker
|
||||||
category=Communication
|
category=Communication
|
||||||
url=https://github.com/hsaturn/TinyMqtt
|
url=https://github.com/hsaturn/TinyMqtt
|
||||||
architectures=*
|
architectures=*
|
||||||
depends=TinyConsole
|
depends=hsaturn/TinyConsole
|
||||||
includes=TinyMqtt.h
|
includes=TinyMqtt.h
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ class MqttClassBinder
|
|||||||
unregister(this);
|
unregister(this);
|
||||||
}
|
}
|
||||||
~MqttClassBinder() { unregister(this); }
|
~MqttClassBinder() { unregister(this); }
|
||||||
|
|
||||||
static void onUnpublished(MqttClient::CallBack handler)
|
static void onUnpublished(MqttClient::CallBack handler)
|
||||||
{
|
{
|
||||||
unrouted_handler = handler;
|
unrouted_handler = handler;
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
#include "StringIndexer.h"
|
#include "StringIndexer.h"
|
||||||
|
|
||||||
std::map<StringIndexer::index_t, StringIndexer::StringCounter> StringIndexer::strings;
|
StringIndexer::Strings StringIndexer::strings;
|
||||||
|
|
||||||
|
|||||||
@@ -1,18 +1,25 @@
|
|||||||
// vim: ts=2 sw=2 expandtab
|
// vim: ts=2 sw=2 expandtab
|
||||||
#pragma once
|
#pragma once
|
||||||
|
#include <assert.h>
|
||||||
#include <map>
|
#include <map>
|
||||||
|
#include <unordered_map>
|
||||||
|
#include "TinyConsole.h"
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
|
using string = TinyConsole::string;
|
||||||
|
|
||||||
/***
|
/***
|
||||||
* Allows to store up to 255 different strings with one byte class
|
* Allows to store up to 255 different strings with one byte class
|
||||||
* very memory efficient when one string is used many times.
|
* very memory efficient when one string is used many times.
|
||||||
*/
|
*/
|
||||||
class StringIndexer
|
class StringIndexer
|
||||||
{
|
{
|
||||||
|
private:
|
||||||
|
|
||||||
class StringCounter
|
class StringCounter
|
||||||
{
|
{
|
||||||
std::string str;
|
string str;
|
||||||
uint8_t used=0;
|
uint8_t used=0;
|
||||||
friend class StringIndexer;
|
friend class StringIndexer;
|
||||||
|
|
||||||
@@ -29,13 +36,13 @@ class StringIndexer
|
|||||||
public:
|
public:
|
||||||
using index_t = uint8_t;
|
using index_t = uint8_t;
|
||||||
|
|
||||||
static const std::string& str(const index_t& index)
|
static const string& str(const index_t& index)
|
||||||
{
|
{
|
||||||
static std::string dummy;
|
static string dummy;
|
||||||
const auto& it=strings.find(index);
|
const auto& it=strings.find(index);
|
||||||
if (it == strings.end()) return dummy;
|
if (it == strings.end()) return dummy;
|
||||||
return it->second.str;
|
return it->second.str;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void use(const index_t& index)
|
static void use(const index_t& index)
|
||||||
{
|
{
|
||||||
@@ -77,7 +84,7 @@ class StringIndexer
|
|||||||
{
|
{
|
||||||
if (strings.find(index)==strings.end())
|
if (strings.find(index)==strings.end())
|
||||||
{
|
{
|
||||||
strings[index].str = std::string(str, len);
|
strings[index].str = string(str, len);
|
||||||
strings[index].used++;
|
strings[index].used++;
|
||||||
// Serial << "Creating index " << index << " for (" << strings[index].str.c_str() << ") len=" << len << endl;
|
// Serial << "Creating index " << index << " for (" << strings[index].str.c_str() << ") len=" << len << endl;
|
||||||
return index;
|
return index;
|
||||||
@@ -86,7 +93,9 @@ class StringIndexer
|
|||||||
return 0; // TODO out of indexes
|
return 0; // TODO out of indexes
|
||||||
}
|
}
|
||||||
|
|
||||||
static std::map<index_t, StringCounter> strings;
|
using Strings = std::unordered_map<index_t, StringCounter>;
|
||||||
|
|
||||||
|
static Strings strings;
|
||||||
};
|
};
|
||||||
|
|
||||||
class IndexedString
|
class IndexedString
|
||||||
@@ -98,12 +107,14 @@ class IndexedString
|
|||||||
index = source.index;
|
index = source.index;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
IndexedString(IndexedString&& i) : index(i.index) {}
|
||||||
|
|
||||||
IndexedString(const char* str, uint8_t len)
|
IndexedString(const char* str, uint8_t len)
|
||||||
{
|
{
|
||||||
index=StringIndexer::strToIndex(str, len);
|
index=StringIndexer::strToIndex(str, len);
|
||||||
}
|
}
|
||||||
|
|
||||||
IndexedString(const std::string& str) : IndexedString(str.c_str(), str.length()) {};
|
IndexedString(const string& str) : IndexedString(str.c_str(), str.length()) {};
|
||||||
|
|
||||||
~IndexedString() { StringIndexer::release(index); }
|
~IndexedString() { StringIndexer::release(index); }
|
||||||
|
|
||||||
@@ -124,7 +135,7 @@ class IndexedString
|
|||||||
return i1.index == i2.index;
|
return i1.index == i2.index;
|
||||||
}
|
}
|
||||||
|
|
||||||
const std::string& str() const { return StringIndexer::str(index); }
|
const string& str() const { return StringIndexer::str(index); }
|
||||||
|
|
||||||
const StringIndexer::index_t& getIndex() const { return index; }
|
const StringIndexer::index_t& getIndex() const { return index; }
|
||||||
|
|
||||||
|
|||||||
283
src/TinyMqtt.cpp
283
src/TinyMqtt.cpp
@@ -9,25 +9,45 @@ static auto red = TinyConsole::red;
|
|||||||
static auto yellow = TinyConsole::yellow;
|
static auto yellow = TinyConsole::yellow;
|
||||||
|
|
||||||
int TinyMqtt::debug=2;
|
int TinyMqtt::debug=2;
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef EPOXY_DUINO
|
#ifdef EPOXY_DUINO
|
||||||
std::map<MqttMessage::Type, int> MqttClient::counters;
|
std::map<MqttMessage::Type, int> MqttClient::counters;
|
||||||
|
int MqttBroker::instances = 0;
|
||||||
|
int MqttClient::instances = 0;
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
MqttBroker::MqttBroker(uint16_t port)
|
MqttBroker::MqttBroker(uint16_t port, uint8_t max_retain_size)
|
||||||
{
|
{
|
||||||
|
debug("New broker" << port);
|
||||||
|
retain_size = max_retain_size;
|
||||||
server = new TcpServer(port);
|
server = new TcpServer(port);
|
||||||
#ifdef TINY_MQTT_ASYNC
|
#ifdef TINY_MQTT_ASYNC
|
||||||
server->onClient(onClient, this);
|
server->onClient(onClient, this);
|
||||||
#endif
|
#endif
|
||||||
|
#ifdef EPOXY_DUINO
|
||||||
|
instances++;
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
MqttBroker::~MqttBroker()
|
MqttBroker::~MqttBroker()
|
||||||
{
|
{
|
||||||
|
#ifdef EPOXY_DUINO
|
||||||
|
instances--;
|
||||||
|
#endif
|
||||||
|
closeRemoteBroker();
|
||||||
while(clients.size())
|
while(clients.size())
|
||||||
{
|
{
|
||||||
delete clients[0];
|
auto client = clients[0];
|
||||||
|
client->local_broker = nullptr;
|
||||||
|
if (client->cltFlags & MqttClient::CltFlags::CltFlagToDelete)
|
||||||
|
{
|
||||||
|
// std::cout << "Deleting client" << std::endl;
|
||||||
|
delete client;
|
||||||
|
}
|
||||||
|
clients.erase(clients.begin());
|
||||||
}
|
}
|
||||||
delete server;
|
delete server;
|
||||||
}
|
}
|
||||||
@@ -38,47 +58,56 @@ MqttClient::MqttClient(MqttBroker* local_broker, TcpClient* new_client)
|
|||||||
{
|
{
|
||||||
debug("MqttClient private with broker");
|
debug("MqttClient private with broker");
|
||||||
#ifdef TINY_MQTT_ASYNC
|
#ifdef TINY_MQTT_ASYNC
|
||||||
client = new_client;
|
tcp_client = new_client;
|
||||||
client->onData(onData, this);
|
tcp_client->onData(onData, this);
|
||||||
// client->onConnect() TODO
|
// client->onConnect() TODO
|
||||||
// client->onDisconnect() TODO
|
// client->onDisconnect() TODO
|
||||||
#else
|
#else
|
||||||
client = new WiFiClient(*new_client);
|
tcp_client = new TcpClient(*new_client);
|
||||||
#endif
|
#endif
|
||||||
#ifdef EPOXY_DUINO
|
#ifdef EPOXY_DUINO
|
||||||
alive = millis()+500000;
|
alive = millis()+500000;
|
||||||
|
instances++;
|
||||||
#else
|
#else
|
||||||
alive = millis()+5000; // TODO MAGIC client expires after 5s if no CONNECT msg
|
alive = millis()+5000; // TODO MAGIC client expires after 5s if no CONNECT msg
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
MqttClient::MqttClient(MqttBroker* local_broker, const std::string& id)
|
MqttClient::MqttClient(MqttBroker* local_broker, const string& id)
|
||||||
: local_broker(local_broker), clientId(id)
|
: local_broker(local_broker), clientId(id)
|
||||||
{
|
{
|
||||||
client = nullptr;
|
alive = 0;
|
||||||
|
keep_alive = 0;
|
||||||
|
|
||||||
if (local_broker) local_broker->addClient(this);
|
if (local_broker) local_broker->addClient(this);
|
||||||
|
#ifdef EPOXY_DUINO
|
||||||
|
instances++;
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
MqttClient::~MqttClient()
|
MqttClient::~MqttClient()
|
||||||
{
|
{
|
||||||
|
#ifdef EPOXY_DUINO
|
||||||
|
instances--;
|
||||||
|
#endif
|
||||||
close();
|
close();
|
||||||
delete client;
|
delete tcp_client;
|
||||||
|
debug("*** MqttClient delete()");
|
||||||
}
|
}
|
||||||
|
|
||||||
void MqttClient::close(bool bSendDisconnect)
|
void MqttClient::close(bool bSendDisconnect)
|
||||||
{
|
{
|
||||||
debug("close " << id().c_str());
|
debug("close " << id().c_str());
|
||||||
mqtt_connected = false;
|
resetFlag(CltFlagConnected);
|
||||||
if (client) // connected to a remote broker
|
if (tcp_client) // connected to a remote broker
|
||||||
{
|
{
|
||||||
if (bSendDisconnect and client->connected())
|
if (bSendDisconnect and tcp_client->connected())
|
||||||
{
|
{
|
||||||
message.create(MqttMessage::Type::Disconnect);
|
message.create(MqttMessage::Type::Disconnect);
|
||||||
message.hexdump("close");
|
message.hexdump("close");
|
||||||
message.sendTo(this);
|
message.sendTo(this);
|
||||||
}
|
}
|
||||||
client->stop();
|
tcp_client->stop();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (local_broker)
|
if (local_broker)
|
||||||
@@ -93,25 +122,26 @@ void MqttClient::connect(MqttBroker* local)
|
|||||||
debug("MqttClient::connect_local");
|
debug("MqttClient::connect_local");
|
||||||
close();
|
close();
|
||||||
local_broker = local;
|
local_broker = local;
|
||||||
|
local_broker->addClient(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
void MqttClient::connect(std::string broker, uint16_t port, uint16_t ka)
|
void MqttClient::connect(string broker, uint16_t port, uint16_t ka)
|
||||||
{
|
{
|
||||||
debug("MqttClient::connect_to_host " << broker << ':' << port);
|
debug("MqttClient::connect_to_host " << broker << ':' << port);
|
||||||
keep_alive = ka;
|
keep_alive = ka;
|
||||||
close();
|
close();
|
||||||
if (client) delete client;
|
if (tcp_client) delete tcp_client;
|
||||||
client = new TcpClient;
|
tcp_client = new TcpClient;
|
||||||
|
|
||||||
#ifdef TINY_MQTT_ASYNC
|
#ifdef TINY_MQTT_ASYNC
|
||||||
client->onData(onData, this);
|
tcp_client->onData(onData, this);
|
||||||
client->onConnect(onConnect, this);
|
tcp_client->onConnect(onConnect, this);
|
||||||
client->connect(broker.c_str(), port, ka);
|
tcp_client->connect(broker.c_str(), port);
|
||||||
#else
|
#else
|
||||||
if (client->connect(broker.c_str(), port))
|
if (tcp_client->connect(broker.c_str(), port))
|
||||||
{
|
{
|
||||||
debug("link established");
|
debug("link established");
|
||||||
onConnect(this, client);
|
onConnect(this, tcp_client);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@@ -126,12 +156,23 @@ void MqttBroker::addClient(MqttClient* client)
|
|||||||
clients.push_back(client);
|
clients.push_back(client);
|
||||||
}
|
}
|
||||||
|
|
||||||
void MqttBroker::connect(const std::string& host, uint16_t port)
|
void MqttBroker::closeRemoteBroker()
|
||||||
|
{
|
||||||
|
if (remote_broker)
|
||||||
|
{
|
||||||
|
delete remote_broker;
|
||||||
|
remote_broker = nullptr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void MqttBroker::connect(const string& host, uint16_t port)
|
||||||
{
|
{
|
||||||
debug("MqttBroker::connect");
|
debug("MqttBroker::connect");
|
||||||
if (broker == nullptr) broker = new MqttClient;
|
closeRemoteBroker();
|
||||||
broker->connect(host, port);
|
if (remote_broker == nullptr) remote_broker = new MqttClient;
|
||||||
broker->local_broker = this; // Because connect removed the link
|
remote_broker->connect(host, port);
|
||||||
|
remote_broker->local_broker = this; // Because connect removed the link
|
||||||
|
// TODO shouldn't we resubscribe to all client subscriptions ?
|
||||||
}
|
}
|
||||||
|
|
||||||
void MqttBroker::removeClient(MqttClient* remove)
|
void MqttBroker::removeClient(MqttClient* remove)
|
||||||
@@ -161,33 +202,32 @@ void MqttBroker::onClient(void* broker_ptr, TcpClient* client)
|
|||||||
debug("MqttBroker::onClient");
|
debug("MqttBroker::onClient");
|
||||||
MqttBroker* broker = static_cast<MqttBroker*>(broker_ptr);
|
MqttBroker* broker = static_cast<MqttBroker*>(broker_ptr);
|
||||||
|
|
||||||
broker->addClient(new MqttClient(broker, client));
|
MqttClient* mqtt = new MqttClient(broker, client);
|
||||||
|
mqtt->setFlag(MqttClient::CltFlags::CltFlagToDelete);
|
||||||
|
broker->addClient(mqtt);
|
||||||
debug("New client");
|
debug("New client");
|
||||||
}
|
}
|
||||||
|
|
||||||
void MqttBroker::loop()
|
void MqttBroker::loop()
|
||||||
{
|
{
|
||||||
#ifndef TINY_MQTT_ASYNC
|
#ifndef TINY_MQTT_ASYNC
|
||||||
WiFiClient client = server->available();
|
TcpClient client = server->accept();
|
||||||
|
|
||||||
if (client)
|
if (client)
|
||||||
{
|
{
|
||||||
onClient(this, &client);
|
onClient(this, &client);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
if (broker)
|
if (remote_broker)
|
||||||
{
|
{
|
||||||
// TODO should monitor broker's activity.
|
// TODO should monitor broker's activity.
|
||||||
// 1 When broker disconnect and reconnect we have to re-subscribe
|
// 1 When broker disconnect and reconnect we have to re-subscribe
|
||||||
broker->loop();
|
remote_broker->loop();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// for(auto it=clients.begin(); it!=clients.end(); it++)
|
|
||||||
// use index because size can change during the loop
|
|
||||||
for(size_t i=0; i<clients.size(); i++)
|
for(size_t i=0; i<clients.size(); i++)
|
||||||
{
|
{
|
||||||
auto client = clients[i];
|
MqttClient* client = clients[i];
|
||||||
if (client->connected())
|
if (client->connected())
|
||||||
{
|
{
|
||||||
client->loop();
|
client->loop();
|
||||||
@@ -202,39 +242,53 @@ void MqttBroker::loop()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
MqttError MqttBroker::subscribe(const Topic& topic, uint8_t qos)
|
// Obvioulsy called when the broker is connected to another broker.
|
||||||
|
MqttError MqttBroker::subscribe(MqttClient* client, const Topic& topic, uint8_t qos)
|
||||||
{
|
{
|
||||||
debug("MqttBroker::subscribe");
|
debug("MqttBroker::subscribe to " << topic.str() << ", retained=" << retained.size() );
|
||||||
if (broker && broker->connected())
|
for(auto& retainItem: retained)
|
||||||
{
|
{
|
||||||
return broker->subscribe(topic, qos);
|
auto &retained_topic = retainItem.first;
|
||||||
|
auto &retain = retainItem.second;
|
||||||
|
debug(" retained: " << retained_topic.str());
|
||||||
|
if (topic.matches(retained_topic))
|
||||||
|
{
|
||||||
|
debug(" -> sending");
|
||||||
|
client->publishIfSubscribed(retained_topic, retain.msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (remote_broker && remote_broker->connected())
|
||||||
|
{
|
||||||
|
return remote_broker->subscribe(topic, qos);
|
||||||
}
|
}
|
||||||
return MqttNowhereToSend;
|
return MqttNowhereToSend;
|
||||||
}
|
}
|
||||||
|
|
||||||
MqttError MqttBroker::publish(const MqttClient* source, const Topic& topic, MqttMessage& msg) const
|
MqttError MqttBroker::publish(const MqttClient* source, const Topic& topic, MqttMessage& msg)
|
||||||
{
|
{
|
||||||
MqttError retval = MqttOk;
|
MqttError retval = MqttOk;
|
||||||
|
|
||||||
|
retain(topic, msg);
|
||||||
|
|
||||||
debug("MqttBroker::publish");
|
debug("MqttBroker::publish");
|
||||||
int i=0;
|
int i=0;
|
||||||
for(auto client: clients)
|
for(auto client: clients)
|
||||||
{
|
{
|
||||||
i++;
|
i++;
|
||||||
#if TINY_MQTT_DEBUG
|
#if TINY_MQTT_DEBUG
|
||||||
Console << __LINE__ << " broker:" << (broker && broker->connected() ? "linked" : "alone") <<
|
Console << __LINE__ << " broker:" << (remote_broker && remote_broker->connected() ? "linked" : "alone") <<
|
||||||
" srce=" << (source->isLocal() ? "loc" : "rem") << " clt#" << i << ", local=" << client->isLocal() << ", con=" << client->connected() << endl;
|
" srce=" << (source->isLocal() ? "loc" : "rem") << " clt#" << i << ", local=" << client->isLocal() << ", con=" << client->connected() << endl;
|
||||||
#endif
|
#endif
|
||||||
bool doit = false;
|
bool doit = false;
|
||||||
if (broker && broker->connected()) // this (MqttBroker) is connected (to a external broker)
|
if (remote_broker && remote_broker->connected()) // this (MqttBroker) is connected (to a external broker)
|
||||||
{
|
{
|
||||||
// ext_broker -> clients or clients -> ext_broker
|
// ext_broker -> clients or clients -> ext_broker
|
||||||
if (source == broker) // external broker -> internal clients
|
if (source == remote_broker) // external broker -> internal clients
|
||||||
doit = true;
|
doit = true;
|
||||||
else // external clients -> this broker
|
else // external clients -> this broker
|
||||||
{
|
{
|
||||||
// As this broker is connected to another broker, simply forward the msg
|
// As this broker is connected to another broker, simply forward the msg
|
||||||
MqttError ret = broker->publishIfSubscribed(topic, msg);
|
MqttError ret = remote_broker->publishIfSubscribed(topic, msg);
|
||||||
if (ret != MqttOk) retval = ret;
|
if (ret != MqttOk) retval = ret;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -285,29 +339,30 @@ void MqttClient::clientAlive(uint32_t more_seconds)
|
|||||||
|
|
||||||
void MqttClient::loop()
|
void MqttClient::loop()
|
||||||
{
|
{
|
||||||
if (alive && (millis() > alive))
|
if (keep_alive && (millis() >= alive))
|
||||||
{
|
{
|
||||||
if (local_broker)
|
if (tcp_client && tcp_client->connected())
|
||||||
{
|
|
||||||
debug(red << "timeout client");
|
|
||||||
close();
|
|
||||||
debug(red << "closed");
|
|
||||||
}
|
|
||||||
else if (client && client->connected())
|
|
||||||
{
|
{
|
||||||
debug("pingreq");
|
debug("pingreq");
|
||||||
uint16_t pingreq = MqttMessage::Type::PingReq;
|
static MqttMessage pingreq(MqttMessage::Type::PingReq);
|
||||||
client->write((const char*)(&pingreq), 2);
|
pingreq.sendTo(this);
|
||||||
clientAlive(0);
|
clientAlive(0);
|
||||||
|
|
||||||
// TODO when many MqttClient passes through a local broker
|
// TODO when many MqttClient passes through a local broker
|
||||||
// there is no need to send one PingReq per instance.
|
// there is no need to send one PingReq per instance.
|
||||||
}
|
}
|
||||||
|
else if (local_broker)
|
||||||
|
{
|
||||||
|
debug(red << "timeout client");
|
||||||
|
close();
|
||||||
|
debug(red << "closed");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifndef TINY_MQTT_ASYNC
|
#ifndef TINY_MQTT_ASYNC
|
||||||
while(client && client->available()>0)
|
while(tcp_client && tcp_client->available()>0)
|
||||||
{
|
{
|
||||||
message.incoming(client->read());
|
message.incoming(tcp_client->read());
|
||||||
if (message.type())
|
if (message.type())
|
||||||
{
|
{
|
||||||
processMessage(&message);
|
processMessage(&message);
|
||||||
@@ -380,15 +435,15 @@ MqttError MqttClient::subscribe(Topic topic, uint8_t qos)
|
|||||||
debug("MqttClient::subsribe(" << topic.c_str() << ")");
|
debug("MqttClient::subsribe(" << topic.c_str() << ")");
|
||||||
MqttError ret = MqttOk;
|
MqttError ret = MqttOk;
|
||||||
|
|
||||||
subscriptions.insert(topic);
|
subscriptions.insert(topic);
|
||||||
|
|
||||||
if (local_broker==nullptr) // remote broker
|
if (local_broker==nullptr) // connected to a remote broker
|
||||||
{
|
{
|
||||||
return sendTopic(topic, MqttMessage::Type::Subscribe, qos);
|
return sendTopic(topic, MqttMessage::Type::Subscribe, qos);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
return local_broker->subscribe(topic, qos);
|
return local_broker->subscribe(this, topic, qos);
|
||||||
}
|
}
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
@@ -441,7 +496,7 @@ void MqttClient::processMessage(MqttMessage* mesg)
|
|||||||
switch(mesg->type())
|
switch(mesg->type())
|
||||||
{
|
{
|
||||||
case MqttMessage::Type::Connect:
|
case MqttMessage::Type::Connect:
|
||||||
if (mqtt_connected)
|
if (mqtt_connected())
|
||||||
{
|
{
|
||||||
debug("already connected");
|
debug("already connected");
|
||||||
break;
|
break;
|
||||||
@@ -456,13 +511,13 @@ void MqttClient::processMessage(MqttMessage* mesg)
|
|||||||
}
|
}
|
||||||
if (header[6]!=0x04)
|
if (header[6]!=0x04)
|
||||||
{
|
{
|
||||||
debug("unknown level");
|
debug("Unsupported MQTT version (" << (int) header[6] << "), only version=4 supported" << endl);
|
||||||
break; // Level 3.1.1
|
break; // Level 3.1.1
|
||||||
}
|
}
|
||||||
|
|
||||||
// ClientId
|
// ClientId
|
||||||
mesg->getString(payload, len);
|
mesg->getString(payload, len);
|
||||||
clientId = std::string(payload, len);
|
clientId = string(payload, len);
|
||||||
payload += len;
|
payload += len;
|
||||||
|
|
||||||
if (mqtt_flags & FlagWill) // Will topic
|
if (mqtt_flags & FlagWill) // Will topic
|
||||||
@@ -491,24 +546,24 @@ void MqttClient::processMessage(MqttMessage* mesg)
|
|||||||
Console << yellow << "Client " << clientId << " connected : keep alive=" << keep_alive << '.' << white << endl;
|
Console << yellow << "Client " << clientId << " connected : keep alive=" << keep_alive << '.' << white << endl;
|
||||||
#endif
|
#endif
|
||||||
bclose = false;
|
bclose = false;
|
||||||
mqtt_connected=true;
|
setFlag(CltFlagConnected);
|
||||||
{
|
{
|
||||||
MqttMessage msg(MqttMessage::Type::ConnAck);
|
MqttMessage msg(MqttMessage::Type::ConnAck);
|
||||||
msg.add(0); // Session present (not implemented)
|
msg.add(0); // Session present (not implemented)
|
||||||
msg.add(0); // Connection accepted
|
msg.add(0); // Connection accepted
|
||||||
msg.sendTo(this);
|
msg.sendTo(this);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case MqttMessage::Type::ConnAck:
|
case MqttMessage::Type::ConnAck:
|
||||||
mqtt_connected = true;
|
setFlag(CltFlagConnected);
|
||||||
bclose = false;
|
bclose = false;
|
||||||
resubscribe();
|
resubscribe();
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case MqttMessage::Type::SubAck:
|
case MqttMessage::Type::SubAck:
|
||||||
case MqttMessage::Type::PubAck:
|
case MqttMessage::Type::PubAck:
|
||||||
if (!mqtt_connected) break;
|
if (not mqtt_connected()) break;
|
||||||
// Ignore acks
|
// Ignore acks
|
||||||
bclose = false;
|
bclose = false;
|
||||||
break;
|
break;
|
||||||
@@ -519,12 +574,12 @@ void MqttClient::processMessage(MqttMessage* mesg)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case MqttMessage::Type::PingReq:
|
case MqttMessage::Type::PingReq:
|
||||||
if (!mqtt_connected) break;
|
if (not mqtt_connected()) break;
|
||||||
if (client)
|
if (tcp_client)
|
||||||
{
|
{
|
||||||
uint16_t pingreq = MqttMessage::Type::PingResp;
|
uint16_t pingreq = MqttMessage::Type::PingResp;
|
||||||
debug(cyan << "Ping response to client ");
|
debug(cyan << "Ping response to client ");
|
||||||
client->write((const char*)(&pingreq), 2);
|
tcp_client->write((const char*)(&pingreq), 2);
|
||||||
bclose = false;
|
bclose = false;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@@ -536,15 +591,15 @@ void MqttClient::processMessage(MqttMessage* mesg)
|
|||||||
case MqttMessage::Type::Subscribe:
|
case MqttMessage::Type::Subscribe:
|
||||||
case MqttMessage::Type::UnSubscribe:
|
case MqttMessage::Type::UnSubscribe:
|
||||||
{
|
{
|
||||||
if (!mqtt_connected) break;
|
if (not mqtt_connected()) break;
|
||||||
payload = header+2;
|
payload = header+2;
|
||||||
|
|
||||||
debug("un/subscribe loop");
|
debug("un/subscribe loop");
|
||||||
std::string qoss;
|
string qoss;
|
||||||
while(payload < mesg->end())
|
while(payload < mesg->end())
|
||||||
{
|
{
|
||||||
mesg->getString(payload, len); // Topic
|
mesg->getString(payload, len); // Topic
|
||||||
debug( " topic (" << std::string(payload, len) << ')');
|
debug( " topic (" << string(payload, len) << ')');
|
||||||
// subscribe(Topic(payload, len));
|
// subscribe(Topic(payload, len));
|
||||||
Topic topic(payload, len);
|
Topic topic(payload, len);
|
||||||
|
|
||||||
@@ -559,7 +614,7 @@ void MqttClient::processMessage(MqttMessage* mesg)
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
qoss.push_back(qos);
|
qoss.push_back(qos);
|
||||||
subscriptions.insert(topic);
|
subscribe(topic);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@@ -580,17 +635,18 @@ void MqttClient::processMessage(MqttMessage* mesg)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case MqttMessage::Type::UnSuback:
|
case MqttMessage::Type::UnSuback:
|
||||||
if (!mqtt_connected) break;
|
if (not mqtt_connected()) break;
|
||||||
bclose = false;
|
bclose = false;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case MqttMessage::Type::Publish:
|
case MqttMessage::Type::Publish:
|
||||||
#if TINY_MQTT_DEBUG
|
#if TINY_MQTT_DEBUG
|
||||||
Console << "publish " << mqtt_connected << '/' << (long) client << endl;
|
Console << "publish " << mqtt_connected() << '/' << (long) tcp_client << endl;
|
||||||
#endif
|
#endif
|
||||||
if (mqtt_connected or client == nullptr)
|
if (mqtt_connected() or tcp_client == nullptr)
|
||||||
{
|
{
|
||||||
uint8_t qos = mesg->flags();
|
uint8_t qos = mesg->flags();
|
||||||
|
qos = (qos / 2) & 3;
|
||||||
payload = header;
|
payload = header;
|
||||||
mesg->getString(payload, len);
|
mesg->getString(payload, len);
|
||||||
Topic published(payload, len);
|
Topic published(payload, len);
|
||||||
@@ -598,24 +654,35 @@ void MqttClient::processMessage(MqttMessage* mesg)
|
|||||||
#if TINY_MQTT_DEBUG
|
#if TINY_MQTT_DEBUG
|
||||||
Console << "Received Publish (" << published.str().c_str() << ") size=" << (int)len << endl;
|
Console << "Received Publish (" << published.str().c_str() << ") size=" << (int)len << endl;
|
||||||
#endif
|
#endif
|
||||||
// << '(' << std::string(payload, len).c_str() << ')' << " msglen=" << mesg->length() << endl;
|
|
||||||
if (qos) payload+=2; // ignore packet identifier if any
|
const char* ID; // remove PublishID() to avoid misuse
|
||||||
|
if (qos) {
|
||||||
|
ID = payload;
|
||||||
|
payload+=2; // ignore packet identifier if any
|
||||||
|
}
|
||||||
len=mesg->end()-payload;
|
len=mesg->end()-payload;
|
||||||
|
if (qos == 1)
|
||||||
|
{
|
||||||
|
MqttMessage msg(MqttMessage::Type::PubAck);
|
||||||
|
msg.add(ID[0]); // MessageID high
|
||||||
|
msg.add(ID[1]); // MessageID low
|
||||||
|
msg.sendTo(this);
|
||||||
|
}
|
||||||
// TODO reset DUP
|
// TODO reset DUP
|
||||||
// TODO reset RETAIN
|
// TODO reset RETAIN
|
||||||
|
|
||||||
if (local_broker==nullptr or client==nullptr) // internal MqttClient receives publish
|
if (local_broker==nullptr or tcp_client==nullptr) // internal MqttClient receives publish
|
||||||
{
|
{
|
||||||
#if TINY_MQTT_DEBUG
|
#if TINY_MQTT_DEBUG
|
||||||
if (TinyMqtt::debug >= 2)
|
if (TinyMqtt::debug >= 2)
|
||||||
{
|
{
|
||||||
Console << (isSubscribedTo(published) ? "not" : "") << " subscribed.\n";
|
Console << (isSubscribedTo(published) ? "not" : "") << " subscribed.\r\n";
|
||||||
Console << "has " << (callback ? "" : "no ") << " callback.\n";
|
Console << "has " << (callback ? "" : "no ") << " callback.\r\n";
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
if (callback and isSubscribedTo(published))
|
if (callback and isSubscribedTo(published))
|
||||||
{
|
{
|
||||||
callback(this, published, payload, len); // TODO send the real payload
|
callback(this, published, payload, len);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (local_broker) // from outside to inside
|
else if (local_broker) // from outside to inside
|
||||||
@@ -629,8 +696,8 @@ void MqttClient::processMessage(MqttMessage* mesg)
|
|||||||
|
|
||||||
case MqttMessage::Type::Disconnect:
|
case MqttMessage::Type::Disconnect:
|
||||||
// TODO should discard any will msg
|
// TODO should discard any will msg
|
||||||
if (!mqtt_connected) break;
|
if (not mqtt_connected()) break;
|
||||||
mqtt_connected = false;
|
resetFlag(CltFlagConnected);
|
||||||
close(false);
|
close(false);
|
||||||
bclose=false;
|
bclose=false;
|
||||||
break;
|
break;
|
||||||
@@ -719,9 +786,9 @@ bool Topic::matches(const Topic& topic) const
|
|||||||
|
|
||||||
|
|
||||||
// publish from local client
|
// publish from local client
|
||||||
MqttError MqttClient::publish(const Topic& topic, const char* payload, size_t pay_length)
|
MqttError MqttClient::publish(const Topic& topic, const char* payload, size_t pay_length, bool retain)
|
||||||
{
|
{
|
||||||
MqttMessage msg(MqttMessage::Publish);
|
MqttMessage msg(MqttMessage::Publish, retain ? 1 : 0);
|
||||||
msg.add(topic);
|
msg.add(topic);
|
||||||
msg.add(payload, pay_length, false);
|
msg.add(payload, pay_length, false);
|
||||||
msg.complete();
|
msg.complete();
|
||||||
@@ -730,7 +797,7 @@ MqttError MqttClient::publish(const Topic& topic, const char* payload, size_t pa
|
|||||||
{
|
{
|
||||||
return local_broker->publish(this, topic, msg);
|
return local_broker->publish(this, topic, msg);
|
||||||
}
|
}
|
||||||
else if (client)
|
else if (tcp_client and connected())
|
||||||
return msg.sendTo(this);
|
return msg.sendTo(this);
|
||||||
else
|
else
|
||||||
return MqttNowhereToSend;
|
return MqttNowhereToSend;
|
||||||
@@ -744,7 +811,7 @@ MqttError MqttClient::publishIfSubscribed(const Topic& topic, MqttMessage& msg)
|
|||||||
debug("mqttclient publishIfSubscribed " << topic.c_str() << ' ' << subscriptions.size());
|
debug("mqttclient publishIfSubscribed " << topic.c_str() << ' ' << subscriptions.size());
|
||||||
if (isSubscribedTo(topic))
|
if (isSubscribedTo(topic))
|
||||||
{
|
{
|
||||||
if (client)
|
if (tcp_client)
|
||||||
retval = msg.sendTo(this);
|
retval = msg.sendTo(this);
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@@ -862,7 +929,7 @@ void MqttMessage::encodeLength()
|
|||||||
buffer[2] = (length >> 7);
|
buffer[2] = (length >> 7);
|
||||||
vheader = 3;
|
vheader = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
// We could check that buffer[2] < 128 (end of length encoding)
|
// We could check that buffer[2] < 128 (end of length encoding)
|
||||||
state = Complete;
|
state = Complete;
|
||||||
}
|
}
|
||||||
@@ -885,12 +952,46 @@ MqttError MqttMessage::sendTo(MqttClient* client)
|
|||||||
return MqttOk;
|
return MqttOk;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void MqttBroker::retainDrop()
|
||||||
|
{
|
||||||
|
if (retained.size() >= retain_size)
|
||||||
|
{
|
||||||
|
std::map<Topic, Retain>::iterator oldest = retained.begin();
|
||||||
|
auto it = oldest;
|
||||||
|
while(++it != retained.end())
|
||||||
|
{
|
||||||
|
if (oldest->second.timestamp > it->second.timestamp)
|
||||||
|
oldest = it;
|
||||||
|
}
|
||||||
|
retained.erase(oldest);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void MqttBroker::retain(const Topic& topic, const MqttMessage& msg)
|
||||||
|
{
|
||||||
|
debug("MqttBroker::retain msg_type=" << _HEX(msg.type()) << ", retain_size=" << retain_size);
|
||||||
|
if (retain_size==0 or msg.type() != MqttMessage::Publish) return;
|
||||||
|
if (msg.flags() & 1) // flag RETAIN
|
||||||
|
{
|
||||||
|
debug(" retaining " << topic.str());
|
||||||
|
auto old = retained.find(topic);
|
||||||
|
if (old == retained.end())
|
||||||
|
retainDrop();
|
||||||
|
else
|
||||||
|
retained.erase(old);
|
||||||
|
// FIXME if payload size == 0 remove message from retained
|
||||||
|
Retain r(micros(), msg);
|
||||||
|
r.msg.retained();
|
||||||
|
retained.insert({ topic, std::move(r)});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void MqttMessage::hexdump(const char* prefix) const
|
void MqttMessage::hexdump(const char* prefix) const
|
||||||
{
|
{
|
||||||
(void)prefix;
|
(void)prefix;
|
||||||
#if TINY_MQTT_DEBUG
|
#if TINY_MQTT_DEBUG
|
||||||
if (TinyMqtt::debug<2) return;
|
if (TinyMqtt::debug<2) return;
|
||||||
static std::map<Type, std::string> tts={
|
static std::map<Type, string> tts={
|
||||||
{ Connect, "Connect" },
|
{ Connect, "Connect" },
|
||||||
{ ConnAck, "Connack" },
|
{ ConnAck, "Connack" },
|
||||||
{ Publish, "Publish" },
|
{ Publish, "Publish" },
|
||||||
@@ -903,7 +1004,7 @@ void MqttMessage::hexdump(const char* prefix) const
|
|||||||
{ PingResp, "Pingresp" },
|
{ PingResp, "Pingresp" },
|
||||||
{ Disconnect, "Disconnect" }
|
{ Disconnect, "Disconnect" }
|
||||||
};
|
};
|
||||||
std::string t("Unknown");
|
string t("Unknown");
|
||||||
Type typ=static_cast<Type>(buffer[0] & 0xF0);
|
Type typ=static_cast<Type>(buffer[0] & 0xF0);
|
||||||
if (tts.find(typ) != tts.end())
|
if (tts.find(typ) != tts.end())
|
||||||
t=tts[typ];
|
t=tts[typ];
|
||||||
@@ -920,7 +1021,7 @@ void MqttMessage::hexdump(const char* prefix) const
|
|||||||
const char* hex_to_str = " | ";
|
const char* hex_to_str = " | ";
|
||||||
const char* separator = hex_to_str;
|
const char* separator = hex_to_str;
|
||||||
const char* half_sep = " - ";
|
const char* half_sep = " - ";
|
||||||
std::string ascii;
|
string ascii;
|
||||||
|
|
||||||
Console << prefix << " size(" << buffer.size() << "), state=" << state << endl;
|
Console << prefix << " size(" << buffer.size() << "), state=" << state << endl;
|
||||||
|
|
||||||
|
|||||||
153
src/TinyMqtt.h
153
src/TinyMqtt.h
@@ -1,6 +1,8 @@
|
|||||||
// vim: ts=2 sw=2 expandtab
|
// vim: ts=2 sw=2 expandtab
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#define TINY_MQTT_REVISION "1.1.3"
|
||||||
|
|
||||||
#ifndef TINY_MQTT_DEBUG
|
#ifndef TINY_MQTT_DEBUG
|
||||||
#define TINY_MQTT_DEBUG 0
|
#define TINY_MQTT_DEBUG 0
|
||||||
#endif
|
#endif
|
||||||
@@ -8,7 +10,10 @@
|
|||||||
// TODO Should add a AUnit with both TINY_MQTT_ASYNC and not TINY_MQTT_ASYNC
|
// TODO Should add a AUnit with both TINY_MQTT_ASYNC and not TINY_MQTT_ASYNC
|
||||||
// #define TINY_MQTT_ASYNC // Uncomment this to use ESPAsyncTCP instead of normal cnx
|
// #define TINY_MQTT_ASYNC // Uncomment this to use ESPAsyncTCP instead of normal cnx
|
||||||
|
|
||||||
#if defined(ESP8266) || defined(EPOXY_DUINO)
|
|
||||||
|
#if defined(TINY_MQTT_ETHERNET)
|
||||||
|
#include <Ethernet.h>
|
||||||
|
#elif defined(ESP8266) || defined(EPOXY_DUINO)
|
||||||
#ifdef TINY_MQTT_ASYNC
|
#ifdef TINY_MQTT_ASYNC
|
||||||
#include <ESPAsyncTCP.h>
|
#include <ESPAsyncTCP.h>
|
||||||
#else
|
#else
|
||||||
@@ -19,6 +24,8 @@
|
|||||||
#ifdef TINY_MQTT_ASYNC
|
#ifdef TINY_MQTT_ASYNC
|
||||||
#include <AsyncTCP.h> // https://github.com/me-no-dev/AsyncTCP
|
#include <AsyncTCP.h> // https://github.com/me-no-dev/AsyncTCP
|
||||||
#endif
|
#endif
|
||||||
|
#elif defined(ARDUINO_ARCH_RP2040)
|
||||||
|
#include <WiFi.h> // works with Raspberry Pi Pico W, earlephilhower
|
||||||
#endif
|
#endif
|
||||||
#ifdef EPOXY_DUINO
|
#ifdef EPOXY_DUINO
|
||||||
#define dbg_ptr uint64_t
|
#define dbg_ptr uint64_t
|
||||||
@@ -53,12 +60,17 @@
|
|||||||
#define debug(what) {}
|
#define debug(what) {}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef TINY_MQTT_ASYNC
|
#if defined(TINY_MQTT_ETHERNET)
|
||||||
using TcpClient = AsyncClient;
|
using TcpClient = EthernetClient;
|
||||||
using TcpServer = AsyncServer;
|
using TcpServer = EthernetServer;
|
||||||
#else
|
#else
|
||||||
using TcpClient = WiFiClient;
|
#ifdef TINY_MQTT_ASYNC
|
||||||
using TcpServer = WiFiServer;
|
using TcpClient = AsyncClient;
|
||||||
|
using TcpServer = AsyncServer;
|
||||||
|
#else
|
||||||
|
using TcpClient = WiFiClient;
|
||||||
|
using TcpServer = WiFiServer;
|
||||||
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
enum __attribute__((packed)) MqttError
|
enum __attribute__((packed)) MqttError
|
||||||
@@ -68,12 +80,15 @@ enum __attribute__((packed)) MqttError
|
|||||||
MqttInvalidMessage=2,
|
MqttInvalidMessage=2,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
using string = TinyConsole::string;
|
||||||
|
|
||||||
class Topic : public IndexedString
|
class Topic : public IndexedString
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
|
Topic(const string& m) : IndexedString(m){}
|
||||||
Topic(const char* s, uint8_t len) : IndexedString(s,len){}
|
Topic(const char* s, uint8_t len) : IndexedString(s,len){}
|
||||||
Topic(const char* s) : Topic(s, strlen(s)) {}
|
Topic(const char* s) : Topic(s, strlen(s)) {}
|
||||||
Topic(const std::string s) : Topic(s.c_str(), s.length()){};
|
// Topic(const string s) : Topic(s.c_str(), s.length()){};
|
||||||
|
|
||||||
const char* c_str() const { return str().c_str(); }
|
const char* c_str() const { return str().c_str(); }
|
||||||
|
|
||||||
@@ -118,15 +133,19 @@ class MqttMessage
|
|||||||
return (*bun << 8) | bun[1]; }
|
return (*bun << 8) | bun[1]; }
|
||||||
|
|
||||||
MqttMessage() { reset(); }
|
MqttMessage() { reset(); }
|
||||||
MqttMessage(Type t, uint8_t bits_d3_d0=0) { create(t); buffer[0] |= bits_d3_d0; }
|
MqttMessage(Type t, uint8_t bits_d3_d0=0) { create(t); buffer[0] |= (bits_d3_d0 & 0xF); }
|
||||||
|
MqttMessage(const MqttMessage& m)
|
||||||
|
: buffer(m.buffer), vheader(m.vheader), size(m.size), state(m.state) {}
|
||||||
|
|
||||||
void incoming(char byte);
|
void incoming(char byte);
|
||||||
void add(char byte) { incoming(byte); }
|
void add(char byte) { incoming(byte); }
|
||||||
void add(const char* p, size_t len, bool addLength=true );
|
void add(const char* p, size_t len, bool addLength=true );
|
||||||
void add(const std::string& s) { add(s.c_str(), s.length()); }
|
void add(const string& s) { add(s.c_str(), s.length()); }
|
||||||
void add(const Topic& t) { add(t.str()); }
|
void add(const Topic& t) { add(t.str()); }
|
||||||
const char* end() const { return &buffer[0]+buffer.size(); }
|
const char* end() const { return &buffer[0]+buffer.size(); }
|
||||||
const char* getVHeader() const { return &buffer[vheader]; }
|
const char* getVHeader() const { return &buffer[vheader]; }
|
||||||
void complete() { encodeLength(); }
|
void complete() { encodeLength(); }
|
||||||
|
void retained() { if ((buffer[0] & 0xF)==Publish) buffer[0] |= 1; }
|
||||||
|
|
||||||
void reset();
|
void reset();
|
||||||
|
|
||||||
@@ -153,10 +172,19 @@ class MqttMessage
|
|||||||
MqttError sendTo(MqttClient*);
|
MqttError sendTo(MqttClient*);
|
||||||
void hexdump(const char* prefix=nullptr) const;
|
void hexdump(const char* prefix=nullptr) const;
|
||||||
|
|
||||||
|
MqttMessage& operator = (MqttMessage&& m)
|
||||||
|
{
|
||||||
|
buffer = std::move(m.buffer);
|
||||||
|
vheader = m.vheader;
|
||||||
|
size = m.size;
|
||||||
|
state = m.state;
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void encodeLength();
|
void encodeLength();
|
||||||
|
|
||||||
std::string buffer;
|
string buffer;
|
||||||
uint8_t vheader;
|
uint8_t vheader;
|
||||||
uint16_t size; // bytes left to receive
|
uint16_t size; // bytes left to receive
|
||||||
State state;
|
State state;
|
||||||
@@ -175,34 +203,44 @@ class MqttClient
|
|||||||
FlagCleanSession = 2, // unsupported
|
FlagCleanSession = 2, // unsupported
|
||||||
FlagReserved = 1
|
FlagReserved = 1
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum __attribute__((packed)) CltFlags
|
||||||
|
{
|
||||||
|
CltFlagNone = 0,
|
||||||
|
CltFlagConnected = 1,
|
||||||
|
CltFlagToDelete = 2
|
||||||
|
};
|
||||||
public:
|
public:
|
||||||
|
|
||||||
using CallBack = void (*)(const MqttClient* source, const Topic& topic, const char* payload, size_t payload_length);
|
using CallBack = void (*)(const MqttClient* source, const Topic& topic, const char* payload, size_t payload_length);
|
||||||
|
|
||||||
/** Constructor. Broker is the adress of a local broker if not null
|
/** Constructor. Broker is the adress of a local broker if not null
|
||||||
If you want to connect elsewhere, leave broker null and use connect() **/
|
If you want to connect elsewhere, leave broker null and use connect() **/
|
||||||
MqttClient(MqttBroker* broker = nullptr, const std::string& id = TINY_MQTT_DEFAULT_CLIENT_ID);
|
MqttClient(MqttBroker* broker = nullptr, const string& id = TINY_MQTT_DEFAULT_CLIENT_ID);
|
||||||
MqttClient(const std::string& id) : MqttClient(nullptr, id){}
|
MqttClient(const string& id) : MqttClient(nullptr, id){}
|
||||||
|
|
||||||
~MqttClient();
|
~MqttClient();
|
||||||
|
|
||||||
void connect(MqttBroker* local_broker);
|
void connect(MqttBroker* local_broker);
|
||||||
void connect(std::string broker, uint16_t port, uint16_t keep_alive = 10);
|
void connect(string broker, uint16_t port = 1883, uint16_t keep_alive = 10);
|
||||||
|
void connect(const IPAddress& ip, uint16_t port = 1883, uint16_t keep_alive = 10)
|
||||||
|
{ connect(ip.toString().c_str(), port, keep_alive); }
|
||||||
|
|
||||||
// TODO it seems that connected returns true in tcp mode even if
|
// TODO it seems that connected returns true in tcp mode even if
|
||||||
// no negociation occurred
|
// no negociation occurred
|
||||||
bool connected()
|
bool connected()
|
||||||
{
|
{
|
||||||
return (local_broker!=nullptr and client==nullptr) or (client and client->connected());
|
return (local_broker!=nullptr and tcp_client==nullptr)
|
||||||
|
or (tcp_client and tcp_client->connected());
|
||||||
}
|
}
|
||||||
|
|
||||||
void write(const char* buf, size_t length)
|
void write(const char* buf, size_t length)
|
||||||
{
|
{
|
||||||
if (client) client->write(buf, length);
|
if (tcp_client) tcp_client->write(buf, length);
|
||||||
}
|
}
|
||||||
|
|
||||||
const std::string& id() const { return clientId; }
|
const string& id() const { return clientId; }
|
||||||
void id(const std::string& new_id) { clientId = new_id; }
|
void id(const string& new_id) { clientId = new_id; }
|
||||||
|
|
||||||
/** Should be called in main loop() */
|
/** Should be called in main loop() */
|
||||||
void loop();
|
void loop();
|
||||||
@@ -217,11 +255,11 @@ class MqttClient
|
|||||||
};
|
};
|
||||||
|
|
||||||
// Publish from client to the world
|
// Publish from client to the world
|
||||||
MqttError publish(const Topic&, const char* payload, size_t pay_length);
|
MqttError publish(const Topic&, const char* payload, size_t pay_length, bool retain=false);
|
||||||
MqttError publish(const Topic& t, const char* payload) { return publish(t, payload, strlen(payload)); }
|
MqttError publish(const Topic& t, const char* payload, bool retain=false) { return publish(t, payload, strlen(payload), retain); }
|
||||||
MqttError publish(const Topic& t, const String& s) { return publish(t, s.c_str(), s.length()); }
|
MqttError publish(const Topic& t, const String& s, bool retain=false) { return publish(t, s.c_str(), s.length(), retain); }
|
||||||
MqttError publish(const Topic& t, const std::string& s) { return publish(t,s.c_str(),s.length());}
|
MqttError publish(const Topic& t, const string& s, bool retain=false) { return publish(t,s.c_str(),s.length(), retain);}
|
||||||
MqttError publish(const Topic& t) { return publish(t, nullptr, 0);};
|
MqttError publish(const Topic& t, bool retain=false) { return publish(t, nullptr, 0, retain);};
|
||||||
|
|
||||||
MqttError subscribe(Topic topic, uint8_t qos=0);
|
MqttError subscribe(Topic topic, uint8_t qos=0);
|
||||||
MqttError unsubscribe(Topic topic);
|
MqttError unsubscribe(Topic topic);
|
||||||
@@ -229,18 +267,18 @@ class MqttClient
|
|||||||
|
|
||||||
// connected to local broker
|
// connected to local broker
|
||||||
// TODO seems to be useless
|
// TODO seems to be useless
|
||||||
bool isLocal() const { return client == nullptr; }
|
bool isLocal() const { return tcp_client == nullptr; }
|
||||||
|
|
||||||
void dump(std::string indent="")
|
void dump(string indent="")
|
||||||
{
|
{
|
||||||
(void)indent;
|
(void)indent;
|
||||||
#if TINY_MQTT_DEBUG
|
#if TINY_MQTT_DEBUG
|
||||||
uint32_t ms=millis();
|
uint32_t ms=millis();
|
||||||
Console << indent << "+-- " << '\'' << clientId.c_str() << "' " << (connected() ? " ON " : " OFF");
|
Console << indent << "+-- " << '\'' << clientId.c_str() << "' " << (connected() ? " ON " : " OFF");
|
||||||
Console << ", alive=" << alive << '/' << ms << ", ka=" << keep_alive << ' ';
|
Console << ", alive=" << alive << '/' << ms << ", ka=" << keep_alive << ' ';
|
||||||
if (client)
|
if (tcp_client)
|
||||||
{
|
{
|
||||||
if (client->connected())
|
if (tcp_client->connected())
|
||||||
Console << TinyConsole::green << "connected";
|
Console << TinyConsole::green << "connected";
|
||||||
else
|
else
|
||||||
Console << TinyConsole::red << "disconnected";
|
Console << TinyConsole::red << "disconnected";
|
||||||
@@ -264,10 +302,14 @@ class MqttClient
|
|||||||
|
|
||||||
#ifdef EPOXY_DUINO
|
#ifdef EPOXY_DUINO
|
||||||
static std::map<MqttMessage::Type, int> counters; // Number of processed messages
|
static std::map<MqttMessage::Type, int> counters; // Number of processed messages
|
||||||
|
static int instances;
|
||||||
#endif
|
#endif
|
||||||
uint32_t keepAlive() const { return keep_alive; }
|
uint32_t keepAlive() const { return keep_alive; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
bool mqtt_connected() const { return cltFlags & CltFlagConnected; }
|
||||||
|
void setFlag(CltFlags f) { cltFlags |= f; }
|
||||||
|
void resetFlag(CltFlags f) { cltFlags &= ~f; }
|
||||||
|
|
||||||
// event when tcp/ip link established (real or fake)
|
// event when tcp/ip link established (real or fake)
|
||||||
static void onConnect(void * client_ptr, TcpClient*);
|
static void onConnect(void * client_ptr, TcpClient*);
|
||||||
@@ -285,7 +327,7 @@ class MqttClient
|
|||||||
void clientAlive(uint32_t more_seconds);
|
void clientAlive(uint32_t more_seconds);
|
||||||
void processMessage(MqttMessage* message);
|
void processMessage(MqttMessage* message);
|
||||||
|
|
||||||
bool mqtt_connected = false;
|
uint8_t cltFlags = CltFlagNone;
|
||||||
char mqtt_flags;
|
char mqtt_flags;
|
||||||
uint32_t keep_alive = 30;
|
uint32_t keep_alive = 30;
|
||||||
uint32_t alive;
|
uint32_t alive;
|
||||||
@@ -295,40 +337,42 @@ class MqttClient
|
|||||||
// when MqttBroker uses MqttClient for each external connexion
|
// when MqttBroker uses MqttClient for each external connexion
|
||||||
MqttBroker* local_broker=nullptr;
|
MqttBroker* local_broker=nullptr;
|
||||||
|
|
||||||
TcpClient* client=nullptr; // connection to remote broker
|
TcpClient* tcp_client=nullptr; // connection to remote broker
|
||||||
std::set<Topic> subscriptions;
|
std::set<Topic> subscriptions;
|
||||||
std::string clientId;
|
string clientId;
|
||||||
CallBack callback = nullptr;
|
CallBack callback = nullptr;
|
||||||
};
|
};
|
||||||
|
|
||||||
class MqttBroker
|
class MqttBroker
|
||||||
{
|
{
|
||||||
enum __attribute__((packed)) State
|
|
||||||
{
|
|
||||||
Disconnected, // Also the initial state
|
|
||||||
Connecting, // connect and sends a fake publish to avoid circular cnx
|
|
||||||
Connected, // this->broker is connected and circular cnx avoided
|
|
||||||
};
|
|
||||||
public:
|
public:
|
||||||
// TODO limit max number of clients
|
// TODO limit max number of clients
|
||||||
MqttBroker(uint16_t port);
|
MqttBroker(uint16_t port, uint8_t retain_size=0);
|
||||||
~MqttBroker();
|
~MqttBroker();
|
||||||
|
|
||||||
void begin() { server->begin(); }
|
void begin() { server->begin(); }
|
||||||
void loop();
|
void loop();
|
||||||
|
|
||||||
void connect(const std::string& host, uint16_t port=1883);
|
/** Connect the broker to a parent broker */
|
||||||
bool connected() const { return state == Connected; }
|
void connect(const string& host, uint16_t port=1883);
|
||||||
|
/** returns true if connected to another broker */
|
||||||
|
bool connected() const { return remote_broker ? remote_broker->connected() : false; }
|
||||||
|
|
||||||
size_t clientsCount() const { return clients.size(); }
|
size_t clientsCount() const { return clients.size(); }
|
||||||
|
uint8_t retain() { return retain_size; }
|
||||||
|
void retain(uint8_t size) { retain_size = size; if (size==0) retained.clear(); }
|
||||||
|
uint8_t retainCount() const { return retained.size(); }
|
||||||
|
|
||||||
void dump(std::string indent="")
|
void dump(string indent="")
|
||||||
{
|
{
|
||||||
for(auto client: clients)
|
for(auto client: clients)
|
||||||
client->dump(indent);
|
client->dump(indent);
|
||||||
}
|
}
|
||||||
|
|
||||||
const std::vector<MqttClient*> getClients() const { return clients; }
|
const std::vector<MqttClient*> getClients() const { return clients; }
|
||||||
|
#ifdef EPOXY_DUINO
|
||||||
|
static int instances;
|
||||||
|
#endif
|
||||||
|
|
||||||
private:
|
private:
|
||||||
friend class MqttClient;
|
friend class MqttClient;
|
||||||
@@ -340,10 +384,9 @@ class MqttBroker
|
|||||||
bool checkPassword(const char* password, uint8_t len) const
|
bool checkPassword(const char* password, uint8_t len) const
|
||||||
{ return compareString(auth_password, password, len); }
|
{ return compareString(auth_password, password, len); }
|
||||||
|
|
||||||
|
MqttError publish(const MqttClient* source, const Topic& topic, MqttMessage& msg);
|
||||||
|
|
||||||
MqttError publish(const MqttClient* source, const Topic& topic, MqttMessage& msg) const;
|
MqttError subscribe(MqttClient*, const Topic& topic, uint8_t qos);
|
||||||
|
|
||||||
MqttError subscribe(const Topic& topic, uint8_t qos);
|
|
||||||
|
|
||||||
// For clients that are added not by the broker itself (local clients)
|
// For clients that are added not by the broker itself (local clients)
|
||||||
void addClient(MqttClient* client);
|
void addClient(MqttClient* client);
|
||||||
@@ -357,7 +400,29 @@ class MqttBroker
|
|||||||
|
|
||||||
const char* auth_user = "guest";
|
const char* auth_user = "guest";
|
||||||
const char* auth_password = "guest";
|
const char* auth_password = "guest";
|
||||||
MqttClient* broker = nullptr;
|
MqttClient* remote_broker = nullptr;
|
||||||
|
|
||||||
State state = Disconnected;
|
void closeRemoteBroker();
|
||||||
|
|
||||||
|
void retain(const Topic& topic, const MqttMessage& msg);
|
||||||
|
void retainDrop();
|
||||||
|
|
||||||
|
struct Retain
|
||||||
|
{
|
||||||
|
Retain(unsigned long ts, const MqttMessage& m) : timestamp(ts), msg(m) {}
|
||||||
|
Retain(Retain&& r) : timestamp(r.timestamp), msg(std::move(r.msg)) {}
|
||||||
|
|
||||||
|
Retain& operator=(Retain&& r)
|
||||||
|
{
|
||||||
|
timestamp = r.timestamp;
|
||||||
|
msg = std::move(r.msg);
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
unsigned long timestamp;
|
||||||
|
MqttMessage msg;
|
||||||
|
};
|
||||||
|
|
||||||
|
std::map<Topic, Retain> retained;
|
||||||
|
uint8_t retain_size;
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,28 +1,40 @@
|
|||||||
|
SUB=
|
||||||
|
|
||||||
|
all:runtests
|
||||||
|
|
||||||
tests:
|
tests:
|
||||||
set -e; \
|
@set -e; \
|
||||||
for i in *-tests/Makefile; do \
|
for i in $(SUB)*-tests/Makefile; do \
|
||||||
echo '==== Making:' $$(dirname $$i); \
|
echo '==== Making:' $$(dirname $$i); \
|
||||||
$(MAKE) -C $$(dirname $$i) -j; \
|
$(MAKE) -C $$(dirname $$i) -j; \
|
||||||
done
|
done
|
||||||
|
$(MAKE) -C compile-test
|
||||||
|
|
||||||
|
valgrind:
|
||||||
|
@set -e; \
|
||||||
|
$(MAKE) tests; \
|
||||||
|
for i in $(SUB)*-tests/Makefile; do \
|
||||||
|
echo '==== Running:' $$(dirname $$i); \
|
||||||
|
valgrind $$(dirname $$i)/$$(dirname $$i).out; \
|
||||||
|
done
|
||||||
|
|
||||||
debugtest:
|
debugtest:
|
||||||
set -e; \
|
@set -e; \
|
||||||
$(MAKE) clean; \
|
$(MAKE) clean; \
|
||||||
$(MAKE) -C debug-mode -j; \
|
$(MAKE) -C debug-mode -j; \
|
||||||
debug-mode/debug-tests.out
|
debug-mode/debug-tests.out
|
||||||
|
|
||||||
runtests: debugtest
|
runtests:
|
||||||
$(MAKE) clean
|
@set -e; \
|
||||||
$(MAKE) tests
|
$(MAKE) tests; \
|
||||||
set -e; \
|
for i in $(SUB)*-tests/Makefile; do \
|
||||||
for i in *-tests/Makefile; do \
|
|
||||||
echo '==== Running:' $$(dirname $$i); \
|
echo '==== Running:' $$(dirname $$i); \
|
||||||
$$(dirname $$i)/$$(dirname $$i).out; \
|
$$(dirname $$i)/$$(dirname $$i).out; \
|
||||||
done
|
done
|
||||||
|
|
||||||
clean:
|
clean:
|
||||||
set -e; \
|
@set -e; \
|
||||||
for i in *-tests/Makefile; do \
|
for i in $(SUB)*-tests/Makefile; do \
|
||||||
echo '==== Cleaning:' $$(dirname $$i); \
|
echo '==== Cleaning:' $$(dirname $$i); \
|
||||||
$(MAKE) -C $$(dirname $$i) clean; \
|
$(MAKE) -C $$(dirname $$i) clean; \
|
||||||
done
|
done
|
||||||
|
|||||||
11
tests/Makefile.opts
Normal file
11
tests/Makefile.opts
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
|
||||||
|
# GCC
|
||||||
|
# CXXFLAGS = -Wextra -Wall -std=gnu++11 -fno-exceptions -fno-threadsafe-statics
|
||||||
|
EXTRA_CXXFLAGS=-g3 -O0 -std=c++17
|
||||||
|
|
||||||
|
CXXFLAGS=-D_GNU_SOURCE -Werror=return-type -std=gnu++17 -Wall -g3 -O0
|
||||||
|
|
||||||
|
# CLANG SANITIZE
|
||||||
|
# CXX=clang
|
||||||
|
# EXTRA_CXXFLAGS=-g3 -O0 -fsanitize=memory
|
||||||
|
# LDFLAGS = -lpthread -lstdc++
|
||||||
10
tests/classbind-tests/Makefile
Normal file
10
tests/classbind-tests/Makefile
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
||||||
|
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
||||||
|
|
||||||
|
EXTRA_CXXFLAGS=-g3 -O0 -DTINY_MQTT_TESTS
|
||||||
|
|
||||||
|
APP_NAME := classbind-tests
|
||||||
|
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsyncTCP TinyConsole
|
||||||
|
ARDUINO_LIB_DIRS := ../../../EspMock/libraries
|
||||||
|
EPOXY_CORE := EPOXY_CORE_ESP8266
|
||||||
|
include ../../../EpoxyDuino/EpoxyDuino.mk
|
||||||
@@ -17,9 +17,10 @@ class TestReceiver : public MqttClassBinder<TestReceiver>
|
|||||||
public:
|
public:
|
||||||
TestReceiver(const char* name) : MqttClassBinder(), name_(name) {}
|
TestReceiver(const char* name) : MqttClassBinder(), name_(name) {}
|
||||||
|
|
||||||
void onPublish(const MqttClient* /* source */, const Topic& topic, const char* payload, size_t /* length */)
|
void onPublish(const MqttClient* /* source */, const Topic& topic, const char* /* payload */, size_t /* length */)
|
||||||
{
|
{
|
||||||
Serial << "--> routed message received by " << name_ << ':' << topic.c_str() << " = " << payload << endl;
|
(void) topic;
|
||||||
|
// Serial << "--> routed message received by " << name_ << ':' << topic.c_str() << " = " << payload << endl;
|
||||||
messages[name_]++;
|
messages[name_]++;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -35,7 +36,8 @@ std::map<std::string, int> TestReceiver::messages;
|
|||||||
static int unrouted = 0;
|
static int unrouted = 0;
|
||||||
void onUnrouted(const MqttClient*, const Topic& topic, const char*, size_t)
|
void onUnrouted(const MqttClient*, const Topic& topic, const char*, size_t)
|
||||||
{
|
{
|
||||||
Serial << "--> unrouted: " << topic.c_str() << endl;
|
(void) topic;
|
||||||
|
// Serial << "--> unrouted: " << topic.c_str() << endl;
|
||||||
unrouted++;
|
unrouted++;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -154,7 +156,7 @@ test(classbind_one_client_receives_the_message)
|
|||||||
client.loop();
|
client.loop();
|
||||||
broker.loop();
|
broker.loop();
|
||||||
}
|
}
|
||||||
|
|
||||||
assertEqual(TestReceiver::messages["receiver"], 1);
|
assertEqual(TestReceiver::messages["receiver"], 1);
|
||||||
assertEqual(unrouted, 0);
|
assertEqual(unrouted, 0);
|
||||||
}
|
}
|
||||||
@@ -190,7 +192,7 @@ test(classbind_routes_should_be_empty_when_receiver_goes_out_of_scope)
|
|||||||
client.loop();
|
client.loop();
|
||||||
broker.loop();
|
broker.loop();
|
||||||
}
|
}
|
||||||
|
|
||||||
assertEqual(TestReceiver::messages["receiver"], 0);
|
assertEqual(TestReceiver::messages["receiver"], 0);
|
||||||
assertEqual(MqttClassBinder<TestReceiver>::size(), (size_t)0);
|
assertEqual(MqttClassBinder<TestReceiver>::size(), (size_t)0);
|
||||||
}
|
}
|
||||||
@@ -224,7 +226,7 @@ test(classbind_publish_should_be_dispatched_to_many_receivers)
|
|||||||
client.loop();
|
client.loop();
|
||||||
broker.loop();
|
broker.loop();
|
||||||
}
|
}
|
||||||
|
|
||||||
assertEqual(TestReceiver::messages["receiver_1"], 1);
|
assertEqual(TestReceiver::messages["receiver_1"], 1);
|
||||||
assertEqual(TestReceiver::messages["receiver_2"], 1);
|
assertEqual(TestReceiver::messages["receiver_2"], 1);
|
||||||
}
|
}
|
||||||
@@ -274,7 +276,7 @@ test(classbind_register_to_many_clients)
|
|||||||
|
|
||||||
// Ensure publishes are processed
|
// Ensure publishes are processed
|
||||||
for (int i =0; i<5; i++) loop();
|
for (int i =0; i<5; i++) loop();
|
||||||
|
|
||||||
assertEqual(TestReceiver::messages["receiver"], 4);
|
assertEqual(TestReceiver::messages["receiver"], 4);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -338,7 +340,7 @@ void setup() {
|
|||||||
while(!Serial);
|
while(!Serial);
|
||||||
*/
|
*/
|
||||||
|
|
||||||
Serial.println("=============[ FAKE NETWORK TinyMqtt TESTS ]========================");
|
Serial.println("=============[ TinyMqtt class-bind TESTS ]========================");
|
||||||
|
|
||||||
WiFi.mode(WIFI_STA);
|
WiFi.mode(WIFI_STA);
|
||||||
WiFi.begin("network", "password");
|
WiFi.begin("network", "password");
|
||||||
|
|||||||
23
tests/compile-test/Makefile
Normal file
23
tests/compile-test/Makefile
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
# vim: noexpandtab
|
||||||
|
|
||||||
|
|
||||||
|
APP_NAME := compile-tests
|
||||||
|
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsync ESP8266mDNS TinyConsole
|
||||||
|
ARDUINO_LIB_DIRS := ../../../EspMock/libraries
|
||||||
|
EPOXY_CORE := EPOXY_CORE_ESP8266
|
||||||
|
DUMMY := 1
|
||||||
|
RUN = no
|
||||||
|
APP_SRCS_COMP += $(shell find ../../examples -name "*.ino")
|
||||||
|
APP_SRCS_CPP += $(APP_SRCS_COMP:%.ino=%.o)
|
||||||
|
CXXFLAGS = -D_GNU_SOURCE -Werror=return-type -std=gnu++17 -Wall -g3 -O0 -DEPOXY_DUINO
|
||||||
|
|
||||||
|
include ../Makefile.opts
|
||||||
|
|
||||||
|
all:
|
||||||
|
|
||||||
|
compile-tests.out: $(OBJS)
|
||||||
|
|
||||||
|
run:
|
||||||
|
@echo "No run for compile tests"
|
||||||
|
exit 0
|
||||||
|
|
||||||
@@ -1,10 +1,7 @@
|
|||||||
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
||||||
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
||||||
|
|
||||||
EXTRA_CXXFLAGS=-g3 -O0
|
include ../Makefile.opts
|
||||||
|
|
||||||
# Remove flto flag from EpoxyDuino (too many <optimized out>)
|
|
||||||
CXXFLAGS = -Wextra -Wall -std=gnu++11 -fno-exceptions -fno-threadsafe-statics
|
|
||||||
|
|
||||||
APP_NAME := local-tests
|
APP_NAME := local-tests
|
||||||
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsyncTCP TinyConsole
|
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsyncTCP TinyConsole
|
||||||
|
|||||||
@@ -12,23 +12,32 @@
|
|||||||
* Also, this will allow to mock and thus run Action on github
|
* Also, this will allow to mock and thus run Action on github
|
||||||
**/
|
**/
|
||||||
|
|
||||||
using namespace std;
|
using string = TinyConsole::string;
|
||||||
|
|
||||||
MqttBroker broker(1883);
|
MqttBroker broker(1883);
|
||||||
|
|
||||||
std::map<std::string, std::map<Topic, int>> published; // map[client_id] => map[topic] = count
|
std::map<string, std::map<Topic, int>> published; // map[client_id] => map[topic] = count
|
||||||
|
|
||||||
const char* lastPayload;
|
std::string lastPayload;
|
||||||
size_t lastLength;
|
size_t lastLength;
|
||||||
|
|
||||||
void onPublish(const MqttClient* srce, const Topic& topic, const char* payload, size_t length)
|
void onPublish(const MqttClient* srce, const Topic& topic, const char* payload, size_t length)
|
||||||
{
|
{
|
||||||
if (srce)
|
if (srce)
|
||||||
published[srce->id()][topic]++;
|
published[srce->id()][topic]++;
|
||||||
lastPayload = payload;
|
lastPayload = std::string(payload, length);
|
||||||
lastLength = length;
|
lastLength = length;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
test(local_not_connected_by_default)
|
||||||
|
{
|
||||||
|
MqttClient client;
|
||||||
|
assertEqual(client.connected(), false);
|
||||||
|
|
||||||
|
MqttBroker broker(1883);
|
||||||
|
assertEqual(broker.connected(), false);
|
||||||
|
}
|
||||||
|
|
||||||
test(local_client_should_unregister_when_destroyed)
|
test(local_client_should_unregister_when_destroyed)
|
||||||
{
|
{
|
||||||
assertEqual(broker.clientsCount(), (size_t)0);
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
@@ -40,27 +49,70 @@ test(local_client_should_unregister_when_destroyed)
|
|||||||
assertEqual(broker.clientsCount(), (size_t)0);
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
}
|
}
|
||||||
|
|
||||||
#if 0
|
test(local_client_do_not_disconnect_after_publishing_and_long_inactivity)
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
EpoxyTest::set_millis(0);
|
||||||
|
MqttBroker broker(1883);
|
||||||
|
MqttClient client(&broker, "client");
|
||||||
|
MqttClient sender(&broker, "sender");
|
||||||
|
broker.loop();
|
||||||
|
|
||||||
|
client.subscribe("#");
|
||||||
|
client.subscribe("test");
|
||||||
|
client.setCallback(onPublish);
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)2);
|
||||||
|
|
||||||
|
sender.publish("test", "value");
|
||||||
|
broker.loop();
|
||||||
|
|
||||||
|
EpoxyTest::add_seconds(600);
|
||||||
|
client.loop();
|
||||||
|
sender.loop();
|
||||||
|
broker.loop();
|
||||||
|
|
||||||
|
sender.publish("test", "value");
|
||||||
|
broker.loop();
|
||||||
|
sender.loop();
|
||||||
|
broker.loop();
|
||||||
|
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)2);
|
||||||
|
assertEqual(sender.connected(), true);
|
||||||
|
assertEqual(client.connected(), true);
|
||||||
|
|
||||||
|
assertEqual(published.size(), (size_t)1); // client has received something
|
||||||
|
}
|
||||||
|
|
||||||
test(local_connect)
|
test(local_connect)
|
||||||
{
|
{
|
||||||
assertEqual(broker.clientsCount(), (size_t)0);
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
MqttClient client;
|
MqttClient client(&broker);
|
||||||
assertTrue(client.connected());
|
assertTrue(client.connected());
|
||||||
assertEqual(broker.clientsCount(), (size_t)1);
|
assertEqual(broker.clientsCount(), (size_t)1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
test(local_publish_to_nowhere)
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
|
MqttClient publisher;
|
||||||
|
MqttError status = publisher.publish("a/b");
|
||||||
|
assertEqual(status, MqttError::MqttNowhereToSend);
|
||||||
|
}
|
||||||
|
|
||||||
test(local_publish_should_be_dispatched)
|
test(local_publish_should_be_dispatched)
|
||||||
{
|
{
|
||||||
published.clear();
|
published.clear();
|
||||||
assertEqual(broker.clientsCount(), (size_t)0);
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
MqttClient subscriber;
|
MqttClient subscriber(&broker, "");
|
||||||
subscriber.subscribe("a/b");
|
subscriber.subscribe("a/b");
|
||||||
subscriber.subscribe("a/c");
|
subscriber.subscribe("a/c");
|
||||||
subscriber.setCallback(onPublish);
|
subscriber.setCallback(onPublish);
|
||||||
|
|
||||||
MqttClient publisher;
|
MqttClient publisher(&broker);
|
||||||
publisher.publish("a/b");
|
publisher.publish("a/b");
|
||||||
publisher.publish("a/c");
|
publisher.publish("a/c");
|
||||||
publisher.publish("a/c");
|
publisher.publish("a/c");
|
||||||
@@ -70,21 +122,40 @@ test(local_publish_should_be_dispatched)
|
|||||||
assertEqual(published[""]["a/c"], 2);
|
assertEqual(published[""]["a/c"], 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
test(hudge_payload)
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
const char* payload="This payload is hudge, just because its length exceeds 127. Thus when encoding length, we have to encode it on two bytes at min. This should not prevent the message from being encoded and decoded successfully !";
|
||||||
|
|
||||||
|
MqttClient subscriber(&broker);
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)1);
|
||||||
|
subscriber.setCallback(onPublish);
|
||||||
|
subscriber.subscribe("a/b"); // Note -> this does not send any byte .... (nowhere to send)
|
||||||
|
|
||||||
|
MqttClient publisher(&broker);
|
||||||
|
publisher.publish("a/b", payload); // This publish is received
|
||||||
|
|
||||||
|
// onPublish should have filled lastPayload and lastLength
|
||||||
|
assertEqual(payload, lastPayload.c_str());
|
||||||
|
assertEqual(lastLength, strlen(payload));
|
||||||
|
assertEqual(strncmp(payload, lastPayload.c_str(), lastLength), 0);
|
||||||
|
}
|
||||||
|
|
||||||
test(local_publish_should_be_dispatched_to_local_clients)
|
test(local_publish_should_be_dispatched_to_local_clients)
|
||||||
{
|
{
|
||||||
published.clear();
|
published.clear();
|
||||||
assertEqual(broker.clientsCount(), (size_t)0);
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
MqttClient subscriber_a("A");
|
MqttClient subscriber_a(&broker, "A");
|
||||||
subscriber_a.setCallback(onPublish);
|
subscriber_a.setCallback(onPublish);
|
||||||
subscriber_a.subscribe("a/b");
|
subscriber_a.subscribe("a/b");
|
||||||
subscriber_a.subscribe("a/c");
|
subscriber_a.subscribe("a/c");
|
||||||
|
|
||||||
MqttClient subscriber_b("B");
|
MqttClient subscriber_b(&broker, "B");
|
||||||
subscriber_b.setCallback(onPublish);
|
subscriber_b.setCallback(onPublish);
|
||||||
subscriber_b.subscribe("a/b");
|
subscriber_b.subscribe("a/b");
|
||||||
|
|
||||||
MqttClient publisher;
|
MqttClient publisher(&broker);
|
||||||
publisher.publish("a/b");
|
publisher.publish("a/b");
|
||||||
publisher.publish("a/c");
|
publisher.publish("a/c");
|
||||||
|
|
||||||
@@ -100,11 +171,11 @@ test(local_unsubscribe)
|
|||||||
published.clear();
|
published.clear();
|
||||||
assertEqual(broker.clientsCount(), (size_t)0);
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
MqttClient subscriber;
|
MqttClient subscriber(&broker, "");
|
||||||
subscriber.setCallback(onPublish);
|
subscriber.setCallback(onPublish);
|
||||||
subscriber.subscribe("a/b");
|
subscriber.subscribe("a/b");
|
||||||
|
|
||||||
MqttClient publisher;
|
MqttClient publisher(&broker);
|
||||||
publisher.publish("a/b");
|
publisher.publish("a/b");
|
||||||
|
|
||||||
subscriber.unsubscribe("a/b");
|
subscriber.unsubscribe("a/b");
|
||||||
@@ -120,9 +191,9 @@ test(local_nocallback_when_destroyed)
|
|||||||
published.clear();
|
published.clear();
|
||||||
assertEqual(broker.clientsCount(), (size_t)0);
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
MqttClient publisher;
|
MqttClient publisher(&broker);
|
||||||
{
|
{
|
||||||
MqttClient subscriber;
|
MqttClient subscriber(&broker);
|
||||||
subscriber.setCallback(onPublish);
|
subscriber.setCallback(onPublish);
|
||||||
subscriber.subscribe("a/b");
|
subscriber.subscribe("a/b");
|
||||||
publisher.publish("a/b");
|
publisher.publish("a/b");
|
||||||
@@ -132,7 +203,6 @@ test(local_nocallback_when_destroyed)
|
|||||||
|
|
||||||
assertEqual(published.size(), (size_t)1); // Only one publish has been received
|
assertEqual(published.size(), (size_t)1); // Only one publish has been received
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
//----------------------------------------------------------------------------
|
//----------------------------------------------------------------------------
|
||||||
// setup() and loop()
|
// setup() and loop()
|
||||||
|
|||||||
@@ -1,10 +1,10 @@
|
|||||||
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
||||||
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
||||||
|
|
||||||
EXTRA_CXXFLAGS=-g3 -O0
|
include ../Makefile.opts
|
||||||
|
|
||||||
# Remove flto flag from EpoxyDuino (too many <optimized out>)
|
# Remove flto flag from EpoxyDuino (too many <optimized out>)
|
||||||
CXXFLAGS = -Wextra -Wall -std=gnu++11 -fno-exceptions -fno-threadsafe-statics
|
# CXXFLAGS = -Wextra -Wall -std=gnu++11 -fno-exceptions -fno-threadsafe-statics
|
||||||
|
|
||||||
APP_NAME := network-tests
|
APP_NAME := network-tests
|
||||||
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsyncTCP TinyConsole
|
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsyncTCP TinyConsole
|
||||||
|
|||||||
@@ -16,6 +16,8 @@
|
|||||||
* Checks with a local broker. Clients must connect to the local broker
|
* Checks with a local broker. Clients must connect to the local broker
|
||||||
**/
|
**/
|
||||||
|
|
||||||
|
using string = TinyConsole::string;
|
||||||
|
|
||||||
// if ascii_pos = 0, no ascii dump, else ascii dump starts after column ascii_pos
|
// if ascii_pos = 0, no ascii dump, else ascii dump starts after column ascii_pos
|
||||||
std::string bufferToHexa(const uint8_t* buffer, size_t length, char sep = 0, size_t ascii_pos = 0)
|
std::string bufferToHexa(const uint8_t* buffer, size_t length, char sep = 0, size_t ascii_pos = 0)
|
||||||
{
|
{
|
||||||
@@ -78,12 +80,12 @@ String toString(const IPAddress& ip)
|
|||||||
|
|
||||||
MqttBroker broker(1883);
|
MqttBroker broker(1883);
|
||||||
|
|
||||||
std::map<std::string, std::map<Topic, int>> published; // map[client_id] => map[topic] = count
|
std::map<string, std::map<Topic, int>> published; // map[client_id] => map[topic] = count
|
||||||
|
|
||||||
char* lastPayload = nullptr;
|
char* lastPayload = nullptr;
|
||||||
size_t lastLength;
|
size_t lastLength;
|
||||||
|
|
||||||
void start_servers(int n, bool early_accept = true)
|
void start_many_wifi_esp(int n, bool early_accept = true)
|
||||||
{
|
{
|
||||||
ESP8266WiFiClass::resetInstances();
|
ESP8266WiFiClass::resetInstances();
|
||||||
ESP8266WiFiClass::earlyAccept = early_accept;
|
ESP8266WiFiClass::earlyAccept = early_accept;
|
||||||
@@ -99,13 +101,13 @@ void onPublish(const MqttClient* srce, const Topic& topic, const char* payload,
|
|||||||
{
|
{
|
||||||
if (srce)
|
if (srce)
|
||||||
published[srce->id()][topic]++;
|
published[srce->id()][topic]++;
|
||||||
|
|
||||||
if (lastPayload) free(lastPayload);
|
if (lastPayload) free(lastPayload);
|
||||||
lastPayload = strdup(payload);
|
lastPayload = strdup(payload);
|
||||||
lastLength = length;
|
lastLength = length;
|
||||||
}
|
}
|
||||||
|
|
||||||
test(network_single_broker_begin)
|
test(single_broker_begin)
|
||||||
{
|
{
|
||||||
assertEqual(WiFi.status(), WL_CONNECTED);
|
assertEqual(WiFi.status(), WL_CONNECTED);
|
||||||
|
|
||||||
@@ -117,7 +119,7 @@ test(network_single_broker_begin)
|
|||||||
|
|
||||||
test(suback)
|
test(suback)
|
||||||
{
|
{
|
||||||
start_servers(2, true);
|
start_many_wifi_esp(2, true);
|
||||||
assertEqual(WiFi.status(), WL_CONNECTED);
|
assertEqual(WiFi.status(), WL_CONNECTED);
|
||||||
|
|
||||||
MqttBroker broker(1883);
|
MqttBroker broker(1883);
|
||||||
@@ -142,10 +144,97 @@ test(suback)
|
|||||||
assertEqual(MqttClient::counters[MqttMessage::Type::SubAck], 1);
|
assertEqual(MqttClient::counters[MqttMessage::Type::SubAck], 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
test(network_client_keep_alive_high)
|
test(remote_client_deletion)
|
||||||
|
{
|
||||||
|
assertEqual(MqttClient::instances, 0);
|
||||||
|
{
|
||||||
|
start_many_wifi_esp(3, true);
|
||||||
|
assertEqual(WiFi.status(), WL_CONNECTED);
|
||||||
|
|
||||||
|
MqttBroker broker(1883);
|
||||||
|
broker.begin();
|
||||||
|
IPAddress broker_ip = WiFi.localIP();
|
||||||
|
|
||||||
|
// A first remote client
|
||||||
|
ESP8266WiFiClass::selectInstance(2);
|
||||||
|
MqttClient remote_client;
|
||||||
|
assertEqual(MqttClient::instances, 1);
|
||||||
|
|
||||||
|
remote_client.connect(broker_ip.toString().c_str());
|
||||||
|
broker.loop(); remote_client.loop();
|
||||||
|
assertEqual(MqttClient::instances, 2); // broker creates a client to manage remote_client
|
||||||
|
|
||||||
|
// A second remote client
|
||||||
|
ESP8266WiFiClass::selectInstance(3);
|
||||||
|
MqttClient secund_client;
|
||||||
|
assertEqual(MqttClient::instances, 3);
|
||||||
|
|
||||||
|
secund_client.connect(broker_ip.toString().c_str());
|
||||||
|
broker.loop(); remote_client.loop();
|
||||||
|
assertEqual(MqttClient::instances, 4);
|
||||||
|
|
||||||
|
// Now disconnect remote clients
|
||||||
|
remote_client.close();
|
||||||
|
broker.loop(); remote_client.loop();
|
||||||
|
assertEqual(MqttClient::instances, 3);
|
||||||
|
|
||||||
|
secund_client.close();
|
||||||
|
broker.loop(); remote_client.loop();
|
||||||
|
assertEqual(MqttClient::instances, 2); // These instances are in this scope
|
||||||
|
|
||||||
|
// Now simulate that the external client is dead without disconnecting
|
||||||
|
secund_client.connect(broker_ip.toString().c_str());
|
||||||
|
broker.loop(); remote_client.loop();
|
||||||
|
assertEqual(MqttClient::instances, 3);
|
||||||
|
|
||||||
|
WiFi.disconnect();
|
||||||
|
broker.loop(); remote_client.loop();
|
||||||
|
broker.loop(); remote_client.loop();
|
||||||
|
assertEqual(MqttClient::instances, 2);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
assertEqual(MqttClient::instances, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(broker_connect_and_client_deletion)
|
||||||
|
{
|
||||||
|
assertEqual(MqttClient::instances, 0);
|
||||||
|
{
|
||||||
|
start_many_wifi_esp(2, true);
|
||||||
|
assertEqual(WiFi.status(), WL_CONNECTED);
|
||||||
|
|
||||||
|
MqttBroker broker(1883);
|
||||||
|
broker.begin();
|
||||||
|
|
||||||
|
ESP8266WiFiClass::selectInstance(2);
|
||||||
|
MqttBroker remote_broker(1883);
|
||||||
|
remote_broker.begin();
|
||||||
|
IPAddress remote_broker_ip = WiFi.localIP();
|
||||||
|
assertEqual(MqttClient::instances, 0);
|
||||||
|
|
||||||
|
ESP8266WiFiClass::selectInstance(1);
|
||||||
|
broker.connect(remote_broker_ip.toString().c_str());
|
||||||
|
remote_broker.loop();
|
||||||
|
assertEqual(remote_broker.clientsCount(), (size_t)1);
|
||||||
|
|
||||||
|
// Here, we have two MqttClient
|
||||||
|
// The client that connects broker to remote_broker
|
||||||
|
// The client created by remote_broker
|
||||||
|
assertEqual(MqttClient::instances, 2);
|
||||||
|
|
||||||
|
broker.connect("");
|
||||||
|
remote_broker.loop(); broker.loop();
|
||||||
|
remote_broker.loop(); broker.loop();
|
||||||
|
assertEqual(remote_broker.clientsCount(), (size_t)0);
|
||||||
|
}
|
||||||
|
assertEqual(MqttClient::instances, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(client_keep_alive_high)
|
||||||
{
|
{
|
||||||
const uint32_t keep_alive=1000;
|
const uint32_t keep_alive=1000;
|
||||||
start_servers(2, true);
|
start_many_wifi_esp(2, true);
|
||||||
assertEqual(WiFi.status(), WL_CONNECTED);
|
assertEqual(WiFi.status(), WL_CONNECTED);
|
||||||
|
|
||||||
MqttBroker broker(1883);
|
MqttBroker broker(1883);
|
||||||
@@ -177,9 +266,153 @@ test(network_client_keep_alive_high)
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
test(network_client_to_broker_connexion)
|
test(retained_message)
|
||||||
{
|
{
|
||||||
start_servers(2, true);
|
published.clear();
|
||||||
|
|
||||||
|
start_many_wifi_esp(2, true);
|
||||||
|
assertEqual(WiFi.status(), WL_CONNECTED);
|
||||||
|
|
||||||
|
MqttBroker broker(1883);
|
||||||
|
broker.begin();
|
||||||
|
broker.retain(10);
|
||||||
|
IPAddress broker_ip = WiFi.localIP();
|
||||||
|
|
||||||
|
MqttClient local_client(&broker, "sender");
|
||||||
|
|
||||||
|
// Send a retained message
|
||||||
|
// No remote client connected
|
||||||
|
local_client.publish("topic", "retained once", true);
|
||||||
|
for(int i=0; i<2; i++) { broker.loop(); local_client.loop(); };
|
||||||
|
|
||||||
|
// Send a second message on the same topic (issue 86)
|
||||||
|
local_client.publish("topic", "retained once", true);
|
||||||
|
for(int i=0; i<2; i++) { broker.loop(); local_client.loop(); };
|
||||||
|
|
||||||
|
// Send a second message on the same topic (issue 86)
|
||||||
|
local_client.publish("topic", "retained last", true);
|
||||||
|
for(int i=0; i<2; i++) { broker.loop(); local_client.loop(); };
|
||||||
|
|
||||||
|
// No connect a client from 2nd Esp
|
||||||
|
ESP8266WiFiClass::selectInstance(2);
|
||||||
|
MqttClient remote_client("receiver");
|
||||||
|
remote_client.connect(broker_ip, 1883);
|
||||||
|
remote_client.setCallback(onPublish);
|
||||||
|
|
||||||
|
assertTrue(remote_client.connected());
|
||||||
|
for(int i=0; i<10; i++) { broker.loop(); local_client.loop(); remote_client.loop(); };
|
||||||
|
assertEqual(broker.clientsCount(), (size_t) 2);
|
||||||
|
|
||||||
|
// Should not have received anything yet
|
||||||
|
assertEqual(published.size(), (size_t)0);
|
||||||
|
|
||||||
|
// Now, remote client subscribes to topic
|
||||||
|
remote_client.subscribe("#");
|
||||||
|
for(int i=0; i<10; i++) { broker.loop(); local_client.loop(); remote_client.loop(); };
|
||||||
|
|
||||||
|
// Check that the retained message is published once
|
||||||
|
assertEqual(published.size(), (size_t)1);
|
||||||
|
assertEqual(published["receiver"]["topic"], 1);
|
||||||
|
|
||||||
|
// FIXME we should check that
|
||||||
|
// 1 - Retained message has the retain flag set
|
||||||
|
// 2 - Published retained messages that are send normally have their retain flag off
|
||||||
|
|
||||||
|
// The next part of this test does not pass yet (due to remote_client.close()
|
||||||
|
// that does not work well.
|
||||||
|
return;
|
||||||
|
|
||||||
|
// Now remove the retained message
|
||||||
|
remote_client.close();
|
||||||
|
for(int i=0; i<4; i++) { broker.loop(); local_client.loop(); remote_client.loop(); };
|
||||||
|
assertFalse(remote_client.connected());
|
||||||
|
assertEqual(broker.clientsCount(), (size_t) 1);
|
||||||
|
|
||||||
|
// Disconnect / reconnect the remote clien that should receive again the message
|
||||||
|
remote_client.connect(broker_ip, 1883);
|
||||||
|
remote_client.subscribe("topic");
|
||||||
|
assertTrue(remote_client.connected());
|
||||||
|
for(int i=0; i<4; i++) { broker.loop(); local_client.loop(); remote_client.loop(); };
|
||||||
|
assertEqual(broker.clientsCount(), (size_t) 2);
|
||||||
|
assertEqual(published.size(), (size_t)2);
|
||||||
|
|
||||||
|
// Remove the retained message now
|
||||||
|
local_client.publish("topic", "", true);
|
||||||
|
assertEqual(published.size(), (size_t)1);
|
||||||
|
|
||||||
|
// And reconnect the remote client
|
||||||
|
remote_client.connect(broker_ip, 1883);
|
||||||
|
for(int i=0; i<4; i++) { broker.loop(); local_client.loop(); remote_client.loop(); };
|
||||||
|
assertEqual(broker.clientsCount(), (size_t) 2);
|
||||||
|
|
||||||
|
// check that the message was received
|
||||||
|
assertEqual(published.size(), (size_t)2);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(retained_payload) // # issue #84
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
|
||||||
|
start_many_wifi_esp(2, true);
|
||||||
|
assertEqual(WiFi.status(), WL_CONNECTED);
|
||||||
|
|
||||||
|
MqttBroker broker(1883);
|
||||||
|
broker.begin();
|
||||||
|
broker.retain(10);
|
||||||
|
IPAddress broker_ip = WiFi.localIP();
|
||||||
|
|
||||||
|
MqttClient local_client(&broker, "sender");
|
||||||
|
|
||||||
|
// Send a retained message
|
||||||
|
// No remote client connected
|
||||||
|
local_client.publish("topic", "retained", true);
|
||||||
|
for(int i=0; i<2; i++) { broker.loop(); local_client.loop(); };
|
||||||
|
|
||||||
|
// No connect a client from 2nd Esp
|
||||||
|
ESP8266WiFiClass::selectInstance(2);
|
||||||
|
MqttClient remote_client("receiver");
|
||||||
|
remote_client.connect(broker_ip, 1883);
|
||||||
|
remote_client.setCallback(onPublish);
|
||||||
|
remote_client.subscribe("#");
|
||||||
|
|
||||||
|
assertTrue(remote_client.connected());
|
||||||
|
for(int i=0; i<10; i++) { broker.loop(); local_client.loop(); remote_client.loop(); };
|
||||||
|
|
||||||
|
// Check that the retained message is published
|
||||||
|
assertEqual(lastPayload, "retained");
|
||||||
|
}
|
||||||
|
|
||||||
|
test(remote_client_disconnect_reconnect)
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
start_many_wifi_esp(2, true);
|
||||||
|
assertEqual(WiFi.status(), WL_CONNECTED);
|
||||||
|
|
||||||
|
MqttBroker broker(1883);
|
||||||
|
broker.begin();
|
||||||
|
IPAddress broker_ip = WiFi.localIP();
|
||||||
|
|
||||||
|
ESP8266WiFiClass::selectInstance(2);
|
||||||
|
MqttClient client;
|
||||||
|
client.connect(broker_ip, 1883);
|
||||||
|
|
||||||
|
for(int i=0; i<4; i++) { broker.loop(); client.loop(); };
|
||||||
|
assertEqual(broker.clientsCount(), (size_t) 1);
|
||||||
|
|
||||||
|
// Disconnect the client
|
||||||
|
client.close();
|
||||||
|
for(int i=0; i<4; i++) { broker.loop(); client.loop(); };
|
||||||
|
assertEqual(broker.clientsCount(), (size_t) 0);
|
||||||
|
|
||||||
|
// Reconnect the client
|
||||||
|
client.connect(broker_ip, 1883);
|
||||||
|
for(int i=0; i<4; i++) { broker.loop(); client.loop(); };
|
||||||
|
assertEqual(broker.clientsCount(), (size_t) 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(client_to_broker_connexion)
|
||||||
|
{
|
||||||
|
start_many_wifi_esp(2, true);
|
||||||
assertEqual(WiFi.status(), WL_CONNECTED);
|
assertEqual(WiFi.status(), WL_CONNECTED);
|
||||||
|
|
||||||
MqttBroker broker(1883);
|
MqttBroker broker(1883);
|
||||||
@@ -195,9 +428,9 @@ test(network_client_to_broker_connexion)
|
|||||||
assertTrue(client.connected());
|
assertTrue(client.connected());
|
||||||
}
|
}
|
||||||
|
|
||||||
test(network_one_client_one_broker_publish_and_subscribe_through_network)
|
test(one_client_one_broker_publish_and_subscribe)
|
||||||
{
|
{
|
||||||
start_servers(2, true);
|
start_many_wifi_esp(2, true);
|
||||||
published.clear();
|
published.clear();
|
||||||
assertEqual(WiFi.status(), WL_CONNECTED);
|
assertEqual(WiFi.status(), WL_CONNECTED);
|
||||||
|
|
||||||
@@ -221,14 +454,14 @@ test(network_one_client_one_broker_publish_and_subscribe_through_network)
|
|||||||
client.loop();
|
client.loop();
|
||||||
broker.loop();
|
broker.loop();
|
||||||
}
|
}
|
||||||
|
|
||||||
assertEqual(published.size(), (size_t)1);
|
assertEqual(published.size(), (size_t)1);
|
||||||
assertEqual((int)lastLength, (int)2); // sizeof(ab)
|
assertEqual((int)lastLength, (int)2); // sizeof(ab)
|
||||||
}
|
}
|
||||||
|
|
||||||
test(network_one_client_one_broker_hudge_publish_and_subscribe_through_network)
|
test(one_client_one_broker_hudge_payload)
|
||||||
{
|
{
|
||||||
start_servers(2, true);
|
start_many_wifi_esp(2, true);
|
||||||
published.clear();
|
published.clear();
|
||||||
assertEqual(WiFi.status(), WL_CONNECTED);
|
assertEqual(WiFi.status(), WL_CONNECTED);
|
||||||
|
|
||||||
@@ -245,8 +478,8 @@ test(network_one_client_one_broker_hudge_publish_and_subscribe_through_network)
|
|||||||
|
|
||||||
std::string sent;
|
std::string sent;
|
||||||
|
|
||||||
for(int i=0; i<200; i++)
|
for(int i=0; i<400; i++)
|
||||||
sent += char('0'+i%10);
|
sent += char('a'+i%26);
|
||||||
|
|
||||||
client.setCallback(onPublish);
|
client.setCallback(onPublish);
|
||||||
client.subscribe("a/b");
|
client.subscribe("a/b");
|
||||||
@@ -262,7 +495,7 @@ test(network_one_client_one_broker_hudge_publish_and_subscribe_through_network)
|
|||||||
assertEqual((unsigned int)lastLength, (unsigned int)sent.size());
|
assertEqual((unsigned int)lastLength, (unsigned int)sent.size());
|
||||||
}
|
}
|
||||||
|
|
||||||
test(network_client_should_unregister_when_destroyed)
|
test(client_should_unregister_when_destroyed)
|
||||||
{
|
{
|
||||||
assertEqual(broker.clientsCount(), (size_t)0);
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
{
|
{
|
||||||
@@ -276,7 +509,7 @@ test(network_client_should_unregister_when_destroyed)
|
|||||||
// THESE TESTS ARE IN LOCAL MODE
|
// THESE TESTS ARE IN LOCAL MODE
|
||||||
// WE HAVE TO CONVERT THEM TO WIFI MODE (pass through virtual TCP link)
|
// WE HAVE TO CONVERT THEM TO WIFI MODE (pass through virtual TCP link)
|
||||||
|
|
||||||
test(network_connect)
|
test(connect)
|
||||||
{
|
{
|
||||||
assertEqual(broker.clientsCount(), (size_t)0);
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
@@ -285,7 +518,7 @@ test(network_connect)
|
|||||||
assertEqual(broker.clientsCount(), (size_t)1);
|
assertEqual(broker.clientsCount(), (size_t)1);
|
||||||
}
|
}
|
||||||
|
|
||||||
test(network_publish_should_be_dispatched)
|
test(publish_should_be_dispatched)
|
||||||
{
|
{
|
||||||
published.clear();
|
published.clear();
|
||||||
assertEqual(broker.clientsCount(), (size_t)0);
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
@@ -305,7 +538,7 @@ test(network_publish_should_be_dispatched)
|
|||||||
assertEqual(published[TINY_MQTT_DEFAULT_CLIENT_ID]["a/c"], 2);
|
assertEqual(published[TINY_MQTT_DEFAULT_CLIENT_ID]["a/c"], 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
test(network_publish_should_be_dispatched_to_clients)
|
test(publish_should_be_dispatched_to_clients)
|
||||||
{
|
{
|
||||||
published.clear();
|
published.clear();
|
||||||
assertEqual(broker.clientsCount(), (size_t)0);
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
@@ -321,7 +554,7 @@ test(network_publish_should_be_dispatched_to_clients)
|
|||||||
|
|
||||||
MqttClient publisher(&broker);
|
MqttClient publisher(&broker);
|
||||||
publisher.publish("a/b"); // A and B should receive this
|
publisher.publish("a/b"); // A and B should receive this
|
||||||
publisher.publish("a/c"); // A should receive this
|
publisher.publish("a/c"); // A should receive this
|
||||||
|
|
||||||
assertEqual(published.size(), (size_t)2); // 2 clients have received something
|
assertEqual(published.size(), (size_t)2); // 2 clients have received something
|
||||||
assertEqual(published["A"]["a/b"], 1);
|
assertEqual(published["A"]["a/b"], 1);
|
||||||
@@ -330,7 +563,7 @@ test(network_publish_should_be_dispatched_to_clients)
|
|||||||
assertEqual(published["B"]["a/c"], 0);
|
assertEqual(published["B"]["a/c"], 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
test(network_unsubscribe)
|
test(unsubscribe)
|
||||||
{
|
{
|
||||||
published.clear();
|
published.clear();
|
||||||
assertEqual(broker.clientsCount(), (size_t)0);
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
@@ -350,7 +583,7 @@ test(network_unsubscribe)
|
|||||||
assertEqual(published[TINY_MQTT_DEFAULT_CLIENT_ID]["a/b"], 1); // Only one publish has been received
|
assertEqual(published[TINY_MQTT_DEFAULT_CLIENT_ID]["a/b"], 1); // Only one publish has been received
|
||||||
}
|
}
|
||||||
|
|
||||||
test(network_nocallback_when_destroyed)
|
test(nocallback_when_destroyed)
|
||||||
{
|
{
|
||||||
published.clear();
|
published.clear();
|
||||||
assertEqual(broker.clientsCount(), (size_t)0);
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
@@ -369,7 +602,7 @@ test(network_nocallback_when_destroyed)
|
|||||||
assertEqual(published.size(), (size_t)1); // Only one publish has been received
|
assertEqual(published.size(), (size_t)1); // Only one publish has been received
|
||||||
}
|
}
|
||||||
|
|
||||||
test(network_small_payload)
|
test(small_payload)
|
||||||
{
|
{
|
||||||
published.clear();
|
published.clear();
|
||||||
|
|
||||||
@@ -387,7 +620,7 @@ test(network_small_payload)
|
|||||||
assertEqual(lastLength, (size_t)4);
|
assertEqual(lastLength, (size_t)4);
|
||||||
}
|
}
|
||||||
|
|
||||||
test(network_hudge_payload)
|
test(hudge_payload)
|
||||||
{
|
{
|
||||||
const char* payload="This payload is hudge, just because its length exceeds 127. Thus when encoding length, we have to encode it on two bytes at min. This should not prevent the message from being encoded and decoded successfully !";
|
const char* payload="This payload is hudge, just because its length exceeds 127. Thus when encoding length, we have to encode it on two bytes at min. This should not prevent the message from being encoded and decoded successfully !";
|
||||||
|
|
||||||
@@ -404,6 +637,20 @@ test(network_hudge_payload)
|
|||||||
assertEqual(strcmp(payload, lastPayload), 0);
|
assertEqual(strcmp(payload, lastPayload), 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
test(disconnected_when_broker_is_deleted)
|
||||||
|
{
|
||||||
|
MqttBroker* broker = new MqttBroker(1883);
|
||||||
|
broker->begin();
|
||||||
|
|
||||||
|
MqttClient client;
|
||||||
|
client.connect(broker);
|
||||||
|
assertEqual(client.connected(), true);
|
||||||
|
client.publish("a", "b");
|
||||||
|
|
||||||
|
delete broker;
|
||||||
|
assertEqual(client.connected(), false);
|
||||||
|
}
|
||||||
|
|
||||||
test(connack)
|
test(connack)
|
||||||
{
|
{
|
||||||
const bool view = false;
|
const bool view = false;
|
||||||
@@ -420,7 +667,7 @@ test(connack)
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
start_servers(2, true);
|
start_many_wifi_esp(2, true);
|
||||||
assertEqual(WiFi.status(), WL_CONNECTED);
|
assertEqual(WiFi.status(), WL_CONNECTED);
|
||||||
|
|
||||||
MqttBroker broker(1883);
|
MqttBroker broker(1883);
|
||||||
@@ -453,7 +700,7 @@ void setup() {
|
|||||||
while(!Serial);
|
while(!Serial);
|
||||||
*/
|
*/
|
||||||
|
|
||||||
Serial.println("=============[ FAKE NETWORK TinyMqtt TESTS ]========================");
|
Serial.println("=============[ NETWORK TinyMqtt TESTS ]========================");
|
||||||
|
|
||||||
WiFi.mode(WIFI_STA);
|
WiFi.mode(WIFI_STA);
|
||||||
WiFi.begin("network", "password");
|
WiFi.begin("network", "password");
|
||||||
@@ -463,4 +710,5 @@ void loop() {
|
|||||||
aunit::TestRunner::run();
|
aunit::TestRunner::run();
|
||||||
|
|
||||||
if (Serial.available()) ESP.reset();
|
if (Serial.available()) ESP.reset();
|
||||||
|
published.clear(); // Avoid crash in unit tests due to exit handlers
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,10 +1,7 @@
|
|||||||
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
||||||
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
||||||
|
|
||||||
EXTRA_CXXFLAGS=-g3 -O0
|
include ../Makefile.opts
|
||||||
|
|
||||||
# Remove flto flag from EpoxyDuino (too many <optimized out>)
|
|
||||||
CXXFLAGS = -Wextra -Wall -std=gnu++11 -fno-exceptions -fno-threadsafe-statics
|
|
||||||
|
|
||||||
APP_NAME := nowifi-tests
|
APP_NAME := nowifi-tests
|
||||||
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsyncTCP TinyConsole
|
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsyncTCP TinyConsole
|
||||||
|
|||||||
@@ -11,11 +11,11 @@
|
|||||||
* Checks with a local broker. Clients must connect to the local broker
|
* Checks with a local broker. Clients must connect to the local broker
|
||||||
**/
|
**/
|
||||||
|
|
||||||
using namespace std;
|
using string = TinyConsole::string;
|
||||||
|
|
||||||
MqttBroker broker(1883);
|
MqttBroker broker(1883);
|
||||||
|
|
||||||
std::map<std::string, std::map<Topic, int>> published; // map[client_id] => map[topic] = count
|
std::map<string, std::map<Topic, int>> published; // map[client_id] => map[topic] = count
|
||||||
|
|
||||||
char* lastPayload = nullptr;
|
char* lastPayload = nullptr;
|
||||||
size_t lastLength;
|
size_t lastLength;
|
||||||
@@ -24,7 +24,7 @@ void onPublish(const MqttClient* srce, const Topic& topic, const char* payload,
|
|||||||
{
|
{
|
||||||
if (srce)
|
if (srce)
|
||||||
published[srce->id()][topic]++;
|
published[srce->id()][topic]++;
|
||||||
|
|
||||||
if (lastPayload) free(lastPayload);
|
if (lastPayload) free(lastPayload);
|
||||||
lastPayload = strdup(payload);
|
lastPayload = strdup(payload);
|
||||||
lastLength = length;
|
lastLength = length;
|
||||||
@@ -85,7 +85,7 @@ test(nowifi_publish_should_be_dispatched_to_clients)
|
|||||||
|
|
||||||
MqttClient publisher(&broker);
|
MqttClient publisher(&broker);
|
||||||
publisher.publish("a/b"); // A and B should receive this
|
publisher.publish("a/b"); // A and B should receive this
|
||||||
publisher.publish("a/c"); // A should receive this
|
publisher.publish("a/c"); // A should receive this
|
||||||
|
|
||||||
assertEqual(published.size(), (size_t)2); // 2 clients have received something
|
assertEqual(published.size(), (size_t)2); // 2 clients have received something
|
||||||
assertEqual(published["A"]["a/b"], 1);
|
assertEqual(published["A"]["a/b"], 1);
|
||||||
@@ -279,4 +279,5 @@ void loop() {
|
|||||||
aunit::TestRunner::run();
|
aunit::TestRunner::run();
|
||||||
|
|
||||||
if (Serial.available()) ESP.reset();
|
if (Serial.available()) ESP.reset();
|
||||||
|
published.clear(); // Avoid crash at exit handlers
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
||||||
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
||||||
|
|
||||||
EXTRA_CXXFLAGS=-g3 -O0
|
include ../Makefile.opts
|
||||||
|
|
||||||
APP_NAME := string-indexer-tests
|
APP_NAME := string-indexer-tests
|
||||||
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsync TinyConsole
|
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsync TinyConsole
|
||||||
|
|||||||
@@ -9,7 +9,7 @@
|
|||||||
*
|
*
|
||||||
**/
|
**/
|
||||||
|
|
||||||
using namespace std;
|
using string = TinyConsole::string;
|
||||||
|
|
||||||
test(indexer_empty)
|
test(indexer_empty)
|
||||||
{
|
{
|
||||||
@@ -84,7 +84,7 @@ test(indexer_indexed_operator_eq)
|
|||||||
|
|
||||||
test(indexer_get_string)
|
test(indexer_get_string)
|
||||||
{
|
{
|
||||||
std::string sone("one");
|
string sone("one");
|
||||||
IndexedString one(sone);
|
IndexedString one(sone);
|
||||||
|
|
||||||
assertTrue(sone==one.str());
|
assertTrue(sone==one.str());
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
||||||
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
||||||
|
|
||||||
EXTRA_CXXFLAGS=-g3 -O0
|
include ../Makefile.opts
|
||||||
|
|
||||||
APP_NAME := topic-tests
|
APP_NAME := topic-tests
|
||||||
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsync TinyConsole
|
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsync TinyConsole
|
||||||
|
|||||||
@@ -5,26 +5,22 @@
|
|||||||
#include <map>
|
#include <map>
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
|
|
||||||
#define endl "\n"
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* TinyMqtt / StringIndexer unit tests.
|
* TinyMqtt / StringIndexer unit tests.
|
||||||
*
|
*
|
||||||
**/
|
**/
|
||||||
|
|
||||||
using namespace std;
|
|
||||||
|
|
||||||
bool testTopicMatch(const char* a, const char* b, bool expected)
|
bool testTopicMatch(const char* a, const char* b, bool expected)
|
||||||
{
|
{
|
||||||
Topic ta(a);
|
Topic ta(a);
|
||||||
Topic tb(b);
|
Topic tb(b);
|
||||||
bool match(ta.matches(tb));
|
bool match(ta.matches(tb));
|
||||||
cout << " " << ta.c_str() << ' ';
|
std::cout << " " << ta.c_str() << ' ';
|
||||||
if (match != expected)
|
if (match != expected)
|
||||||
cout << (expected ? " should match " : " should not match ");
|
std::cout << (expected ? " should match " : " should not match ");
|
||||||
else
|
else
|
||||||
cout << (expected ? " matches " : " unmatches ");
|
std::cout << (expected ? " matches " : " unmatches ");
|
||||||
cout << tb.c_str() << endl;
|
std::cout << tb.c_str() << std::endl;
|
||||||
return expected == match;
|
return expected == match;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user