Compare commits

...

9 Commits

Author SHA1 Message Date
hsaturn
6e601228e6 Added compatibility with me-no-dev/ESPAsyncTCP@^1.2.2 2023-03-11 18:49:51 +01:00
hsaturn
46798ff0de Fix bug with Async Tcp 2023-03-11 18:47:41 +01:00
Francois BIOT
45fedf84c9 tinymqtt-tests.ino fix bad color after underrun 2023-02-24 00:18:34 +01:00
Francois BIOT
f9c8dca1e5 tinymqtt-tests fix prompt bug 2023-02-23 23:56:22 +01:00
Francois BIOT
7e1586c0b5 platformio, make tiny-tests example compilation ok 2023-02-23 23:08:30 +01:00
hsaturn
123c5a8fa5 Release 0.9.18 2023-02-23 20:43:27 +01:00
hsaturn
21fb01848d Release 0.9.17 2023-02-23 20:42:26 +01:00
hsaturn
66b1e71ee2 Fix depends 2023-02-23 20:36:10 +01:00
hsaturn
e5115087ea Release 0.9.16 2023-02-23 20:15:38 +01:00
7 changed files with 272 additions and 217 deletions

View File

@@ -24,6 +24,7 @@ TinyMqtt is a small, fast and capable Mqtt Broker and Client for Esp8266 / Esp32
## Features ## Features
- 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)

View File

@@ -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}"
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
echo " " sed -i "s@#$name@$value@g" library.json
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
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"
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"

View File

@@ -1,22 +1,20 @@
// 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>
#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,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) {}
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,
* 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;
@@ -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;
@@ -235,58 +253,60 @@ string getip(string& str, const char* if_empty=nullptr, char sep=' ')
std::map<string, string> vars; std::map<string, string> vars;
std::set<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(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);
@@ -820,14 +848,15 @@ void loop()
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])
{ {
@@ -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();

View File

@@ -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.15", "dependencies":
{ "hsaturn/TinyConsole" : "*" },
"version": "0.9.18",
"exclude": "", "exclude": "",
"examples": "examples/*/*.ino", "examples": "examples/*/*.ino",
"frameworks": "arduino", "frameworks": "arduino",

View File

@@ -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",

View File

@@ -1,5 +1,5 @@
name=TinyMqtt name=TinyMqtt
version=0.9.16 version=0.9.18
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.
@@ -7,5 +7,5 @@ paragraph=MQTT is a lightweight messaging protocol. This library allows to host
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

View File

@@ -116,7 +116,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))
{ {