Compare commits
19 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
5d07294fde | ||
|
|
088071d17f | ||
|
|
e41452edf0 | ||
|
|
f8a2e35dd9 | ||
|
|
43dbea1f17 | ||
|
|
02496bef73 | ||
|
|
e4ad27c805 | ||
|
|
37fb46ec3b | ||
|
|
245e74666e | ||
|
|
294657f2ca | ||
|
|
bf84e29831 | ||
|
|
0c7c830a26 | ||
|
|
6e601228e6 | ||
|
|
46798ff0de | ||
|
|
45fedf84c9 | ||
|
|
f9c8dca1e5 | ||
|
|
7e1586c0b5 | ||
|
|
123c5a8fa5 | ||
|
|
21fb01848d |
@@ -24,6 +24,8 @@ TinyMqtt is a small, fast and capable Mqtt Broker and Client for Esp8266 / Esp32
|
|||||||
|
|
||||||
## 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)
|
||||||
@@ -63,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.
|
||||||
|
|||||||
@@ -1,16 +1,17 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
current_version=$(git describe --tags --abbrev=0)
|
current_version=$(git describe --tags --abbrev=0)
|
||||||
cp library.json.skeleton library.json
|
if [ "$1" == "-d" ]; then
|
||||||
while ifs= read -r line; do
|
do=0
|
||||||
name=$(echo "$line" | sed "s/=.*//g")
|
shift
|
||||||
value=$(echo "$line" | cut -d= -f 2 | sed 's/"//g')
|
else
|
||||||
sed -i "s/#$name/$value/g" library.json
|
do=1
|
||||||
done < library.properties
|
fi
|
||||||
|
|
||||||
if [ "$1" == "" ]; then
|
if [ "$1" == "" ]; then
|
||||||
echo
|
echo
|
||||||
echo "Syntax: $0 {new_version}"
|
echo "Syntax: $0 [-d] {new_version} [commit message]"
|
||||||
echo
|
echo
|
||||||
|
echo " -d : dry run, generate json and update properties but do not run git commands"
|
||||||
|
echo ""
|
||||||
echo " Current version: $current_version"
|
echo " Current version: $current_version"
|
||||||
echo
|
echo
|
||||||
else
|
else
|
||||||
@@ -23,15 +24,35 @@ else
|
|||||||
grep $current_version library.properties
|
grep $current_version library.properties
|
||||||
if [ "$?" == "0" ]; then
|
if [ "$?" == "0" ]; then
|
||||||
sed -i "s/$current_version/$1/" library.properties
|
sed -i "s/$current_version/$1/" library.properties
|
||||||
if [ 0 == 1 ]; then
|
|
||||||
|
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 tag $1
|
git tag $1
|
||||||
git add library.properties
|
git add library.properties
|
||||||
git add library.json
|
git add library.json
|
||||||
git commit -m "Release $1"
|
git commit -m "Release $1 $2"
|
||||||
git push
|
git push
|
||||||
git push --tags
|
git push --tags
|
||||||
else
|
|
||||||
echo "No git operation made"
|
|
||||||
fi
|
fi
|
||||||
else
|
else
|
||||||
echo "Current version does not match library.property version, aborting"
|
echo "Current version does not match library.property version, aborting"
|
||||||
|
|||||||
@@ -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,7 +1,7 @@
|
|||||||
#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)
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
* +--------+
|
* +--------+
|
||||||
|
|||||||
@@ -3,20 +3,18 @@
|
|||||||
#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>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <map>
|
#include <map>
|
||||||
|
|
||||||
using string = TinyString;
|
|
||||||
|
|
||||||
bool echo_on = true;
|
bool echo_on = true;
|
||||||
auto green = TinyConsole::green;
|
auto green = TinyConsole::green;
|
||||||
auto red = TinyConsole::red;
|
auto red = TinyConsole::red;
|
||||||
@@ -28,23 +26,25 @@ 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) {}
|
||||||
|
|
||||||
string url;
|
string url;
|
||||||
uint16_t port;
|
uint16_t port;
|
||||||
};
|
};
|
||||||
|
|
||||||
const std::map<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,
|
||||||
@@ -57,16 +57,16 @@ const std::map<string, free_broker> list =
|
|||||||
* 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;
|
||||||
@@ -75,8 +75,8 @@ void onPublish(const MqttClient* srce, const Topic& topic, const char* payload,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
std::map<string, MqttClient*> clients;
|
std::map<string, MqttClient *> clients;
|
||||||
std::map<string, MqttBroker*> brokers;
|
std::map<string, MqttBroker *> brokers;
|
||||||
|
|
||||||
void setup()
|
void setup()
|
||||||
{
|
{
|
||||||
@@ -88,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;
|
||||||
@@ -96,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 << "' ";
|
||||||
|
|
||||||
@@ -109,123 +111,139 @@ 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();
|
||||||
}
|
}
|
||||||
|
|
||||||
string getword(string& str, const char* if_empty=nullptr, char sep=' ');
|
string getword(string &str, const char *if_empty = nullptr, char sep = ' ');
|
||||||
|
|
||||||
int getint(string& str, const int if_empty=0)
|
int getint(string &str, const int if_empty = 0)
|
||||||
{
|
{
|
||||||
string str2=str;
|
string str2 = str;
|
||||||
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;
|
||||||
}
|
}
|
||||||
|
|
||||||
string getword(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)
|
||||||
|
str.erase(0, 1);
|
||||||
string sword;
|
string sword;
|
||||||
while(str.length() and (str[0]!=sep or quote))
|
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(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;
|
||||||
}
|
}
|
||||||
|
|
||||||
string getip(string& str, const char* if_empty=nullptr, char sep=' ')
|
string getip(string &str, const char *if_empty = nullptr, char sep = ' ')
|
||||||
{
|
{
|
||||||
string addr=getword(str, if_empty, sep);
|
string addr = getword(str, if_empty, sep);
|
||||||
string ip=addr;
|
string ip = addr;
|
||||||
std::vector<string> build;
|
std::vector<string> build;
|
||||||
while(ip.length())
|
while (ip.length())
|
||||||
{
|
{
|
||||||
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().c_str());
|
build.insert(build.begin(), b.str().c_str());
|
||||||
}
|
}
|
||||||
for(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;
|
||||||
@@ -238,55 +256,57 @@ 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(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);
|
||||||
|
if (search.length() == 0)
|
||||||
|
return;
|
||||||
string matches;
|
string matches;
|
||||||
int count=0;
|
int count = 0;
|
||||||
for(string cmd: commands)
|
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, string& str, string srch, 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)) != 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(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);
|
||||||
@@ -294,30 +314,31 @@ void replaceVars(string& cmd)
|
|||||||
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(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(*)(string& cmd, MqttClient* publish);
|
using ClientFunction = void (*)(string &cmd, MqttClient *publish);
|
||||||
|
|
||||||
struct Every
|
struct Every
|
||||||
{
|
{
|
||||||
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()
|
||||||
{
|
{
|
||||||
@@ -326,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";
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -343,49 +364,49 @@ int16_t blink;
|
|||||||
|
|
||||||
std::vector<Every> everies;
|
std::vector<Every> everies;
|
||||||
|
|
||||||
void onCommand(const string& command)
|
void onCommand(const string &command)
|
||||||
{
|
{
|
||||||
Console << endl;
|
Console << endl;
|
||||||
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, string& cmd)
|
void clientConnect(MqttClient *client, string &cmd)
|
||||||
{
|
{
|
||||||
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(string& cmd)
|
void eval(string &cmd)
|
||||||
{
|
{
|
||||||
while(cmd.length())
|
while (cmd.length())
|
||||||
{
|
{
|
||||||
MqttError retval = MqttOk;
|
MqttError retval = MqttOk;
|
||||||
|
|
||||||
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('.') != 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())
|
||||||
{
|
{
|
||||||
@@ -406,9 +427,11 @@ void eval(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"))
|
else if (compare(s, "debug"))
|
||||||
{
|
{
|
||||||
#if TINY_MQTT_DEBUG
|
#if TINY_MQTT_DEBUG
|
||||||
@@ -420,14 +443,14 @@ void eval(string& cmd)
|
|||||||
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, "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())
|
||||||
@@ -443,9 +466,10 @@ void eval(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);
|
||||||
@@ -455,9 +479,10 @@ void eval(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);
|
||||||
@@ -470,7 +495,7 @@ void eval(string& cmd)
|
|||||||
}
|
}
|
||||||
else if (broker)
|
else if (broker)
|
||||||
{
|
{
|
||||||
if (compare(s,"connect"))
|
if (compare(s, "connect"))
|
||||||
{
|
{
|
||||||
Console << "NYI" << endl;
|
Console << "NYI" << endl;
|
||||||
}
|
}
|
||||||
@@ -486,15 +511,15 @@ void eval(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, "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));
|
||||||
}
|
}
|
||||||
@@ -514,27 +539,27 @@ void eval(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) << ' ';
|
||||||
}
|
}
|
||||||
@@ -548,9 +573,9 @@ void eval(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;
|
||||||
@@ -559,16 +584,16 @@ void eval(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;
|
||||||
}
|
}
|
||||||
@@ -576,12 +601,12 @@ void eval(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();
|
||||||
@@ -593,20 +618,20 @@ void eval(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;
|
||||||
@@ -619,33 +644,33 @@ void eval(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"))
|
||||||
{
|
{
|
||||||
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;
|
||||||
@@ -665,25 +690,26 @@ void eval(string& cmd)
|
|||||||
}
|
}
|
||||||
else if (compare(s, "client"))
|
else if (compare(s, "client"))
|
||||||
{
|
{
|
||||||
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;
|
||||||
@@ -703,9 +729,9 @@ void eval(string& cmd)
|
|||||||
else if (compare(s, "set"))
|
else if (compare(s, "set"))
|
||||||
{
|
{
|
||||||
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;
|
||||||
}
|
}
|
||||||
@@ -732,18 +758,18 @@ void eval(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 << endl;
|
||||||
for(auto it: clients)
|
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)
|
||||||
@@ -756,7 +782,7 @@ void eval(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;
|
||||||
@@ -790,7 +816,8 @@ void eval(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;
|
||||||
}
|
}
|
||||||
@@ -804,14 +831,15 @@ void eval(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)
|
||||||
{
|
{
|
||||||
string cmd(every.cmd);
|
string cmd(every.cmd);
|
||||||
@@ -821,13 +849,14 @@ void loop()
|
|||||||
{
|
{
|
||||||
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])
|
||||||
{
|
{
|
||||||
@@ -843,7 +872,7 @@ void loop()
|
|||||||
}
|
}
|
||||||
blink_state[out] = not blink_state[out];
|
blink_state[out] = not blink_state[out];
|
||||||
}
|
}
|
||||||
blink_bits >>=1;
|
blink_bits >>= 1;
|
||||||
out++;
|
out++;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -851,10 +880,10 @@ void loop()
|
|||||||
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();
|
||||||
|
|||||||
@@ -6,7 +6,9 @@
|
|||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/hsaturn/TinyMqtt.git"
|
"url": "https://github.com/hsaturn/TinyMqtt.git"
|
||||||
},
|
},
|
||||||
"version": "0.9.16",
|
"dependencies":
|
||||||
|
{ "hsaturn/TinyConsole" : "*" },
|
||||||
|
"version": "0.9.19",
|
||||||
"exclude": "",
|
"exclude": "",
|
||||||
"examples": "examples/*/*.ino",
|
"examples": "examples/*/*.ino",
|
||||||
"frameworks": "arduino",
|
"frameworks": "arduino",
|
||||||
|
|||||||
@@ -6,6 +6,8 @@
|
|||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/hsaturn/TinyMqtt.git"
|
"url": "https://github.com/hsaturn/TinyMqtt.git"
|
||||||
},
|
},
|
||||||
|
"dependencies":
|
||||||
|
{ #dependencies },
|
||||||
"version": "#version",
|
"version": "#version",
|
||||||
"exclude": "",
|
"exclude": "",
|
||||||
"examples": "examples/*/*.ino",
|
"examples": "examples/*/*.ino",
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
name=TinyMqtt
|
name=TinyMqtt
|
||||||
version=0.9.16
|
version=0.9.19
|
||||||
author=Francois BIOT, HSaturn, <hsaturn@gmail.com>
|
author=Francois BIOT, HSaturn, <hsaturn@gmail.com>
|
||||||
maintainer=Francois BIOT <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.
|
||||||
|
|||||||
@@ -107,6 +107,8 @@ 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);
|
||||||
|
|||||||
@@ -15,8 +15,10 @@ int TinyMqtt::debug=2;
|
|||||||
std::map<MqttMessage::Type, int> MqttClient::counters;
|
std::map<MqttMessage::Type, int> MqttClient::counters;
|
||||||
#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);
|
||||||
@@ -116,7 +118,7 @@ void MqttClient::connect(string broker, uint16_t port, uint16_t ka)
|
|||||||
#ifdef TINY_MQTT_ASYNC
|
#ifdef TINY_MQTT_ASYNC
|
||||||
tcp_client->onData(onData, this);
|
tcp_client->onData(onData, this);
|
||||||
tcp_client->onConnect(onConnect, this);
|
tcp_client->onConnect(onConnect, this);
|
||||||
tcp_client->connect(broker.c_str(), port, ka);
|
tcp_client->connect(broker.c_str(), port);
|
||||||
#else
|
#else
|
||||||
if (tcp_client->connect(broker.c_str(), port))
|
if (tcp_client->connect(broker.c_str(), port))
|
||||||
{
|
{
|
||||||
@@ -142,6 +144,7 @@ void MqttBroker::connect(const string& host, uint16_t port)
|
|||||||
if (remote_broker == nullptr) remote_broker = new MqttClient;
|
if (remote_broker == nullptr) remote_broker = new MqttClient;
|
||||||
remote_broker->connect(host, port);
|
remote_broker->connect(host, port);
|
||||||
remote_broker->local_broker = this; // Because connect removed the link
|
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)
|
||||||
@@ -211,9 +214,19 @@ 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() );
|
||||||
|
for(auto& [retained_topic, retain]: retained)
|
||||||
|
{
|
||||||
|
debug(" retained: " << retained_topic.str());
|
||||||
|
if (topic.matches(retained_topic))
|
||||||
|
{
|
||||||
|
debug(" -> sending");
|
||||||
|
client->publishIfSubscribed(retained_topic, retain.msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
if (remote_broker && remote_broker->connected())
|
if (remote_broker && remote_broker->connected())
|
||||||
{
|
{
|
||||||
return remote_broker->subscribe(topic, qos);
|
return remote_broker->subscribe(topic, qos);
|
||||||
@@ -221,10 +234,12 @@ MqttError MqttBroker::subscribe(const Topic& topic, uint8_t 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)
|
||||||
@@ -294,25 +309,26 @@ void MqttClient::clientAlive(uint32_t more_seconds)
|
|||||||
|
|
||||||
void MqttClient::loop()
|
void MqttClient::loop()
|
||||||
{
|
{
|
||||||
if (keep_alive && (millis() >= alive))
|
if (keep_alive && (millis() >= alive - 5000))
|
||||||
{
|
{
|
||||||
if (local_broker)
|
if (tcp_client && tcp_client->connected())
|
||||||
{
|
|
||||||
debug(red << "timeout client");
|
|
||||||
close();
|
|
||||||
debug(red << "closed");
|
|
||||||
}
|
|
||||||
else if (tcp_client && tcp_client->connected())
|
|
||||||
{
|
{
|
||||||
debug("pingreq");
|
debug("pingreq");
|
||||||
uint16_t pingreq = MqttMessage::Type::PingReq;
|
static MqttMessage pingreq(MqttMessage::Type::PingReq);
|
||||||
tcp_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(tcp_client && tcp_client->available()>0)
|
while(tcp_client && tcp_client->available()>0)
|
||||||
{
|
{
|
||||||
@@ -391,13 +407,13 @@ MqttError MqttClient::subscribe(Topic topic, uint8_t qos)
|
|||||||
|
|
||||||
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;
|
||||||
}
|
}
|
||||||
@@ -568,7 +584,7 @@ void MqttClient::processMessage(MqttMessage* mesg)
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
qoss.push_back(qos);
|
qoss.push_back(qos);
|
||||||
subscriptions.insert(topic);
|
subscribe(topic);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@@ -618,13 +634,13 @@ void MqttClient::processMessage(MqttMessage* mesg)
|
|||||||
#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
|
||||||
@@ -728,9 +744,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();
|
||||||
@@ -894,6 +910,35 @@ 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()));
|
||||||
|
if (retain_size==0 or msg.type() != MqttMessage::Publish) return;
|
||||||
|
if (msg.flags() & 1) // flag RETAIN
|
||||||
|
{
|
||||||
|
debug(" retaining " << topic.str());
|
||||||
|
if (retained.find(topic) == retained.end()) retainDrop();
|
||||||
|
// FIXME if payload size == 0 remove message from retained
|
||||||
|
Retain r(micros(), msg);
|
||||||
|
retained.insert({ topic, std::move(r)});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void MqttMessage::hexdump(const char* prefix) const
|
void MqttMessage::hexdump(const char* prefix) const
|
||||||
{
|
{
|
||||||
(void)prefix;
|
(void)prefix;
|
||||||
|
|||||||
@@ -121,7 +121,10 @@ 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 );
|
||||||
@@ -156,6 +159,15 @@ 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();
|
||||||
|
|
||||||
@@ -198,6 +210,8 @@ class MqttClient
|
|||||||
|
|
||||||
void connect(MqttBroker* local_broker);
|
void connect(MqttBroker* local_broker);
|
||||||
void connect(string broker, uint16_t port, uint16_t keep_alive = 10);
|
void connect(string broker, uint16_t port, uint16_t keep_alive = 10);
|
||||||
|
void connect(const IPAddress& ip, uint16_t port, 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
|
||||||
@@ -228,11 +242,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 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);
|
||||||
@@ -317,15 +331,9 @@ class MqttClient
|
|||||||
|
|
||||||
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(); }
|
||||||
@@ -334,9 +342,10 @@ class MqttBroker
|
|||||||
/** Connect the broker to a parent broker */
|
/** Connect the broker to a parent broker */
|
||||||
void connect(const string& host, uint16_t port=1883);
|
void connect(const string& host, uint16_t port=1883);
|
||||||
/** returns true if connected to another broker */
|
/** returns true if connected to another broker */
|
||||||
bool connected() const { return state == Connected; }
|
bool connected() const { return remote_broker ? remote_broker->connected() : false; }
|
||||||
|
|
||||||
size_t clientsCount() const { return clients.size(); }
|
size_t clientsCount() const { return clients.size(); }
|
||||||
|
void retain(uint8_t size) { retain_size = size; }
|
||||||
|
|
||||||
void dump(string indent="")
|
void dump(string indent="")
|
||||||
{
|
{
|
||||||
@@ -356,10 +365,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);
|
||||||
@@ -375,5 +383,25 @@ class MqttBroker
|
|||||||
const char* auth_password = "guest";
|
const char* auth_password = "guest";
|
||||||
MqttClient* remote_broker = nullptr;
|
MqttClient* remote_broker = nullptr;
|
||||||
|
|
||||||
State state = Disconnected;
|
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,7 +1,7 @@
|
|||||||
|
|
||||||
# GCC
|
# GCC
|
||||||
# CXXFLAGS = -Wextra -Wall -std=gnu++11 -fno-exceptions -fno-threadsafe-statics
|
# CXXFLAGS = -Wextra -Wall -std=gnu++11 -fno-exceptions -fno-threadsafe-statics
|
||||||
EXTRA_CXXFLAGS=-g3 -O0
|
EXTRA_CXXFLAGS=-g3 -O0 -std=c++17
|
||||||
|
|
||||||
CXXFLAGS=-D_GNU_SOURCE -Werror=return-type -std=gnu++17 -Wall -g3 -O0
|
CXXFLAGS=-D_GNU_SOURCE -Werror=return-type -std=gnu++17 -Wall -g3 -O0
|
||||||
|
|
||||||
|
|||||||
@@ -18,14 +18,14 @@ MqttBroker broker(1883);
|
|||||||
|
|
||||||
std::map<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;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -51,6 +51,7 @@ test(local_client_should_unregister_when_destroyed)
|
|||||||
|
|
||||||
test(local_client_do_not_disconnect_after_publishing_and_long_inactivity)
|
test(local_client_do_not_disconnect_after_publishing_and_long_inactivity)
|
||||||
{
|
{
|
||||||
|
published.clear();
|
||||||
EpoxyTest::set_millis(0);
|
EpoxyTest::set_millis(0);
|
||||||
MqttBroker broker(1883);
|
MqttBroker broker(1883);
|
||||||
MqttClient client(&broker, "client");
|
MqttClient client(&broker, "client");
|
||||||
@@ -121,6 +122,25 @@ 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();
|
||||||
|
|||||||
@@ -85,7 +85,7 @@ std::map<string, std::map<Topic, int>> published; // map[client_id] => map[t
|
|||||||
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;
|
||||||
@@ -107,7 +107,7 @@ void onPublish(const MqttClient* srce, const Topic& topic, const char* payload,
|
|||||||
lastLength = length;
|
lastLength = length;
|
||||||
}
|
}
|
||||||
|
|
||||||
test(network_single_broker_begin)
|
test(single_broker_begin)
|
||||||
{
|
{
|
||||||
assertEqual(WiFi.status(), WL_CONNECTED);
|
assertEqual(WiFi.status(), WL_CONNECTED);
|
||||||
|
|
||||||
@@ -119,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);
|
||||||
@@ -144,10 +144,10 @@ test(suback)
|
|||||||
assertEqual(MqttClient::counters[MqttMessage::Type::SubAck], 1);
|
assertEqual(MqttClient::counters[MqttMessage::Type::SubAck], 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
test(network_client_keep_alive_high)
|
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);
|
||||||
@@ -179,9 +179,116 @@ 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);
|
||||||
|
|
||||||
|
// 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;
|
||||||
|
remote_client.connect(broker_ip, 1883);
|
||||||
|
remote_client.setCallback(onPublish);
|
||||||
|
|
||||||
|
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);
|
||||||
|
|
||||||
|
// Should not have received anything yet
|
||||||
|
assertEqual(published.size(), (size_t)0);
|
||||||
|
|
||||||
|
// Now, remote client subscribes to topic
|
||||||
|
remote_client.subscribe("topic");
|
||||||
|
for(int i=0; i<4; i++) { broker.loop(); local_client.loop(); remote_client.loop(); };
|
||||||
|
|
||||||
|
// Check that the retained message is published
|
||||||
|
assertEqual(published.size(), (size_t)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(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);
|
||||||
@@ -197,9 +304,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);
|
||||||
|
|
||||||
@@ -228,9 +335,9 @@ test(network_one_client_one_broker_publish_and_subscribe_through_network)
|
|||||||
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);
|
||||||
|
|
||||||
@@ -247,8 +354,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");
|
||||||
@@ -264,7 +371,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);
|
||||||
{
|
{
|
||||||
@@ -278,7 +385,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);
|
||||||
|
|
||||||
@@ -287,7 +394,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);
|
||||||
@@ -307,7 +414,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);
|
||||||
@@ -332,7 +439,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);
|
||||||
@@ -352,7 +459,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);
|
||||||
@@ -371,7 +478,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();
|
||||||
|
|
||||||
@@ -389,7 +496,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 !";
|
||||||
|
|
||||||
@@ -436,7 +543,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);
|
||||||
@@ -469,7 +576,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");
|
||||||
|
|||||||
Reference in New Issue
Block a user