Compare commits
64 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
87a78c549f | ||
|
|
5211360b91 | ||
|
|
e71ffefc5a | ||
|
|
b6a0dde2b1 | ||
|
|
babc391632 | ||
|
|
27bdbb9a0b | ||
|
|
6a9e158428 | ||
|
|
6fc6794dc3 | ||
|
|
1eaa514579 | ||
|
|
7af4c2ca69 | ||
|
|
a340558460 | ||
|
|
9a7db237d3 | ||
|
|
91e083e7b0 | ||
|
|
97adc985e6 | ||
|
|
6fcfc9dfc0 | ||
|
|
a6596ffc89 | ||
|
|
533ab0c70d | ||
|
|
d5dd896b45 | ||
|
|
bd7fa8f39c | ||
|
|
6395e931ce | ||
|
|
635fee6f7c | ||
|
|
dc2420d88e | ||
|
|
2fbc46cbe2 | ||
|
|
a003156ae1 | ||
|
|
913e1aa7ae | ||
|
|
8272515bd7 | ||
|
|
9a7f6a3020 | ||
|
|
fead702d9f | ||
|
|
eaf938f2fd | ||
|
|
8eefa63f45 | ||
|
|
9d48c436d8 | ||
|
|
792a28e831 | ||
|
|
9407193454 | ||
|
|
602050f309 | ||
|
|
1a70c90af2 | ||
|
|
ed4091c53e | ||
|
|
f2a805f724 | ||
|
|
3083bcf071 | ||
|
|
d01f46dbc1 | ||
|
|
39b2257619 | ||
|
|
60d385189b | ||
|
|
82c5b971e9 | ||
|
|
3f2c1c57e1 | ||
|
|
e550197d0a | ||
|
|
253bc9b3f5 | ||
|
|
96d8018960 | ||
|
|
505cacc2df | ||
|
|
62848056a2 | ||
|
|
01998e74ec | ||
|
|
5f46fd304c | ||
|
|
213d637eaf | ||
|
|
3bb2dd5a81 | ||
|
|
7d9ab6381d | ||
|
|
470cde62da | ||
|
|
3fb9b6317d | ||
|
|
ee9ad93bfd | ||
|
|
0b1a932244 | ||
|
|
972759237c | ||
|
|
cb00d7f82a | ||
|
|
0b735d22a5 | ||
|
|
9178aac02c | ||
|
|
c706fbcff2 | ||
|
|
a0c41a0ccb | ||
|
|
b780dcf99c |
28
.github/workflows/aunit.yml
vendored
Normal file
28
.github/workflows/aunit.yml
vendored
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
# See https://docs.github.com/en/actions/guides for documentation about GitHub
|
||||||
|
# Actions.
|
||||||
|
|
||||||
|
name: AUnit Tests
|
||||||
|
|
||||||
|
# Run on all branches.
|
||||||
|
on: [push]
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
|
||||||
|
runs-on: ubuntu-18.04
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
|
||||||
|
- name: Setup
|
||||||
|
run: |
|
||||||
|
cd ..
|
||||||
|
git clone https://github.com/bxparks/EpoxyDuino
|
||||||
|
git clone https://github.com/bxparks/AceRoutine
|
||||||
|
git clone https://github.com/bxparks/AUnit
|
||||||
|
git clone https://github.com/bxparks/AceCommon
|
||||||
|
git clone https://github.com/hsaturn/EspMock
|
||||||
|
- name: Verify tests
|
||||||
|
run: |
|
||||||
|
make -C tests
|
||||||
|
make -C tests runtests
|
||||||
26
.github/workflows/superlinter.yml
vendored
Normal file
26
.github/workflows/superlinter.yml
vendored
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
name: Super-Linter
|
||||||
|
|
||||||
|
# Run this workflow every time a new commit pushed to your repository
|
||||||
|
#
|
||||||
|
on: push
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
# Set the job key. The key is displayed as the job name
|
||||||
|
# when a job name is not provided
|
||||||
|
super-lint:
|
||||||
|
# Name the Job
|
||||||
|
name: Lint code base
|
||||||
|
# Set the type of machine to run on
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
# Checks out a copy of your repository on the ubuntu-latest machine
|
||||||
|
- name: Checkout code
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
|
||||||
|
# Runs the Super-Linter action
|
||||||
|
- name: Run Super-Linter
|
||||||
|
uses: github/super-linter@v3
|
||||||
|
env:
|
||||||
|
DEFAULT_BRANCH: main
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
3
.gitignore
vendored
3
.gitignore
vendored
@@ -1,2 +1,5 @@
|
|||||||
*~
|
*~
|
||||||
src/my_credentials.h
|
src/my_credentials.h
|
||||||
|
*.o
|
||||||
|
*.swp
|
||||||
|
*.out
|
||||||
|
|||||||
22
README.md
22
README.md
@@ -1,15 +1,18 @@
|
|||||||
# TinyMqtt
|
# TinyMqtt
|
||||||
|
|
||||||

|

|
||||||

|

|
||||||

|

|
||||||

|

|
||||||

|

|
||||||
|
|
||||||
ESP 8266 is a small and very capable Mqtt Broker and Client
|
ESP 8266 is a small, fast and capable Mqtt Broker and Client
|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
|
|
||||||
|
- Very (very !!) fast broker I saw it re-sent 1000 topics per second for two
|
||||||
|
clients that had subscribed (payload ~15 bytes). No topic lost.
|
||||||
|
The max I've seen was 2k msg/s (1 client 1 subscription)
|
||||||
- Act as as a mqtt broker and/or a mqtt client
|
- Act as as a mqtt broker and/or a mqtt client
|
||||||
- Mqtt 3.1.1 / Qos 0 supported
|
- Mqtt 3.1.1 / Qos 0 supported
|
||||||
- Standalone (can work without WiFi) (degraded/local mode)
|
- Standalone (can work without WiFi) (degraded/local mode)
|
||||||
@@ -23,13 +26,16 @@ ESP 8266 is a small and very capable Mqtt Broker and Client
|
|||||||
* Implement zeroconf mode (needs async)
|
* Implement zeroconf mode (needs async)
|
||||||
* Add a max_clients in MqttBroker. Used with zeroconf, there will be
|
* Add a max_clients in MqttBroker. Used with zeroconf, there will be
|
||||||
no need for having tons of clients (also RAM is the problem with many clients)
|
no need for having tons of clients (also RAM is the problem with many clients)
|
||||||
|
* Why not a 'global' TinyMqtt::loop() instead of having to call loop for all broker/clients instances
|
||||||
* Test what is the real max number of clients for broker. As far as I saw, 1k is needed per client which would make more than 30 clients critical.
|
* Test what is the real max number of clients for broker. As far as I saw, 1k is needed per client which would make more than 30 clients critical.
|
||||||
* ~~MqttMessage uses a buffer 256 bytes which is usually far than needed.~~
|
* ~~MqttMessage uses a buffer 256 bytes which is usually far than needed.~~
|
||||||
* ~~MqttClient does not support more than one subscription at time~~
|
* ~~MqttClient does not support more than one subscription at time~~
|
||||||
* MqttClient auto re-subscribe
|
* ~~MqttClient auto re-subscribe (::resubscribe works bad on broker.emqx.io)~~
|
||||||
* MqttClient auto reconnection
|
* MqttClient auto reconnection
|
||||||
* MqttClient does not callback payload...
|
* ~~MqttClient unsubscribe~~
|
||||||
|
* MqttClient does not sent payload to callback...
|
||||||
* MqttClient user/password
|
* MqttClient user/password
|
||||||
|
* Wildcards (I may implement only # as I'm not interrested by a clever and cpu consuming matching)
|
||||||
|
|
||||||
## Quickstart
|
## Quickstart
|
||||||
|
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
// vim: ts=30
|
// vim: ts=40
|
||||||
Exemple of commands that can be sent via the serial monitor to tinymqtt-test
|
Exemple of commands that can be sent via the serial monitor to tinymqtt-test
|
||||||
----------------------------------------------------------------------------
|
----------------------------------------------------------------------------
|
||||||
|
|
||||||
Commands can usually be abbreviated to their first letters.
|
Commands can usually be abbreviated to their first letters.
|
||||||
ex: cl for client, a / a.con / a.sub / a.p for publish.
|
ex: cl for client, a / a.con / a.sub / a.p for publish.
|
||||||
|
--------
|
||||||
|
|
||||||
set name value set variable name to value (later replaced)
|
set name value set variable name to value (later replaced)
|
||||||
set name if no value, then var is erased
|
set name if no value, then var is erased
|
||||||
@@ -17,9 +17,12 @@ a.publish topic [payload] send a topic with a payload
|
|||||||
a.subscribe topic subscribes to a topic
|
a.subscribe topic subscribes to a topic
|
||||||
delete a destroy the client
|
delete a destroy the client
|
||||||
|
|
||||||
----------------------------------------------------
|
* note, if 'server' is a number, then it replaces the end of the local ip.
|
||||||
|
i.e. if local ip is 192.168.1.10, connect 2.35 becomes 192.168.2.35
|
||||||
|
|
||||||
|
--------
|
||||||
example:
|
example:
|
||||||
|
--------
|
||||||
|
|
||||||
client c
|
client c
|
||||||
c.connect broker.emqx.io
|
c.connect broker.emqx.io
|
||||||
@@ -30,5 +33,7 @@ c.publish topic 15
|
|||||||
c.publish topic 20
|
c.publish topic 20
|
||||||
|
|
||||||
macro exansion example
|
macro exansion example
|
||||||
|
----------------------
|
||||||
|
|
||||||
set temp publish sensor/temperature
|
set temp publish sensor/temperature
|
||||||
c.temp 20 -> c.publish sensor/temperature 20
|
c.temp 20 -> c.publish sensor/temperature 20
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
#define TINY_MQTT_DEBUG
|
#define TINY_MQTT_DEBUG
|
||||||
#include <TinyMqtt.h> // https://github.com/hsaturn/TinyMqtt
|
#include <TinyMqtt.h> // https://github.com/hsaturn/TinyMqtt
|
||||||
#include <MqttStreaming.h>
|
#include <MqttStreaming.h>
|
||||||
|
#include <sstream>
|
||||||
#include <map>
|
#include <map>
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -15,11 +16,6 @@
|
|||||||
* cons - Takes more memory
|
* cons - Takes more memory
|
||||||
* - a bit hard to understand
|
* - a bit hard to understand
|
||||||
*
|
*
|
||||||
* This sounds crazy: a mqtt mqtt that do not need a broker !
|
|
||||||
* The use case arise when one ESP wants to publish topics and subscribe to them at the same time.
|
|
||||||
* Without broker, the ESP won't react to its own topics.
|
|
||||||
*
|
|
||||||
* TinyMqtt mqtt allows this use case to work.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <my_credentials.h>
|
#include <my_credentials.h>
|
||||||
@@ -27,18 +23,26 @@
|
|||||||
std::string topic="sensor/temperature";
|
std::string topic="sensor/temperature";
|
||||||
|
|
||||||
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)
|
||||||
{ Serial << "--> " << srce->id().c_str() << ": ======> received " << topic.c_str() << endl; }
|
{
|
||||||
|
Serial << "--> " << srce->id().c_str() << ": ======> received " << topic.c_str();
|
||||||
|
if (payload) Serial << ", payload[" << length << "]=[";
|
||||||
|
while(length--)
|
||||||
|
{
|
||||||
|
const char c=*payload++;
|
||||||
|
if (c!=10 and c!=13 and c <32) Serial << '?';
|
||||||
|
Serial << *payload++;
|
||||||
|
}
|
||||||
|
Serial<< endl;
|
||||||
|
}
|
||||||
|
|
||||||
std::map<std::string, MqttClient*> clients;
|
std::map<std::string, MqttClient*> clients;
|
||||||
std::map<std::string, MqttBroker*> brokers;
|
std::map<std::string, MqttBroker*> brokers;
|
||||||
|
|
||||||
|
|
||||||
void setup()
|
void setup()
|
||||||
{
|
{
|
||||||
Serial.begin(115200);
|
Serial.begin(115200);
|
||||||
delay(500);
|
delay(500);
|
||||||
Serial << endl << endl << endl
|
Serial << endl << endl << endl
|
||||||
<< "Demo started. Type help for more..." << endl
|
|
||||||
<< "Connecting to '" << ssid << "' ";
|
<< "Connecting to '" << ssid << "' ";
|
||||||
|
|
||||||
WiFi.mode(WIFI_STA);
|
WiFi.mode(WIFI_STA);
|
||||||
@@ -48,6 +52,7 @@ void setup()
|
|||||||
{ Serial << '-'; delay(500); }
|
{ Serial << '-'; delay(500); }
|
||||||
|
|
||||||
Serial << "Connected to " << ssid << "IP address: " << WiFi.localIP() << endl;
|
Serial << "Connected to " << ssid << "IP address: " << WiFi.localIP() << endl;
|
||||||
|
Serial << "Type help for more..." << endl;
|
||||||
|
|
||||||
MqttBroker* broker = new MqttBroker(1883);
|
MqttBroker* broker = new MqttBroker(1883);
|
||||||
broker->begin();
|
broker->begin();
|
||||||
@@ -78,6 +83,45 @@ std::string getword(std::string& str, const char* if_empty=nullptr, char sep=' '
|
|||||||
return sword;
|
return sword;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool isaddr(std::string s)
|
||||||
|
{
|
||||||
|
if (s.length()==0 or s.length()>3) return false;
|
||||||
|
for(char c: s)
|
||||||
|
if (c<'0' or c>'9') return false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string getip(std::string& str, const char* if_empty=nullptr, char sep=' ')
|
||||||
|
{
|
||||||
|
std::string addr=getword(str, if_empty, sep);
|
||||||
|
std::string ip=addr;
|
||||||
|
std::vector<std::string> build;
|
||||||
|
while(ip.length())
|
||||||
|
{
|
||||||
|
std::string b=getword(ip,nullptr,'.');
|
||||||
|
if (isaddr(b) && build.size()<4)
|
||||||
|
{
|
||||||
|
build.push_back(b);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return addr;
|
||||||
|
}
|
||||||
|
IPAddress local=WiFi.localIP();
|
||||||
|
addr="";
|
||||||
|
while(build.size()!=4)
|
||||||
|
{
|
||||||
|
std::stringstream b;
|
||||||
|
b << (int)local[3-build.size()];
|
||||||
|
build.insert(build.begin(), b.str());
|
||||||
|
}
|
||||||
|
for(std::string s: build)
|
||||||
|
{
|
||||||
|
if (addr.length()) addr += '.';
|
||||||
|
addr += s;
|
||||||
|
}
|
||||||
|
return addr;
|
||||||
|
}
|
||||||
|
|
||||||
std::map<std::string, std::string> vars;
|
std::map<std::string, std::string> vars;
|
||||||
|
|
||||||
std::set<std::string> commands = {
|
std::set<std::string> commands = {
|
||||||
@@ -270,6 +314,12 @@ using ClientFunction = void(*)(std::string& cmd, MqttClient* publish);
|
|||||||
|
|
||||||
void loop()
|
void loop()
|
||||||
{
|
{
|
||||||
|
static long count;
|
||||||
|
if (MqttClient::counter != count)
|
||||||
|
{
|
||||||
|
Serial << "# " << MqttClient::counter << endl;
|
||||||
|
count = MqttClient::counter;
|
||||||
|
}
|
||||||
for(auto it: brokers)
|
for(auto it: brokers)
|
||||||
it.second->loop();
|
it.second->loop();
|
||||||
|
|
||||||
@@ -293,8 +343,7 @@ void loop()
|
|||||||
else
|
else
|
||||||
last_cmd=cmd;
|
last_cmd=cmd;
|
||||||
|
|
||||||
replaceVars(cmd);
|
if (cmd.substr(0,3)!="set") replaceVars(cmd);
|
||||||
Serial << "---------------@[ " << cmd.c_str() << " ]--------------" << endl;
|
|
||||||
while(cmd.length())
|
while(cmd.length())
|
||||||
{
|
{
|
||||||
MqttError retval = MqttOk;
|
MqttError retval = MqttOk;
|
||||||
@@ -390,7 +439,7 @@ void loop()
|
|||||||
{
|
{
|
||||||
if (compare(s,"connect"))
|
if (compare(s,"connect"))
|
||||||
{
|
{
|
||||||
client->connect(getword(cmd,"192.168.1.40").c_str(), getint(cmd, 1883), getint(cmd, 60));
|
client->connect(getip(cmd,"192.168.1.40").c_str(), getint(cmd, 1883), getint(cmd, 60));
|
||||||
Serial << (client->connected() ? "connected." : "not connected") << endl;
|
Serial << (client->connected() ? "connected." : "not connected") << endl;
|
||||||
}
|
}
|
||||||
else if (compare(s,"publish"))
|
else if (compare(s,"publish"))
|
||||||
|
|||||||
@@ -6,7 +6,7 @@
|
|||||||
"type": "git",
|
"type": "git",
|
||||||
"url": "https://github.com/hsaturn/TinyMqtt.git"
|
"url": "https://github.com/hsaturn/TinyMqtt.git"
|
||||||
},
|
},
|
||||||
"version": "0.5.1",
|
"version": "0.7.1",
|
||||||
"exclude": "",
|
"exclude": "",
|
||||||
"examples": "examples/*/*.ino",
|
"examples": "examples/*/*.ino",
|
||||||
"frameworks": "arduino",
|
"frameworks": "arduino",
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
name=TinyMqtt
|
name=TinyMqtt
|
||||||
version=0.5.1
|
version=0.7.1
|
||||||
author=Francois BIOT, HSaturn, <hsaturn@gmail.com>
|
author=Francois BIOT, HSaturn, <hsaturn@gmail.com>
|
||||||
maintainer=Francois BIOT, HSaturn, <hsaturn@gmail.com>
|
maintainer=Francois BIOT, HSaturn, <hsaturn@gmail.com>
|
||||||
sentence=A tiny broker and client library for MQTT messaging.
|
sentence=A tiny broker and client library for MQTT messaging.
|
||||||
|
|||||||
@@ -15,6 +15,16 @@ class StringIndexer
|
|||||||
std::string str;
|
std::string str;
|
||||||
uint8_t used=0;
|
uint8_t used=0;
|
||||||
friend class StringIndexer;
|
friend class StringIndexer;
|
||||||
|
|
||||||
|
#if EPOXY_DUINO
|
||||||
|
public:
|
||||||
|
// Workaround to avoid coredump in Indexer::release
|
||||||
|
// when destroying a Topic after the deletion of
|
||||||
|
// StringIndexer::strings map (which can occurs only with AUnit,
|
||||||
|
// never in the ESP itself, because ESP never ends)
|
||||||
|
// (I hate static vars)
|
||||||
|
~StringCounter() { used=255; }
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
public:
|
public:
|
||||||
using index_t=uint8_t;
|
using index_t=uint8_t;
|
||||||
@@ -104,7 +114,7 @@ class IndexedString
|
|||||||
|
|
||||||
const std::string& str() const { return StringIndexer::str(index); }
|
const std::string& str() const { return StringIndexer::str(index); }
|
||||||
|
|
||||||
const StringIndexer::index_t getIndex() const { return index; }
|
const StringIndexer::index_t& getIndex() const { return index; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
StringIndexer::index_t index;
|
StringIndexer::index_t index;
|
||||||
|
|||||||
147
src/TinyMqtt.cpp
147
src/TinyMqtt.cpp
@@ -29,8 +29,8 @@ MqttClient::MqttClient(MqttBroker* parent, WiFiClient& new_client)
|
|||||||
alive = millis()+5000; // client expires after 5s if no CONNECT msg
|
alive = millis()+5000; // client expires after 5s if no CONNECT msg
|
||||||
}
|
}
|
||||||
|
|
||||||
MqttClient::MqttClient(MqttBroker* parent)
|
MqttClient::MqttClient(MqttBroker* parent, const std::string& id)
|
||||||
: parent(parent)
|
: parent(parent), clientId(id)
|
||||||
{
|
{
|
||||||
client = nullptr;
|
client = nullptr;
|
||||||
|
|
||||||
@@ -43,12 +43,17 @@ MqttClient::~MqttClient()
|
|||||||
delete client;
|
delete client;
|
||||||
}
|
}
|
||||||
|
|
||||||
void MqttClient::close()
|
void MqttClient::close(bool bSendDisconnect)
|
||||||
{
|
{
|
||||||
debug("close " << id().c_str());
|
debug("close " << id().c_str());
|
||||||
mqtt_connected = false;
|
mqtt_connected = false;
|
||||||
if (client)
|
if (client)
|
||||||
{
|
{
|
||||||
|
if (bSendDisconnect and client->connected())
|
||||||
|
{
|
||||||
|
message.create(MqttMessage::Type::Disconnect);
|
||||||
|
message.sendTo(this);
|
||||||
|
}
|
||||||
client->stop();
|
client->stop();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -120,7 +125,7 @@ void MqttBroker::loop()
|
|||||||
|
|
||||||
// for(auto it=clients.begin(); it!=clients.end(); it++)
|
// for(auto it=clients.begin(); it!=clients.end(); it++)
|
||||||
// use index because size can change during the loop
|
// use index because size can change during the loop
|
||||||
for(int i=0; i<clients.size(); i++)
|
for(size_t i=0; i<clients.size(); i++)
|
||||||
{
|
{
|
||||||
auto client = clients[i];
|
auto client = clients[i];
|
||||||
if (client->connected())
|
if (client->connected())
|
||||||
@@ -146,8 +151,10 @@ MqttError MqttBroker::publish(const MqttClient* source, const Topic& topic, Mqtt
|
|||||||
for(auto client: clients)
|
for(auto client: clients)
|
||||||
{
|
{
|
||||||
i++;
|
i++;
|
||||||
|
#if TINY_MQTT_DEBUG
|
||||||
Serial << "brk_" << (broker && broker->connected() ? "con" : "dis") <<
|
Serial << "brk_" << (broker && broker->connected() ? "con" : "dis") <<
|
||||||
" srce=" << (source->isLocal() ? "loc" : "rem") << " clt#" << i << ", local=" << client->isLocal() << ", con=" << client->connected() << endl;
|
" srce=" << (source->isLocal() ? "loc" : "rem") << " clt#" << i << ", local=" << client->isLocal() << ", con=" << client->connected() << endl;
|
||||||
|
#endif
|
||||||
bool doit = false;
|
bool doit = false;
|
||||||
if (broker && broker->connected()) // Broker is connected
|
if (broker && broker->connected()) // Broker is connected
|
||||||
{
|
{
|
||||||
@@ -166,7 +173,9 @@ MqttError MqttBroker::publish(const MqttClient* source, const Topic& topic, Mqtt
|
|||||||
// All is allowed
|
// All is allowed
|
||||||
doit = true;
|
doit = true;
|
||||||
}
|
}
|
||||||
|
#if TINY_MQTT_DEBUG
|
||||||
Serial << ", doit=" << doit << ' ';
|
Serial << ", doit=" << doit << ' ';
|
||||||
|
#endif
|
||||||
|
|
||||||
if (doit) retval = client->publish(topic, msg);
|
if (doit) retval = client->publish(topic, msg);
|
||||||
debug("");
|
debug("");
|
||||||
@@ -232,6 +241,26 @@ void MqttClient::loop()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void MqttClient::resubscribe()
|
||||||
|
{
|
||||||
|
// TODO resubscription limited to 256 bytes
|
||||||
|
if (subscriptions.size())
|
||||||
|
{
|
||||||
|
MqttMessage msg(MqttMessage::Type::Subscribe, 2);
|
||||||
|
|
||||||
|
// TODO manage packet identifier
|
||||||
|
msg.add(0);
|
||||||
|
msg.add(0);
|
||||||
|
|
||||||
|
for(auto topic: subscriptions)
|
||||||
|
{
|
||||||
|
msg.add(topic);
|
||||||
|
msg.add(0); // TODO qos
|
||||||
|
}
|
||||||
|
msg.sendTo(this); // TODO return value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
MqttError MqttClient::subscribe(Topic topic, uint8_t qos)
|
MqttError MqttClient::subscribe(Topic topic, uint8_t qos)
|
||||||
{
|
{
|
||||||
debug("subsribe(" << topic.c_str() << ")");
|
debug("subsribe(" << topic.c_str() << ")");
|
||||||
@@ -239,33 +268,54 @@ MqttError MqttClient::subscribe(Topic topic, uint8_t qos)
|
|||||||
|
|
||||||
subscriptions.insert(topic);
|
subscriptions.insert(topic);
|
||||||
|
|
||||||
if (parent==nullptr) // remote broker ?
|
if (parent==nullptr) // remote broker
|
||||||
{
|
{
|
||||||
debug("remote subscribe");
|
return sendTopic(topic, MqttMessage::Type::Subscribe, qos);
|
||||||
MqttMessage msg(MqttMessage::Type::Subscribe, 2);
|
|
||||||
|
|
||||||
// TODO manage packet identifier
|
|
||||||
msg.add(0);
|
|
||||||
msg.add(0);
|
|
||||||
|
|
||||||
msg.add(topic.str());
|
|
||||||
msg.add(qos);
|
|
||||||
ret = msg.sendTo(this);
|
|
||||||
|
|
||||||
// TODO we should wait (state machine) for SUBACK
|
|
||||||
}
|
}
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
MqttError MqttClient::unsubscribe(Topic topic)
|
||||||
|
{
|
||||||
|
auto it=subscriptions.find(topic);
|
||||||
|
if (it != subscriptions.end())
|
||||||
|
{
|
||||||
|
subscriptions.erase(it);
|
||||||
|
if (parent==nullptr) // remote broker
|
||||||
|
{
|
||||||
|
return sendTopic(topic, MqttMessage::Type::UnSubscribe, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return MqttOk;
|
||||||
|
}
|
||||||
|
|
||||||
|
MqttError MqttClient::sendTopic(const Topic& topic, MqttMessage::Type type, uint8_t qos)
|
||||||
|
{
|
||||||
|
MqttMessage msg(type, 2);
|
||||||
|
|
||||||
|
// TODO manage packet identifier
|
||||||
|
msg.add(0);
|
||||||
|
msg.add(0);
|
||||||
|
|
||||||
|
msg.add(topic);
|
||||||
|
msg.add(qos);
|
||||||
|
|
||||||
|
// TODO instead we should wait (state machine) for SUBACK / UNSUBACK ?
|
||||||
|
return msg.sendTo(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
long MqttClient::counter=0;
|
||||||
|
|
||||||
void MqttClient::processMessage()
|
void MqttClient::processMessage()
|
||||||
{
|
{
|
||||||
std::string error;
|
counter++;
|
||||||
std::string s;
|
#if TINY_MQTT_DEBUG
|
||||||
if (message.type() != MqttMessage::Type::PingReq && message.type() != MqttMessage::Type::PingResp)
|
if (message.type() != MqttMessage::Type::PingReq && message.type() != MqttMessage::Type::PingResp)
|
||||||
{
|
{
|
||||||
Serial << "---> INCOMING " << _HEX(message.type()) << " client(" << (int)client << ':' << clientId << ") mem=" << ESP.getFreeHeap() << endl;
|
Serial << "---> INCOMING " << _HEX(message.type()) << " client(" << (int)client << ':' << clientId << ") mem=" << ESP.getFreeHeap() << endl;
|
||||||
message.hexdump("Incoming");
|
// message.hexdump("Incoming");
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
auto header = message.getVHeader();
|
auto header = message.getVHeader();
|
||||||
const char* payload;
|
const char* payload;
|
||||||
uint16_t len;
|
uint16_t len;
|
||||||
@@ -295,20 +345,6 @@ if (message.type() != MqttMessage::Type::PingReq && message.type() != MqttMessag
|
|||||||
|
|
||||||
// ClientId
|
// ClientId
|
||||||
message.getString(payload, len);
|
message.getString(payload, len);
|
||||||
debug("client id len=" << len);
|
|
||||||
if (len>30)
|
|
||||||
{
|
|
||||||
Serial << '(';
|
|
||||||
for(int i=0; i<30; i++)
|
|
||||||
{
|
|
||||||
if (i%5==0) Serial << ' ';
|
|
||||||
char c=*(header+i);
|
|
||||||
Serial << (c < 32 ? '.' : c);
|
|
||||||
}
|
|
||||||
Serial << " )" << endl;
|
|
||||||
debug("Bad client id length");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
clientId = std::string(payload, len);
|
clientId = std::string(payload, len);
|
||||||
payload += len;
|
payload += len;
|
||||||
|
|
||||||
@@ -348,9 +384,9 @@ if (message.type() != MqttMessage::Type::PingReq && message.type() != MqttMessag
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case MqttMessage::Type::ConnAck:
|
case MqttMessage::Type::ConnAck:
|
||||||
// TODO what more on connack ?
|
|
||||||
mqtt_connected = true;
|
mqtt_connected = true;
|
||||||
bclose = false;
|
bclose = false;
|
||||||
|
resubscribe();
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case MqttMessage::Type::SubAck:
|
case MqttMessage::Type::SubAck:
|
||||||
@@ -380,6 +416,7 @@ if (message.type() != MqttMessage::Type::PingReq && message.type() != MqttMessag
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case MqttMessage::Type::Subscribe:
|
case MqttMessage::Type::Subscribe:
|
||||||
|
case MqttMessage::Type::UnSubscribe:
|
||||||
{
|
{
|
||||||
if (!mqtt_connected) break;
|
if (!mqtt_connected) break;
|
||||||
payload = header+2;
|
payload = header+2;
|
||||||
@@ -391,7 +428,15 @@ if (message.type() != MqttMessage::Type::PingReq && message.type() != MqttMessag
|
|||||||
debug( " topic (" << std::string(payload, len) << ')');
|
debug( " topic (" << std::string(payload, len) << ')');
|
||||||
outstring("Subscribes", payload, len);
|
outstring("Subscribes", payload, len);
|
||||||
// subscribe(Topic(payload, len));
|
// subscribe(Topic(payload, len));
|
||||||
subscriptions.insert(Topic(payload, len));
|
Topic topic(payload, len);
|
||||||
|
if ((message.type() & 0XF0) == MqttMessage::Type::Subscribe)
|
||||||
|
subscriptions.insert(topic);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
auto it=subscriptions.find(topic);
|
||||||
|
if (it != subscriptions.end())
|
||||||
|
subscriptions.erase(it);
|
||||||
|
}
|
||||||
payload += len;
|
payload += len;
|
||||||
uint8_t qos = *payload++;
|
uint8_t qos = *payload++;
|
||||||
debug(" qos=" << qos);
|
debug(" qos=" << qos);
|
||||||
@@ -425,11 +470,21 @@ if (message.type() != MqttMessage::Type::PingReq && message.type() != MqttMessag
|
|||||||
{
|
{
|
||||||
callback(this, published, nullptr, 0); // TODO send the real payload
|
callback(this, published, nullptr, 0); // TODO send the real payload
|
||||||
}
|
}
|
||||||
// TODO should send PUBACK
|
message.create(MqttMessage::Type::PubAck);
|
||||||
|
// TODO re-add packet identifier if any
|
||||||
|
message.sendTo(this);
|
||||||
bclose = false;
|
bclose = false;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case MqttMessage::Type::Disconnect:
|
||||||
|
// TODO should discard any will message
|
||||||
|
if (!mqtt_connected) break;
|
||||||
|
mqtt_connected = false;
|
||||||
|
close(false);
|
||||||
|
bclose=false;
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
bclose=true;
|
bclose=true;
|
||||||
break;
|
break;
|
||||||
@@ -437,7 +492,7 @@ if (message.type() != MqttMessage::Type::PingReq && message.type() != MqttMessag
|
|||||||
if (bclose)
|
if (bclose)
|
||||||
{
|
{
|
||||||
Serial << "*************** Error msg 0x" << _HEX(message.type());
|
Serial << "*************** Error msg 0x" << _HEX(message.type());
|
||||||
if (error.length()) Serial << ':' << error.c_str();
|
message.hexdump("-------ERROR ------");
|
||||||
Serial << endl;
|
Serial << endl;
|
||||||
close();
|
close();
|
||||||
}
|
}
|
||||||
@@ -477,10 +532,9 @@ MqttError MqttClient::publish(const Topic& topic, MqttMessage& msg)
|
|||||||
debug("mqttclient publish " << subscriptions.size());
|
debug("mqttclient publish " << subscriptions.size());
|
||||||
for(const auto& subscription: subscriptions)
|
for(const auto& subscription: subscriptions)
|
||||||
{
|
{
|
||||||
Serial << " client=" << (int32_t)client << ", topic " << topic.str().c_str() << ' ';
|
|
||||||
if (subscription.matches(topic))
|
if (subscription.matches(topic))
|
||||||
{
|
{
|
||||||
Serial << " match/send";
|
debug(" match client=" << (int32_t)client << ", topic " << topic.str().c_str() << ' ');
|
||||||
if (client)
|
if (client)
|
||||||
{
|
{
|
||||||
retval = msg.sendTo(this);
|
retval = msg.sendTo(this);
|
||||||
@@ -490,7 +544,6 @@ MqttError MqttClient::publish(const Topic& topic, MqttMessage& msg)
|
|||||||
callback(this, topic, nullptr, 0); // TODO Payload
|
callback(this, topic, nullptr, 0); // TODO Payload
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Serial << endl;
|
|
||||||
}
|
}
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
@@ -522,8 +575,15 @@ void MqttMessage::incoming(char in_byte)
|
|||||||
vheader = buffer.length();
|
vheader = buffer.length();
|
||||||
if (size==0)
|
if (size==0)
|
||||||
state = Complete;
|
state = Complete;
|
||||||
|
else if (size > 500) // TODO magic
|
||||||
|
{
|
||||||
|
state = Error;
|
||||||
|
}
|
||||||
else
|
else
|
||||||
|
{
|
||||||
|
buffer.reserve(size);
|
||||||
state = VariableHeader;
|
state = VariableHeader;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case VariableHeader:
|
case VariableHeader:
|
||||||
@@ -541,6 +601,7 @@ void MqttMessage::incoming(char in_byte)
|
|||||||
case Complete:
|
case Complete:
|
||||||
default:
|
default:
|
||||||
Serial << "Spurious " << _HEX(in_byte) << endl;
|
Serial << "Spurious " << _HEX(in_byte) << endl;
|
||||||
|
hexdump("spurious");
|
||||||
reset();
|
reset();
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -574,11 +635,11 @@ void MqttMessage::encodeLength(char* msb, int length)
|
|||||||
|
|
||||||
MqttError MqttMessage::sendTo(MqttClient* client)
|
MqttError MqttMessage::sendTo(MqttClient* client)
|
||||||
{
|
{
|
||||||
if (buffer.size()>2)
|
if (buffer.size())
|
||||||
{
|
{
|
||||||
debug("sending " << buffer.size() << " bytes");
|
debug("sending " << buffer.size() << " bytes");
|
||||||
encodeLength(&buffer[1], buffer.size()-2);
|
encodeLength(&buffer[1], buffer.size()-2);
|
||||||
hexdump("snd");
|
// hexdump("snd");
|
||||||
client->write(&buffer[0], buffer.size());
|
client->write(&buffer[0], buffer.size());
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
|||||||
@@ -5,11 +5,11 @@
|
|||||||
#include "StringIndexer.h"
|
#include "StringIndexer.h"
|
||||||
#include <MqttStreaming.h>
|
#include <MqttStreaming.h>
|
||||||
|
|
||||||
#define TINY_MQTT_DEBUG
|
#if 0
|
||||||
|
|
||||||
#ifdef TINY_MQTT_DEBUG
|
|
||||||
#define debug(what) { Serial << __LINE__ << ' ' << what << endl; delay(100); }
|
#define debug(what) { Serial << __LINE__ << ' ' << what << endl; delay(100); }
|
||||||
|
#define TINY_MQTT_DEBUG 1
|
||||||
#else
|
#else
|
||||||
|
#define TINY_MQTT_DEBUG 0
|
||||||
#define debug(what) {}
|
#define debug(what) {}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@@ -39,15 +39,17 @@ class MqttMessage
|
|||||||
public:
|
public:
|
||||||
enum Type
|
enum Type
|
||||||
{
|
{
|
||||||
Unknown = 0,
|
Unknown = 0,
|
||||||
Connect = 0x10,
|
Connect = 0x10,
|
||||||
ConnAck = 0x20,
|
ConnAck = 0x20,
|
||||||
Publish = 0x30,
|
Publish = 0x30,
|
||||||
PubAck = 0x40,
|
PubAck = 0x40,
|
||||||
Subscribe = 0x80,
|
Subscribe = 0x80,
|
||||||
SubAck = 0x90,
|
SubAck = 0x90,
|
||||||
PingReq = 0xC0,
|
UnSubscribe = 0xA0,
|
||||||
PingResp = 0xD0,
|
PingReq = 0xC0,
|
||||||
|
PingResp = 0xD0,
|
||||||
|
Disconnect = 0xE0
|
||||||
};
|
};
|
||||||
enum State
|
enum State
|
||||||
{
|
{
|
||||||
@@ -118,8 +120,8 @@ class MqttClient
|
|||||||
FlagReserved = 1
|
FlagReserved = 1
|
||||||
};
|
};
|
||||||
public:
|
public:
|
||||||
MqttClient(MqttBroker*);
|
MqttClient(MqttBroker* brk = nullptr, const std::string& id="");
|
||||||
MqttClient() : MqttClient(nullptr) {};
|
MqttClient(const std::string& id) : MqttClient(nullptr, id){}
|
||||||
|
|
||||||
~MqttClient();
|
~MqttClient();
|
||||||
|
|
||||||
@@ -136,7 +138,7 @@ class MqttClient
|
|||||||
void id(std::string& new_id) { clientId = new_id; }
|
void id(std::string& new_id) { clientId = new_id; }
|
||||||
|
|
||||||
void loop();
|
void loop();
|
||||||
void close();
|
void close(bool bSendDisconnect=true);
|
||||||
void setCallback(CallBack fun) {callback=fun; };
|
void setCallback(CallBack fun) {callback=fun; };
|
||||||
|
|
||||||
// Publish from client to the world
|
// Publish from client to the world
|
||||||
@@ -146,23 +148,21 @@ class MqttClient
|
|||||||
MqttError publish(const Topic& t) { return publish(t, nullptr, 0);};
|
MqttError publish(const Topic& t) { return publish(t, nullptr, 0);};
|
||||||
|
|
||||||
MqttError subscribe(Topic topic, uint8_t qos=0);
|
MqttError subscribe(Topic topic, uint8_t qos=0);
|
||||||
MqttError unsubscribe(Topic& topic);
|
MqttError unsubscribe(Topic topic);
|
||||||
|
|
||||||
// connected to local broker
|
// connected to local broker
|
||||||
// TODO seems to be useless
|
// TODO seems to be useless
|
||||||
bool isLocal() const { return client == nullptr; }
|
bool isLocal() const { return client == nullptr; }
|
||||||
|
|
||||||
#ifdef TINY_MQTT_DEBUG
|
|
||||||
void dump()
|
void dump()
|
||||||
{
|
{
|
||||||
uint32_t ms=millis();
|
uint32_t ms=millis();
|
||||||
Serial << "MqttClient (" << clientId.c_str() << ") p=" << (int32_t) parent
|
Serial << "MqttClient (" << clientId.c_str() << ") " << (connected() ? " ON " : " OFF");
|
||||||
<< " c=" << (int32_t)client << (connected() ? " ON " : " OFF");
|
Serial << ", alive=" << alive << '/' << ms << ", ka=" << keep_alive;
|
||||||
Serial << ", alive=" << (uint32_t)alive << '/' << ms << ", ka=" << keep_alive;
|
Serial << (client && client->connected() ? "" : "dis") << "connected";
|
||||||
Serial << " cnx " << (client && client->connected());
|
|
||||||
Serial << " [";
|
|
||||||
message.hexdump("entrant msg");
|
message.hexdump("entrant msg");
|
||||||
bool c=false;
|
bool c=false;
|
||||||
|
Serial << " [";
|
||||||
for(auto s: subscriptions)
|
for(auto s: subscriptions)
|
||||||
{
|
{
|
||||||
Serial << (c?", ": "")<< s.str().c_str();
|
Serial << (c?", ": "")<< s.str().c_str();
|
||||||
@@ -172,9 +172,13 @@ class MqttClient
|
|||||||
|
|
||||||
Serial << "]" << endl;
|
Serial << "]" << endl;
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
static long counter; // Number of messages sent
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
MqttError sendTopic(const Topic& topic, MqttMessage::Type type, uint8_t qos);
|
||||||
|
void resubscribe();
|
||||||
|
|
||||||
friend class MqttBroker;
|
friend class MqttBroker;
|
||||||
MqttClient(MqttBroker* parent, WiFiClient& client);
|
MqttClient(MqttBroker* parent, WiFiClient& client);
|
||||||
// republish a received publish if topic matches any in subscriptions
|
// republish a received publish if topic matches any in subscriptions
|
||||||
@@ -216,12 +220,13 @@ class MqttBroker
|
|||||||
void begin() { server.begin(); }
|
void begin() { server.begin(); }
|
||||||
void loop();
|
void loop();
|
||||||
|
|
||||||
uint8_t port() const { return server.port(); }
|
uint16_t port() const { return server.port(); }
|
||||||
|
|
||||||
void connect(std::string host, uint32_t port=1883);
|
void connect(std::string host, uint16_t port=1883);
|
||||||
bool connected() const { return state == Connected; }
|
bool connected() const { return state == Connected; }
|
||||||
|
|
||||||
#ifdef TINY_MQTT_DEBUG
|
size_t clientsCount() const { return clients.size(); }
|
||||||
|
|
||||||
void dump()
|
void dump()
|
||||||
{
|
{
|
||||||
Serial << clients.size() << " client/s" << endl;
|
Serial << clients.size() << " client/s" << endl;
|
||||||
@@ -231,7 +236,6 @@ class MqttBroker
|
|||||||
client->dump();
|
client->dump();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
friend class MqttClient;
|
friend class MqttClient;
|
||||||
|
|||||||
20
tests/Makefile
Normal file
20
tests/Makefile
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
tests:
|
||||||
|
set -e; \
|
||||||
|
for i in *-tests/Makefile; do \
|
||||||
|
echo '==== Making:' $$(dirname $$i); \
|
||||||
|
$(MAKE) -C $$(dirname $$i) -j; \
|
||||||
|
done
|
||||||
|
|
||||||
|
runtests:
|
||||||
|
set -e; \
|
||||||
|
for i in *-tests/Makefile; do \
|
||||||
|
echo '==== Running:' $$(dirname $$i); \
|
||||||
|
$$(dirname $$i)/$$(dirname $$i).out; \
|
||||||
|
done
|
||||||
|
|
||||||
|
clean:
|
||||||
|
set -e; \
|
||||||
|
for i in *-tests/Makefile; do \
|
||||||
|
echo '==== Cleaning:' $$(dirname $$i); \
|
||||||
|
$(MAKE) -C $$(dirname $$i) clean; \
|
||||||
|
done
|
||||||
6
tests/local-tests/Makefile
Normal file
6
tests/local-tests/Makefile
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
||||||
|
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
||||||
|
|
||||||
|
APP_NAME := local-tests
|
||||||
|
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock
|
||||||
|
include ../../../EpoxyDuino/EpoxyDuino.mk
|
||||||
146
tests/local-tests/local-tests.ino
Normal file
146
tests/local-tests/local-tests.ino
Normal file
@@ -0,0 +1,146 @@
|
|||||||
|
#include <AUnit.h>
|
||||||
|
#include <TinyMqtt.h>
|
||||||
|
#include <map>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* TinyMqtt local unit tests.
|
||||||
|
*
|
||||||
|
* Clients are connected to pseudo remote broker
|
||||||
|
* The remote will be 127.0.0.1:1883
|
||||||
|
* We are using 127.0.0.1 because this is simpler to test with a single ESP
|
||||||
|
* Also, this will allow to mock and thus run Action on github
|
||||||
|
**/
|
||||||
|
|
||||||
|
using namespace std;
|
||||||
|
|
||||||
|
MqttBroker broker(1883);
|
||||||
|
|
||||||
|
std::map<std::string, std::map<Topic, int>> published; // map[client_id] => map[topic] = count
|
||||||
|
|
||||||
|
void onPublish(const MqttClient* srce, const Topic& topic, const char* payload, size_t length)
|
||||||
|
{
|
||||||
|
if (srce)
|
||||||
|
published[srce->id()][topic]++;
|
||||||
|
}
|
||||||
|
|
||||||
|
test(local_client_should_unregister_when_destroyed)
|
||||||
|
{
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
{
|
||||||
|
MqttClient client;
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0); // Ensure client is not yet connected
|
||||||
|
client.connect("127.0.0.1", 1883);
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)1); // Ensure client is now connected
|
||||||
|
}
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
}
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
test(local_connect)
|
||||||
|
{
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
|
MqttClient client;
|
||||||
|
assertTrue(client.connected());
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)1);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(local_publish_should_be_dispatched)
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
|
MqttClient subscriber;
|
||||||
|
subscriber.subscribe("a/b");
|
||||||
|
subscriber.subscribe("a/c");
|
||||||
|
subscriber.setCallback(onPublish);
|
||||||
|
|
||||||
|
MqttClient publisher;
|
||||||
|
publisher.publish("a/b");
|
||||||
|
publisher.publish("a/c");
|
||||||
|
publisher.publish("a/c");
|
||||||
|
|
||||||
|
assertEqual(published.size(), (size_t)1); // 1 client has received something
|
||||||
|
assertTrue(published[""]["a/b"] == 1);
|
||||||
|
assertTrue(published[""]["a/c"] == 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(local_publish_should_be_dispatched_to_local_clients)
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
|
MqttClient subscriber_a("A");
|
||||||
|
subscriber_a.setCallback(onPublish);
|
||||||
|
subscriber_a.subscribe("a/b");
|
||||||
|
subscriber_a.subscribe("a/c");
|
||||||
|
|
||||||
|
MqttClient subscriber_b("B");
|
||||||
|
subscriber_b.setCallback(onPublish);
|
||||||
|
subscriber_b.subscribe("a/b");
|
||||||
|
|
||||||
|
MqttClient publisher;
|
||||||
|
publisher.publish("a/b");
|
||||||
|
publisher.publish("a/c");
|
||||||
|
|
||||||
|
assertEqual(published.size(), (size_t)2); // 2 clients have received something
|
||||||
|
assertTrue(published["A"]["a/b"] == 1);
|
||||||
|
assertTrue(published["A"]["a/c"] == 1);
|
||||||
|
assertTrue(published["B"]["a/b"] == 1);
|
||||||
|
assertTrue(published["B"]["a/c"] == 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(local_unsubscribe)
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
|
MqttClient subscriber;
|
||||||
|
subscriber.setCallback(onPublish);
|
||||||
|
subscriber.subscribe("a/b");
|
||||||
|
|
||||||
|
MqttClient publisher;
|
||||||
|
publisher.publish("a/b");
|
||||||
|
|
||||||
|
subscriber.unsubscribe("a/b");
|
||||||
|
|
||||||
|
publisher.publish("a/b");
|
||||||
|
publisher.publish("a/b");
|
||||||
|
|
||||||
|
assertTrue(published[""]["a/b"] == 1); // Only one publish has been received
|
||||||
|
}
|
||||||
|
|
||||||
|
test(local_nocallback_when_destroyed)
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
|
MqttClient publisher;
|
||||||
|
{
|
||||||
|
MqttClient subscriber;
|
||||||
|
subscriber.setCallback(onPublish);
|
||||||
|
subscriber.subscribe("a/b");
|
||||||
|
publisher.publish("a/b");
|
||||||
|
}
|
||||||
|
|
||||||
|
publisher.publish("a/b");
|
||||||
|
|
||||||
|
assertEqual(published.size(), (size_t)1); // Only one publish has been received
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
//----------------------------------------------------------------------------
|
||||||
|
// setup() and loop()
|
||||||
|
void setup() {
|
||||||
|
delay(1000);
|
||||||
|
Serial.begin(115200);
|
||||||
|
while(!Serial);
|
||||||
|
|
||||||
|
Serial.println("=============[ NO WIFI CONNECTION TinyMqtt TESTS ]========================");
|
||||||
|
}
|
||||||
|
|
||||||
|
void loop() {
|
||||||
|
aunit::TestRunner::run();
|
||||||
|
|
||||||
|
if (Serial.available()) ESP.reset();
|
||||||
|
}
|
||||||
6
tests/nowifi-tests/Makefile
Normal file
6
tests/nowifi-tests/Makefile
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
||||||
|
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
||||||
|
|
||||||
|
APP_NAME := nowifi-tests
|
||||||
|
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock
|
||||||
|
include ../../../EpoxyDuino/EpoxyDuino.mk
|
||||||
141
tests/nowifi-tests/nowifi-tests.ino
Normal file
141
tests/nowifi-tests/nowifi-tests.ino
Normal file
@@ -0,0 +1,141 @@
|
|||||||
|
#include <AUnit.h>
|
||||||
|
#include <TinyMqtt.h>
|
||||||
|
#include <map>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* TinyMqtt nowifi unit tests.
|
||||||
|
*
|
||||||
|
* No wifi connection unit tests.
|
||||||
|
* Checks with a local broker. Clients must connect to the local client
|
||||||
|
**/
|
||||||
|
|
||||||
|
using namespace std;
|
||||||
|
|
||||||
|
MqttBroker broker(1883);
|
||||||
|
|
||||||
|
std::map<std::string, std::map<Topic, int>> published; // map[client_id] => map[topic] = count
|
||||||
|
|
||||||
|
void onPublish(const MqttClient* srce, const Topic& topic, const char* payload, size_t length)
|
||||||
|
{
|
||||||
|
if (srce)
|
||||||
|
published[srce->id()][topic]++;
|
||||||
|
}
|
||||||
|
|
||||||
|
test(nowifi_client_should_unregister_when_destroyed)
|
||||||
|
{
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
{
|
||||||
|
MqttClient client(&broker);
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)1);
|
||||||
|
}
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(nowifi_connect)
|
||||||
|
{
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
|
MqttClient client(&broker);
|
||||||
|
assertTrue(client.connected());
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)1);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(nowifi_publish_should_be_dispatched)
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
|
MqttClient subscriber(&broker);
|
||||||
|
subscriber.subscribe("a/b");
|
||||||
|
subscriber.subscribe("a/c");
|
||||||
|
subscriber.setCallback(onPublish);
|
||||||
|
|
||||||
|
MqttClient publisher(&broker);
|
||||||
|
publisher.publish("a/b");
|
||||||
|
publisher.publish("a/c");
|
||||||
|
publisher.publish("a/c");
|
||||||
|
|
||||||
|
assertEqual(published.size(), (size_t)1); // 1 client has received something
|
||||||
|
assertTrue(published[""]["a/b"] == 1);
|
||||||
|
assertTrue(published[""]["a/c"] == 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(nowifi_publish_should_be_dispatched_to_nowifi_clients)
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
|
MqttClient subscriber_a(&broker, "A");
|
||||||
|
subscriber_a.setCallback(onPublish);
|
||||||
|
subscriber_a.subscribe("a/b");
|
||||||
|
subscriber_a.subscribe("a/c");
|
||||||
|
|
||||||
|
MqttClient subscriber_b(&broker, "B");
|
||||||
|
subscriber_b.setCallback(onPublish);
|
||||||
|
subscriber_b.subscribe("a/b");
|
||||||
|
|
||||||
|
MqttClient publisher(&broker);
|
||||||
|
publisher.publish("a/b");
|
||||||
|
publisher.publish("a/c");
|
||||||
|
|
||||||
|
assertEqual(published.size(), (size_t)2); // 2 clients have received something
|
||||||
|
assertTrue(published["A"]["a/b"] == 1);
|
||||||
|
assertTrue(published["A"]["a/c"] == 1);
|
||||||
|
assertTrue(published["B"]["a/b"] == 1);
|
||||||
|
assertTrue(published["B"]["a/c"] == 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(nowifi_unsubscribe)
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
|
MqttClient subscriber(&broker);
|
||||||
|
subscriber.setCallback(onPublish);
|
||||||
|
subscriber.subscribe("a/b");
|
||||||
|
|
||||||
|
MqttClient publisher(&broker);
|
||||||
|
publisher.publish("a/b");
|
||||||
|
|
||||||
|
subscriber.unsubscribe("a/b");
|
||||||
|
|
||||||
|
publisher.publish("a/b");
|
||||||
|
publisher.publish("a/b");
|
||||||
|
|
||||||
|
assertTrue(published[""]["a/b"] == 1); // Only one publish has been received
|
||||||
|
}
|
||||||
|
|
||||||
|
test(nowifi_nocallback_when_destroyed)
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
|
MqttClient publisher(&broker);
|
||||||
|
|
||||||
|
{
|
||||||
|
MqttClient subscriber(&broker);
|
||||||
|
subscriber.setCallback(onPublish);
|
||||||
|
subscriber.subscribe("a/b");
|
||||||
|
publisher.publish("a/b");
|
||||||
|
}
|
||||||
|
|
||||||
|
publisher.publish("a/b");
|
||||||
|
|
||||||
|
assertEqual(published.size(), (size_t)1); // Only one publish has been received
|
||||||
|
}
|
||||||
|
|
||||||
|
//----------------------------------------------------------------------------
|
||||||
|
// setup() and loop()
|
||||||
|
void setup() {
|
||||||
|
delay(1000);
|
||||||
|
Serial.begin(115200);
|
||||||
|
while(!Serial);
|
||||||
|
|
||||||
|
Serial.println("=============[ NO WIFI CONNECTION TinyMqtt TESTS ]========================");
|
||||||
|
}
|
||||||
|
|
||||||
|
void loop() {
|
||||||
|
aunit::TestRunner::run();
|
||||||
|
|
||||||
|
if (Serial.available()) ESP.reset();
|
||||||
|
}
|
||||||
6
tests/result.json
Normal file
6
tests/result.json
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
{
|
||||||
|
"schemaVersion" : 1,
|
||||||
|
"label" : "tests",
|
||||||
|
"message" : "Message content",
|
||||||
|
"color": "red"
|
||||||
|
}
|
||||||
2
tests/result.yaml
Normal file
2
tests/result.yaml
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
result: 1
|
||||||
|
insert: "passed"
|
||||||
Reference in New Issue
Block a user