Compare commits
74 Commits
AsyncAndWi
...
0.9.5
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
d12096ef51 | ||
|
|
ea56d21190 | ||
|
|
c802c895b6 | ||
|
|
074bca971f | ||
|
|
7bd9c27b89 | ||
|
|
3e73673302 | ||
|
|
4b12aaa198 | ||
|
|
5f9cab8992 | ||
|
|
96766f7091 | ||
|
|
354aec239f | ||
|
|
3839a0a830 | ||
|
|
0444a4c348 | ||
|
|
73207e4745 | ||
|
|
b7d44445af | ||
|
|
cce6b2ecfc | ||
|
|
883f1e27e6 | ||
|
|
e7fc147424 | ||
|
|
2147b147fc | ||
|
|
f5e9a43461 | ||
|
|
cabb56fc8c | ||
|
|
58786eb6d9 | ||
|
|
776242b259 | ||
|
|
a9d19c3218 | ||
|
|
7bd299ec07 | ||
|
|
107469cd78 | ||
|
|
709e1fd567 | ||
|
|
4eb8f18ebf | ||
|
|
d5d27c8020 | ||
|
|
da3ec41d71 | ||
|
|
5f7b4537c8 | ||
|
|
cce4fecac6 | ||
|
|
737e217172 | ||
|
|
8fe3517894 | ||
|
|
710503663a | ||
|
|
a5b8afc0bd | ||
|
|
f1d3a15498 | ||
|
|
dfd5983715 | ||
|
|
4dcc6a6cf4 | ||
|
|
b58f3e3d67 | ||
|
|
a6b3540cb8 | ||
|
|
ccbf42f81b | ||
|
|
d39c58d8f5 | ||
|
|
36dde2c063 | ||
|
|
64a05bb60b | ||
|
|
bb89fc5284 | ||
|
|
c1fd1bc907 | ||
|
|
d919188eb0 | ||
|
|
9c7f3b6b8e | ||
|
|
88c7d552cb | ||
|
|
fab242e212 | ||
|
|
56a2be621f | ||
|
|
c4cf39ab68 | ||
|
|
90dea36ab0 | ||
|
|
25a721e06a | ||
|
|
b228d35ab0 | ||
|
|
c70716a595 | ||
|
|
8d5cad5fec | ||
|
|
d5430228e5 | ||
|
|
91c1f96146 | ||
|
|
f04a2a07c0 | ||
|
|
38f306c170 | ||
|
|
024e80c9dc | ||
|
|
2249ddef7f | ||
|
|
e193929f8f | ||
|
|
e00e31de33 | ||
|
|
20292b7b7b | ||
|
|
26de3befa8 | ||
|
|
1098466055 | ||
|
|
2d3663e78c | ||
|
|
5e16282ad0 | ||
|
|
e35a43c4a4 | ||
|
|
087a203ba0 | ||
|
|
5d313bbf5e | ||
|
|
ce896f02c4 |
8
.github/workflows/aunit.yml
vendored
8
.github/workflows/aunit.yml
vendored
@@ -24,5 +24,11 @@ jobs:
|
|||||||
git clone https://github.com/hsaturn/EspMock
|
git clone https://github.com/hsaturn/EspMock
|
||||||
- name: Verify tests
|
- name: Verify tests
|
||||||
run: |
|
run: |
|
||||||
make -C tests
|
# Run tests for ESP8266
|
||||||
|
make -C tests ESP_LIBS="ESP8266WiFi ESPAsyncTCP" tests
|
||||||
make -C tests runtests
|
make -C tests runtests
|
||||||
|
make -C tests clean
|
||||||
|
# Run tests for ESP32
|
||||||
|
#make -C tests ESP_LIBS="ESP8266WiFi ESPAsyncTCP" tests
|
||||||
|
#make -C tests runtests
|
||||||
|
#make -C tests clean
|
||||||
|
|||||||
18
.github/workflows/ci.yml
vendored
Normal file
18
.github/workflows/ci.yml
vendored
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
name: "CI"
|
||||||
|
on:
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
ci:
|
||||||
|
runs-on: ubuntu-20.04
|
||||||
|
steps:
|
||||||
|
- name: Checkout this repository
|
||||||
|
uses: actions/checkout@v2.3.4
|
||||||
|
- name: Cache for arduino-ci
|
||||||
|
uses: actions/cache@v2.1.3
|
||||||
|
with:
|
||||||
|
path: |
|
||||||
|
~/.arduino15
|
||||||
|
key: ${{ runner.os }}-arduino
|
||||||
|
- name: Install nix
|
||||||
|
uses: cachix/install-nix-action@v12
|
||||||
|
- run: nix-shell -I nixpkgs=channel:nixpkgs-unstable -p arduino-ci --run "arduino-ci"
|
||||||
52
.github/workflows/superlinter.yml
vendored
52
.github/workflows/superlinter.yml
vendored
@@ -1,26 +1,52 @@
|
|||||||
name: Super-Linter
|
---
|
||||||
|
#################################
|
||||||
|
#################################
|
||||||
|
## Super Linter GitHub Actions ##
|
||||||
|
#################################
|
||||||
|
#################################
|
||||||
|
name: Lint Code Base
|
||||||
|
|
||||||
# Run this workflow every time a new commit pushed to your repository
|
|
||||||
#
|
#
|
||||||
|
# Documentation:
|
||||||
|
# https://docs.github.com/en/actions/learn-github-actions/workflow-syntax-for-github-actions
|
||||||
|
#
|
||||||
|
|
||||||
|
#############################
|
||||||
|
# Start the job on all push #
|
||||||
|
#############################
|
||||||
on: push
|
on: push
|
||||||
|
|
||||||
|
###############
|
||||||
|
# Set the Job #
|
||||||
|
###############
|
||||||
jobs:
|
jobs:
|
||||||
# Set the job key. The key is displayed as the job name
|
build:
|
||||||
# when a job name is not provided
|
|
||||||
super-lint:
|
|
||||||
# Name the Job
|
# Name the Job
|
||||||
name: Lint code base
|
name: Lint Code Base
|
||||||
# Set the type of machine to run on
|
# Set the agent to run on
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
##################
|
||||||
|
# Load all steps #
|
||||||
|
##################
|
||||||
steps:
|
steps:
|
||||||
# Checks out a copy of your repository on the ubuntu-latest machine
|
##########################
|
||||||
- name: Checkout code
|
# Checkout the code base #
|
||||||
uses: actions/checkout@v2
|
##########################
|
||||||
|
- name: Checkout Code
|
||||||
|
uses: actions/checkout@v3
|
||||||
|
with:
|
||||||
|
# Full git history is needed to get a proper
|
||||||
|
# list of changed files within `super-linter`
|
||||||
|
fetch-depth: 0
|
||||||
|
|
||||||
# Runs the Super-Linter action
|
################################
|
||||||
- name: Run Super-Linter
|
# Run Linter against code base #
|
||||||
uses: github/super-linter@v3
|
################################
|
||||||
|
- name: Lint Code Base
|
||||||
|
uses: github/super-linter/slim@v4
|
||||||
env:
|
env:
|
||||||
|
VALIDATE_ALL_CODEBASE: false
|
||||||
|
# Change to 'master' if your main branch differs
|
||||||
DEFAULT_BRANCH: main
|
DEFAULT_BRANCH: main
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|||||||
37
README.md
37
README.md
@@ -1,26 +1,36 @@
|
|||||||
# TinyMqtt
|
# TinyMqtt
|
||||||
|
|
||||||

|
[](https://github.com/hsaturn/TinyMqtt/releases)
|
||||||
[](https://github.com/hsaturn/TinyMqtt/actions/workflows/aunit.yml)
|
[](https://github.com/hsaturn/TinyMqtt/actions/workflows/aunit.yml)
|
||||||

|
[](https://github.com/hsaturn/TinyMqtt/issues)
|
||||||

|
[](https://www.espressif.com/en/products/socs/esp8266)
|
||||||

|
[](https://www.espressif.com/en/products/socs/esp32)
|
||||||

|
[](https://www.gnu.org/licenses/gpl-3.0.fr.html)
|
||||||
|
[](https://docs.oasis-open.org/mqtt/mqtt/v3.1.1/errata01/os/mqtt-v3.1.1-errata01-os-complete.html#_Toc442180822)
|
||||||
|
|
||||||
ESP 8266 is a small, fast and capable Mqtt Broker and Client
|
TinyMqtt is a small, fast and capable Mqtt Broker and Client for Esp8266 / Esp32 / Esp WROOM
|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
|
|
||||||
- Very (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). 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)
|
||||||
- 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
|
||||||
|
- Wildcards supported (+ # $ and * (even if not part of the spec...))
|
||||||
- Standalone (can work without WiFi) (degraded/local mode)
|
- Standalone (can work without WiFi) (degraded/local mode)
|
||||||
- Brokers can connect to another broker and becomes then a
|
- Brokers can connect to another broker and becomes then a
|
||||||
proxy for clients that are connected to it.
|
proxy for clients that are connected to it.
|
||||||
- zeroconf, this is a strange but very powerful mode where
|
- zeroconf, this is a strange but very powerful mode where
|
||||||
all brokers tries to connect together on the same local network.
|
all brokers tries to connect together on the same local network.
|
||||||
|
- small memory footprint (very efficient topic storage)
|
||||||
|
- long messages are supported (>127 bytes)
|
||||||
|
- TinyMQTT is largely unit tested, so once a bug is fixed, it is fixed forever
|
||||||
|
|
||||||
|
## Limitations
|
||||||
|
|
||||||
|
- Max of 255 different topics can be stored (change index_t type to allow more)
|
||||||
|
- No Qos because messages are not queued but immediately sent to clients
|
||||||
|
|
||||||
## Quickstart
|
## Quickstart
|
||||||
|
|
||||||
@@ -32,10 +42,11 @@ ESP 8266 is a small, fast and capable Mqtt Broker and Client
|
|||||||
|
|
||||||
| Example | Description |
|
| Example | Description |
|
||||||
| ------------------- | ------------------------------------------ |
|
| ------------------- | ------------------------------------------ |
|
||||||
| client-without-wifi | standalone example |
|
| [client-with-wifi](https://github.com/hsaturn/TinyMqtt/tree/main/examples/client-with-wifi/client-with-wifi.ino) | standalone example |
|
||||||
| simple-client | Connect the ESP to an external Mqtt broker |
|
| [client-without-wifi](https://github.com/hsaturn/TinyMqtt/tree/main/examples/client-without-wifi/client-without-wifi.ino) | standalone example |
|
||||||
| simple-broker | Simple Mqtt broker with your ESP |
|
| [simple-client](https://github.com/hsaturn/TinyMqtt/tree/main/examples/simple-client/simple-client.ino) | Connect the ESP to an external Mqtt broker |
|
||||||
| tinymqtt-test | Complex console example |
|
| [simple-broker](https://github.com/hsaturn/TinyMqtt/tree/main/examples/simple-broker/simple-broker.ino) | Simple Mqtt broker with your ESP |
|
||||||
|
| [tinymqtt-test](https://github.com/hsaturn/TinyMqtt/tree/main/examples/tinymqtt-test/tinymqtt-test.ino) | Complex console example |
|
||||||
|
|
||||||
- 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.
|
||||||
@@ -58,7 +69,7 @@ no need for having tons of clients (also RAM is the problem with many clients)
|
|||||||
* ~~MqttClient auto re-subscribe (::resubscribe works bad on broker.emqx.io)~~
|
* ~~MqttClient auto re-subscribe (::resubscribe works bad on broker.emqx.io)~~
|
||||||
* MqttClient auto reconnection
|
* MqttClient auto reconnection
|
||||||
* MqttClient user/password
|
* MqttClient user/password
|
||||||
* Wildcards (I may implement only # as I'm not interrested by a clever and cpu consuming matching)
|
* ~~Wildcards (I may implement only # as I'm not interrested by a clever and cpu consuming matching)~~
|
||||||
* I suspect that MqttClient::parent could be removed and replaced with a simple boolean
|
* I suspect that MqttClient::parent could be removed and replaced with a simple boolean
|
||||||
(this'll need to rewrite a few functions)
|
(this'll need to rewrite a few functions)
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
#include <TinyMqtt.h> // https://github.com/hsaturn/TinyMqtt
|
#include <TinyMqtt.h> // https://github.com/hsaturn/TinyMqtt
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* Local broker that accept connections and two local clients
|
||||||
|
*
|
||||||
*
|
*
|
||||||
* +-----------------------------+
|
* +-----------------------------+
|
||||||
* | ESP |
|
* | ESP |
|
||||||
@@ -9,9 +11,10 @@
|
|||||||
* | | +--------+ |
|
* | | +--------+ |
|
||||||
* | | ^ |
|
* | | ^ |
|
||||||
* | | | |
|
* | | | |
|
||||||
* | v v |
|
* | | | | -----
|
||||||
* | +----------+ +----------+ |
|
* | v v | ---
|
||||||
* | | internal | | internal | |
|
* | +----------+ +----------+ | -
|
||||||
|
* | | internal | | internal | +-------* Wifi
|
||||||
* | | client | | client | |
|
* | | client | | client | |
|
||||||
* | +----------+ +----------+ |
|
* | +----------+ +----------+ |
|
||||||
* | |
|
* | |
|
||||||
@@ -28,7 +31,8 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <my_credentials.h>
|
const char *ssid = "";
|
||||||
|
const char *password = "";
|
||||||
|
|
||||||
std::string topic="sensor/temperature";
|
std::string topic="sensor/temperature";
|
||||||
|
|
||||||
@@ -37,11 +41,11 @@ MqttBroker broker(1883);
|
|||||||
MqttClient mqtt_a(&broker);
|
MqttClient mqtt_a(&broker);
|
||||||
MqttClient mqtt_b(&broker);
|
MqttClient mqtt_b(&broker);
|
||||||
|
|
||||||
void onPublishA(const MqttClient* source, const Topic& topic, const char* payload, size_t length)
|
void onPublishA(const MqttClient* /* source */, const Topic& topic, const char* payload, size_t /* length */)
|
||||||
{ Serial << endl << "---------> A Received " << topic.c_str() << endl; }
|
{ Serial << "--> client A received " << topic.c_str() << ", " << payload << endl; }
|
||||||
|
|
||||||
void onPublishB(const MqttClient* source, const Topic& topic, const char* payload, size_t length)
|
void onPublishB(const MqttClient* /* source */, const Topic& topic, const char* payload, size_t /* length */)
|
||||||
{ Serial << endl << "---------> B Received " << topic.c_str() << endl; }
|
{ Serial << "--> client B Received " << topic.c_str() << ", " << payload << endl; }
|
||||||
|
|
||||||
void setup()
|
void setup()
|
||||||
{
|
{
|
||||||
@@ -49,6 +53,9 @@ void setup()
|
|||||||
delay(500);
|
delay(500);
|
||||||
Serial << "Clients with wifi " << endl;
|
Serial << "Clients with wifi " << endl;
|
||||||
|
|
||||||
|
if (strlen(ssid)==0)
|
||||||
|
Serial << "****** PLEASE EDIT THE EXAMPLE AND MODIFY ssid/password *************" << endl;
|
||||||
|
|
||||||
WiFi.mode(WIFI_STA);
|
WiFi.mode(WIFI_STA);
|
||||||
WiFi.begin(ssid, password);
|
WiFi.begin(ssid, password);
|
||||||
|
|
||||||
@@ -73,14 +80,14 @@ void loop()
|
|||||||
mqtt_b.loop();
|
mqtt_b.loop();
|
||||||
|
|
||||||
// ============= client A publish ================
|
// ============= client A publish ================
|
||||||
static const int intervalA = 5000; // publishes every 5s
|
static const int intervalA = 5000; // publishes every 5s (please avoid usage of delay())
|
||||||
static uint32_t timerA = millis() + intervalA;
|
static uint32_t timerA = millis() + intervalA;
|
||||||
|
|
||||||
if (millis() > timerA)
|
if (millis() > timerA)
|
||||||
{
|
{
|
||||||
Serial << "A is publishing " << topic.c_str() << endl;
|
Serial << "A is publishing " << topic.c_str() << endl;
|
||||||
timerA += intervalA;
|
timerA += intervalA;
|
||||||
mqtt_a.publish(topic);
|
mqtt_a.publish(topic, " sent by A");
|
||||||
}
|
}
|
||||||
|
|
||||||
// ============= client B publish ================
|
// ============= client B publish ================
|
||||||
@@ -89,8 +96,9 @@ void loop()
|
|||||||
|
|
||||||
if (millis() > timerB)
|
if (millis() > timerB)
|
||||||
{
|
{
|
||||||
|
static int temperature;
|
||||||
Serial << "B is publishing " << topic.c_str() << endl;
|
Serial << "B is publishing " << topic.c_str() << endl;
|
||||||
timerB += intervalB;
|
timerB += intervalB;
|
||||||
mqtt_b.publish(topic, std::string(String(15+millis()%10).c_str()));
|
mqtt_b.publish(topic, " sent by B: "+std::string(String(16+temperature++%6).c_str()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,11 +1,29 @@
|
|||||||
#include <TinyMqtt.h> // https://github.com/hsaturn/TinyMqtt
|
#include <TinyMqtt.h> // https://github.com/hsaturn/TinyMqtt
|
||||||
|
|
||||||
/** TinyMQTT allows a disconnected mode:
|
/** TinyMQTT allows a disconnected mode:
|
||||||
|
*
|
||||||
|
* +-----------------------------+
|
||||||
|
* | ESP |
|
||||||
|
* | +--------+ |
|
||||||
|
* | +-------->| broker | |
|
||||||
|
* | | +--------+ |
|
||||||
|
* | | ^ |
|
||||||
|
* | | | |
|
||||||
|
* | v v |
|
||||||
|
* | +----------+ +----------+ |
|
||||||
|
* | | internal | | internal | |
|
||||||
|
* | | client | | client | |
|
||||||
|
* | +----------+ +----------+ |
|
||||||
|
* | |
|
||||||
|
* +-----------------------------+
|
||||||
*
|
*
|
||||||
* In this example, local clients A and B are talking together, no need to be connected.
|
* In this example, local clients A and B are talking together, no need to be connected.
|
||||||
|
*
|
||||||
* A single ESP can use this to be able to comunicate with itself with the power
|
* A single ESP can use this to be able to comunicate with itself with the power
|
||||||
* of MQTT, and once connected still continue to work with others.
|
* of MQTT, and once connected still continue to work with others.
|
||||||
*
|
*
|
||||||
|
* The broker may still be conected if wifi is on.
|
||||||
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
std::string topic="sensor/temperature";
|
std::string topic="sensor/temperature";
|
||||||
@@ -14,11 +32,11 @@ MqttBroker broker(1883);
|
|||||||
MqttClient mqtt_a(&broker);
|
MqttClient mqtt_a(&broker);
|
||||||
MqttClient mqtt_b(&broker);
|
MqttClient mqtt_b(&broker);
|
||||||
|
|
||||||
void onPublishA(const MqttClient* srce, const Topic& topic, const char* payload, size_t length)
|
void onPublishA(const MqttClient* /* srce */, const Topic& topic, const char* payload, size_t /* length */)
|
||||||
{ Serial << "--> A Received " << topic.c_str() << endl; }
|
{ Serial << "--> Client A received msg on topic " << topic.c_str() << ", " << payload << endl; }
|
||||||
|
|
||||||
void onPublishB(const MqttClient* srce, const Topic& topic, const char* payload, size_t length)
|
void onPublishB(const MqttClient* /* srce */, const Topic& topic, const char* payload, size_t /* length */)
|
||||||
{ Serial << "--> B Received " << topic.c_str() << endl; }
|
{ Serial << "--> Client B received msg on topic " << topic.c_str() << ", " << payload << endl; }
|
||||||
|
|
||||||
void setup()
|
void setup()
|
||||||
{
|
{
|
||||||
@@ -35,7 +53,7 @@ void setup()
|
|||||||
|
|
||||||
void loop()
|
void loop()
|
||||||
{
|
{
|
||||||
broker.loop();
|
broker.loop(); // Don't forget to call loop() for all brokers and clients
|
||||||
mqtt_a.loop();
|
mqtt_a.loop();
|
||||||
mqtt_b.loop();
|
mqtt_b.loop();
|
||||||
|
|
||||||
@@ -47,7 +65,7 @@ void loop()
|
|||||||
{
|
{
|
||||||
Serial << "A is publishing " << topic.c_str() << endl;
|
Serial << "A is publishing " << topic.c_str() << endl;
|
||||||
timerA += intervalA;
|
timerA += intervalA;
|
||||||
mqtt_a.publish(topic);
|
mqtt_a.publish(topic, "sent by A");
|
||||||
}
|
}
|
||||||
|
|
||||||
// ============= client B publish ================
|
// ============= client B publish ================
|
||||||
@@ -58,6 +76,6 @@ void loop()
|
|||||||
{
|
{
|
||||||
Serial << "B is publishing " << topic.c_str() << endl;
|
Serial << "B is publishing " << topic.c_str() << endl;
|
||||||
timerB += intervalB;
|
timerB += intervalB;
|
||||||
mqtt_b.publish(topic);
|
mqtt_b.publish(topic, "sent by B");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,14 +1,33 @@
|
|||||||
#include "TinyMqtt.h" // https://github.com/hsaturn/TinyMqtt
|
#include "TinyMqtt.h" // https://github.com/hsaturn/TinyMqtt
|
||||||
|
|
||||||
#include <my_credentials.h>
|
|
||||||
|
|
||||||
#define PORT 1883
|
#define PORT 1883
|
||||||
MqttBroker broker(PORT);
|
MqttBroker broker(PORT);
|
||||||
|
|
||||||
|
/** Basic Mqtt Broker
|
||||||
|
*
|
||||||
|
* +-----------------------------+
|
||||||
|
* | ESP |
|
||||||
|
* | +--------+ |
|
||||||
|
* | | broker | | 1883 <--- External client/s
|
||||||
|
* | +--------+ |
|
||||||
|
* | |
|
||||||
|
* +-----------------------------+
|
||||||
|
*
|
||||||
|
* Your ESP will become a MqttBroker.
|
||||||
|
* You can test it with any client such as mqtt-spy for example
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
const char* ssid = "";
|
||||||
|
const char* password = "";
|
||||||
|
|
||||||
void setup()
|
void setup()
|
||||||
{
|
{
|
||||||
Serial.begin(115200);
|
Serial.begin(115200);
|
||||||
|
|
||||||
|
if (strlen(ssid)==0)
|
||||||
|
Serial << "****** PLEASE MODIFY ssid/password *************" << endl;
|
||||||
|
|
||||||
WiFi.mode(WIFI_STA);
|
WiFi.mode(WIFI_STA);
|
||||||
WiFi.begin(ssid, password);
|
WiFi.begin(ssid, password);
|
||||||
|
|
||||||
|
|||||||
@@ -1,11 +1,25 @@
|
|||||||
#include "TinyMqtt.h" // https://github.com/hsaturn/TinyMqtt
|
#include "TinyMqtt.h" // https://github.com/hsaturn/TinyMqtt
|
||||||
|
|
||||||
/** Simple Client
|
/** Simple Client (The simplest configuration)
|
||||||
*
|
*
|
||||||
* This is the simplest Mqtt client configuration
|
|
||||||
*
|
*
|
||||||
* 1 - edit my_credentials.h to setup wifi essid/password
|
* +--------+
|
||||||
|
* +------>| broker |<--- < Other client
|
||||||
|
* | +--------+
|
||||||
|
* |
|
||||||
|
* +-----------------+
|
||||||
|
* | ESP | |
|
||||||
|
* | +----------+ |
|
||||||
|
* | | internal | |
|
||||||
|
* | | client | |
|
||||||
|
* | +----------+ |
|
||||||
|
* | |
|
||||||
|
* +-----------------+
|
||||||
|
*
|
||||||
|
* 1 - change the ssid/password
|
||||||
* 2 - change BROKER values (or keep emqx.io test broker)
|
* 2 - change BROKER values (or keep emqx.io test broker)
|
||||||
|
* 3 - you can use mqtt-spy to connect to the same broker and
|
||||||
|
* see the sensor/temperature updated by the client.
|
||||||
*
|
*
|
||||||
* pro - small memory footprint (both ram and flash)
|
* pro - small memory footprint (both ram and flash)
|
||||||
* - very simple to setup and use
|
* - very simple to setup and use
|
||||||
@@ -18,7 +32,8 @@
|
|||||||
const char* BROKER = "broker.emqx.io";
|
const char* BROKER = "broker.emqx.io";
|
||||||
const uint16_t BROKER_PORT = 1883;
|
const uint16_t BROKER_PORT = 1883;
|
||||||
|
|
||||||
#include <my_credentials.h>
|
const char* ssid = "";
|
||||||
|
const char* password = "";
|
||||||
|
|
||||||
static float temp=19;
|
static float temp=19;
|
||||||
static MqttClient client;
|
static MqttClient client;
|
||||||
@@ -27,7 +42,10 @@ void setup()
|
|||||||
{
|
{
|
||||||
Serial.begin(115200);
|
Serial.begin(115200);
|
||||||
delay(500);
|
delay(500);
|
||||||
|
|
||||||
Serial << "Simple clients with wifi " << endl;
|
Serial << "Simple clients with wifi " << endl;
|
||||||
|
if (strlen(ssid)==0)
|
||||||
|
Serial << "****** PLEASE MODIFY ssid/password *************" << endl;
|
||||||
|
|
||||||
WiFi.mode(WIFI_STA);
|
WiFi.mode(WIFI_STA);
|
||||||
WiFi.begin(ssid, password);
|
WiFi.begin(ssid, password);
|
||||||
@@ -42,15 +60,28 @@ void setup()
|
|||||||
|
|
||||||
void loop()
|
void loop()
|
||||||
{
|
{
|
||||||
client.loop();
|
client.loop(); // Don't forget to call loop() for each broker and client
|
||||||
|
|
||||||
delay(1000);
|
// delay(1000); please avoid usage of delay (see below how this done using next_send and millis())
|
||||||
|
static auto next_send = millis();
|
||||||
|
|
||||||
|
if (millis() > next_send)
|
||||||
|
{
|
||||||
|
next_send += 1000;
|
||||||
|
|
||||||
|
if (not client.connected())
|
||||||
|
{
|
||||||
|
Serial << millis() << ": Not connected to broker" << endl;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
auto rnd=random(100);
|
auto rnd=random(100);
|
||||||
|
|
||||||
if (rnd > 66) temp += 0.1;
|
if (rnd > 66) temp += 0.1;
|
||||||
else if (rnd < 33) temp -= 0.1;
|
else if (rnd < 33) temp -= 0.1;
|
||||||
|
|
||||||
client.publish("sensor/temperature", String(temp));
|
Serial << "--> Publishing a new sensor/temperature value: " << temp << endl;
|
||||||
|
|
||||||
|
client.publish("sensor/temperature", String(temp));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,39 +1,53 @@
|
|||||||
|
// vim: ts=2 sw=2
|
||||||
|
#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>
|
||||||
|
#if defined(ESP8266)
|
||||||
#include <ESP8266mDNS.h>
|
#include <ESP8266mDNS.h>
|
||||||
|
#elif defined(ESP32)
|
||||||
|
#include <WiFi.h>
|
||||||
|
#include <ESPmDNS.h>
|
||||||
|
#else
|
||||||
|
#error Unsupported platform
|
||||||
|
#endif
|
||||||
|
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
#include <map>
|
#include <map>
|
||||||
|
|
||||||
/**
|
bool echo_on = true;
|
||||||
* Console allowing to make any kind of test.
|
|
||||||
|
/** Very complex example
|
||||||
|
* Console allowing to make any kind of test,
|
||||||
|
* even some stress tests.
|
||||||
*
|
*
|
||||||
* pros - Reduces internal latency (when publish is received by the same ESP)
|
* Upload the sketch, the use the terminal.
|
||||||
* - Reduces wifi traffic
|
* Press H for mini help.
|
||||||
* - No need to have an external broker
|
|
||||||
* - can still report to a 'main' broker (TODO see documentation that have to be written)
|
|
||||||
* - accepts external clients
|
|
||||||
*
|
|
||||||
* cons - Takes more memory
|
|
||||||
* - a bit hard to understand
|
|
||||||
*
|
*
|
||||||
|
* tested with mqtt-spy-0.5.4
|
||||||
|
* TODO examples of scripts
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <my_credentials.h>
|
const char* ssid = "";
|
||||||
|
const char* password = "";
|
||||||
|
|
||||||
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();
|
Serial << "--> " << srce->id().c_str() << ": ======> received " << topic.c_str();
|
||||||
if (payload) Serial << ", payload[" << length << "]=[";
|
if (payload)
|
||||||
|
{
|
||||||
|
Serial << ", payload[" << length << "]=[";
|
||||||
while(length--)
|
while(length--)
|
||||||
{
|
{
|
||||||
const char c=*payload++;
|
const char c=*payload++;
|
||||||
if (c!=10 and c!=13 and c <32) Serial << '?';
|
if (c<32)
|
||||||
Serial << *payload++;
|
Serial << '?';
|
||||||
|
else
|
||||||
|
Serial << c;
|
||||||
|
}
|
||||||
|
Serial << ']' << endl;
|
||||||
}
|
}
|
||||||
Serial<< endl;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
std::map<std::string, MqttClient*> clients;
|
std::map<std::string, MqttClient*> clients;
|
||||||
@@ -44,17 +58,32 @@ void setup()
|
|||||||
WiFi.persistent(false); // https://github.com/esp8266/Arduino/issues/1054
|
WiFi.persistent(false); // https://github.com/esp8266/Arduino/issues/1054
|
||||||
Serial.begin(115200);
|
Serial.begin(115200);
|
||||||
delay(500);
|
delay(500);
|
||||||
Serial << endl << endl << endl
|
|
||||||
<< "Connecting to '" << ssid << "' ";
|
|
||||||
|
|
||||||
|
Serial << endl << endl;
|
||||||
|
Serial << "***************************************************************" << endl;
|
||||||
|
Serial << "* Welcome to the TinyMqtt console" << endl;
|
||||||
|
Serial << endl;
|
||||||
|
Serial << "* The console allows to test all features of the libraries." << endl;
|
||||||
|
Serial << endl;
|
||||||
|
if (strlen(ssid)==0)
|
||||||
|
Serial << "* WARNING: You may want to modify ssid/password in order" << endl
|
||||||
|
<< " to reflect your Wifi configuration." << endl;
|
||||||
|
Serial << endl;
|
||||||
|
Serial << "* Enter help to view the list of commands." << endl;
|
||||||
|
Serial << "***************************************************************" << endl;
|
||||||
|
Serial << endl;
|
||||||
|
|
||||||
|
Serial << "Connecting to '" << ssid << "' ";
|
||||||
|
|
||||||
|
WiFi.mode(WIFI_STA);
|
||||||
|
WiFi.begin(ssid, password);
|
||||||
WiFi.mode(WIFI_STA);
|
WiFi.mode(WIFI_STA);
|
||||||
WiFi.begin(ssid, password);
|
WiFi.begin(ssid, password);
|
||||||
|
|
||||||
while (WiFi.status() != WL_CONNECTED)
|
while (WiFi.status() != WL_CONNECTED)
|
||||||
{ Serial << '-'; delay(500); }
|
{ Serial << '-'; delay(500); }
|
||||||
|
|
||||||
Serial << "Connected to " << ssid << "IP address: " << WiFi.localIP() << endl;
|
Serial << endl << "Connected to " << ssid << "IP address: " << WiFi.localIP() << endl;
|
||||||
Serial << "Type help for more..." << endl;
|
|
||||||
|
|
||||||
const char* name="tinytest";
|
const char* name="tinytest";
|
||||||
Serial << "Starting MDNS, name= " << name;
|
Serial << "Starting MDNS, name= " << name;
|
||||||
@@ -69,27 +98,67 @@ void setup()
|
|||||||
brokers["broker"] = broker;
|
brokers["broker"] = broker;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::string getword(std::string& str, const char* if_empty=nullptr, char sep=' ');
|
||||||
|
|
||||||
int getint(std::string& str, const int if_empty=0)
|
int getint(std::string& str, const int if_empty=0)
|
||||||
{
|
{
|
||||||
std::string sword;
|
std::string str2=str;
|
||||||
while(str.length() && str[0]>='0' && str[0]<='9')
|
std::string sword = getword(str);
|
||||||
|
if (sword[0] and isdigit(sword[0]))
|
||||||
{
|
{
|
||||||
sword += str[0]; str.erase(0,1);
|
int ret=atoi(sword.c_str());
|
||||||
|
while(isdigit(sword[0]) or sword[0]==' ') sword.erase(0,1);
|
||||||
|
if (sword.length()) str = sword+' '+str;
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
while(str[0]==' ') str.erase(0,1);
|
str=str2;
|
||||||
if (if_empty and sword.length()==0) return if_empty;
|
return if_empty;
|
||||||
return atoi(sword.c_str());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string getword(std::string& str, const char* if_empty=nullptr, char sep=' ')
|
std::string getword(std::string& str, const char* if_empty/*=nullptr*/, char sep/*=' '*/)
|
||||||
{
|
{
|
||||||
|
char quote=(str[0]=='"' or str[0]=='\'' ? str[0] : 0);
|
||||||
|
if (quote) str.erase(0,1);
|
||||||
std::string sword;
|
std::string sword;
|
||||||
while(str.length() && str[0]!=sep)
|
while(str.length() and (str[0]!=sep or quote))
|
||||||
{
|
{
|
||||||
sword += str[0]; str.erase(0,1);
|
if (str[0]==quote)
|
||||||
|
{
|
||||||
|
str.erase(0,1);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
sword += str[0];
|
||||||
|
str.erase(0,1);
|
||||||
}
|
}
|
||||||
while(str[0]==sep) str.erase(0,1);
|
while(str[0]==sep) str.erase(0,1);
|
||||||
if (if_empty and sword.length()==0) return if_empty;
|
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);
|
||||||
|
if (sword[0]=='(')
|
||||||
|
{
|
||||||
|
int to = 100;
|
||||||
|
sword.erase(0,1);
|
||||||
|
int from=getint(sword);
|
||||||
|
if (sword[0]==',')
|
||||||
|
{
|
||||||
|
sword.erase(0,1);
|
||||||
|
to = getint(sword);
|
||||||
|
if (sword[0]!=')') Serial << "Missing ')'" << endl;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
to=from;
|
||||||
|
from=0;
|
||||||
|
}
|
||||||
|
return String(random(from,to)).c_str();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Serial << "Missing '('" << endl;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
while(str[0]==' ') str.erase(0,1);
|
||||||
return sword;
|
return sword;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -117,7 +186,7 @@ std::string getip(std::string& str, const char* if_empty=nullptr, char sep=' ')
|
|||||||
return addr;
|
return addr;
|
||||||
}
|
}
|
||||||
IPAddress local=WiFi.localIP();
|
IPAddress local=WiFi.localIP();
|
||||||
addr="";
|
addr.clear();
|
||||||
while(build.size()!=4)
|
while(build.size()!=4)
|
||||||
{
|
{
|
||||||
std::stringstream b;
|
std::stringstream b;
|
||||||
@@ -138,10 +207,10 @@ std::set<std::string> commands = {
|
|||||||
"auto", "broker", "blink", "client", "connect",
|
"auto", "broker", "blink", "client", "connect",
|
||||||
"create", "delete", "help", "interval",
|
"create", "delete", "help", "interval",
|
||||||
"ls", "ip", "off", "on", "set",
|
"ls", "ip", "off", "on", "set",
|
||||||
"publish", "reset", "subscribe", "unsubscribe", "view", "every"
|
"publish", "reset", "subscribe", "unsubscribe", "view", "echo", "every"
|
||||||
};
|
};
|
||||||
|
|
||||||
void getCommand(std::string& search)
|
void convertToCommand(std::string& search)
|
||||||
{
|
{
|
||||||
while(search[0]==' ') search.erase(0,1);
|
while(search[0]==' ') search.erase(0,1);
|
||||||
if (search.length()==0) return;
|
if (search.length()==0) return;
|
||||||
@@ -161,7 +230,7 @@ void getCommand(std::string& search)
|
|||||||
else if (count>1)
|
else if (count>1)
|
||||||
{
|
{
|
||||||
Serial << "Ambiguous command: " << matches << endl;
|
Serial << "Ambiguous command: " << matches << endl;
|
||||||
search="";
|
search.clear();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -177,7 +246,7 @@ void replace(const char* d, std::string& str, std::string srch, std::string to)
|
|||||||
{
|
{
|
||||||
str.erase(pos, srch.length());
|
str.erase(pos, srch.length());
|
||||||
str.insert(pos, to);
|
str.insert(pos, to);
|
||||||
pos += to.length();
|
pos += to.length()-1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -195,6 +264,7 @@ void replaceVars(std::string& cmd)
|
|||||||
}
|
}
|
||||||
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);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// publish at regular interval
|
// publish at regular interval
|
||||||
@@ -332,9 +402,12 @@ struct Every
|
|||||||
std::string cmd;
|
std::string cmd;
|
||||||
uint32_t ms;
|
uint32_t ms;
|
||||||
uint32_t next;
|
uint32_t next;
|
||||||
|
uint32_t underrun=0;
|
||||||
|
bool active=true;
|
||||||
|
|
||||||
void dump()
|
void dump()
|
||||||
{
|
{
|
||||||
|
Serial << (active ? "enabled " : "disabled ");
|
||||||
auto mill=millis();
|
auto mill=millis();
|
||||||
Serial << ms << "ms [" << cmd << "] next in ";
|
Serial << ms << "ms [" << cmd << "] next in ";
|
||||||
if (mill > next)
|
if (mill > next)
|
||||||
@@ -363,7 +436,6 @@ void eval(std::string& cmd)
|
|||||||
MqttClient* client = nullptr;
|
MqttClient* client = nullptr;
|
||||||
|
|
||||||
// client.function notation
|
// client.function notation
|
||||||
// ("a.fun " becomes "fun a ")
|
|
||||||
if (cmd.find('.') != std::string::npos &&
|
if (cmd.find('.') != std::string::npos &&
|
||||||
cmd.find('.') < cmd.find(' '))
|
cmd.find('.') < cmd.find(' '))
|
||||||
{
|
{
|
||||||
@@ -382,13 +454,13 @@ void eval(std::string& cmd)
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
Serial << "Unknown class (" << s.c_str() << ")" << endl;
|
Serial << "Unknown class (" << s.c_str() << ")" << endl;
|
||||||
cmd="";
|
cmd.clear();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
s = getword(cmd);
|
s = getword(cmd);
|
||||||
if (s.length()) getCommand(s);
|
if (s.length()) convertToCommand(s);
|
||||||
if (s.length()==0)
|
if (s.length()==0)
|
||||||
{}
|
{}
|
||||||
else if (compare(s, "delete"))
|
else if (compare(s, "delete"))
|
||||||
@@ -444,6 +516,11 @@ void eval(std::string& cmd)
|
|||||||
{
|
{
|
||||||
broker->dump();
|
broker->dump();
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Serial << "Unknown broker command (" << s << ")" << endl;
|
||||||
|
s.clear();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else if (client)
|
else if (client)
|
||||||
{
|
{
|
||||||
@@ -454,9 +531,7 @@ void eval(std::string& cmd)
|
|||||||
}
|
}
|
||||||
else if (compare(s,"publish"))
|
else if (compare(s,"publish"))
|
||||||
{
|
{
|
||||||
while (cmd[0]==' ') cmd.erase(0,1);
|
retval = client->publish(getword(cmd, topic.c_str()), getword(cmd));
|
||||||
retval = client->publish(getword(cmd, topic.c_str()), cmd.c_str(), cmd.length());
|
|
||||||
cmd=""; // remove payload
|
|
||||||
}
|
}
|
||||||
else if (compare(s,"subscribe"))
|
else if (compare(s,"subscribe"))
|
||||||
{
|
{
|
||||||
@@ -470,6 +545,11 @@ void eval(std::string& cmd)
|
|||||||
{
|
{
|
||||||
client->dump();
|
client->dump();
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Serial << "Unknown client command (" << s << ")" << endl;
|
||||||
|
s.clear();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else if (compare(s, "on"))
|
else if (compare(s, "on"))
|
||||||
{
|
{
|
||||||
@@ -483,10 +563,28 @@ void eval(std::string& cmd)
|
|||||||
pinMode(pin, OUTPUT);
|
pinMode(pin, OUTPUT);
|
||||||
digitalWrite(pin, LOW);
|
digitalWrite(pin, LOW);
|
||||||
}
|
}
|
||||||
|
else if (compare(s, "echo"))
|
||||||
|
{
|
||||||
|
s=getword(cmd);
|
||||||
|
if (s=="on")
|
||||||
|
echo_on = true;
|
||||||
|
else if (s=="off")
|
||||||
|
echo_on = false;
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Serial << s << ' ';
|
||||||
|
while(cmd.length())
|
||||||
|
{
|
||||||
|
Serial << getword(cmd) << ' ';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
else if (compare(s, "every"))
|
else if (compare(s, "every"))
|
||||||
{
|
{
|
||||||
uint32_t ms = getint(cmd, 0);
|
uint32_t ms = getint(cmd, 0);
|
||||||
if (ms and cmd.length())
|
if (ms)
|
||||||
|
{
|
||||||
|
if (cmd.length())
|
||||||
{
|
{
|
||||||
Every every;
|
Every every;
|
||||||
every.ms=ms;
|
every.ms=ms;
|
||||||
@@ -495,9 +593,29 @@ void eval(std::string& cmd)
|
|||||||
everies.push_back(every);
|
everies.push_back(every);
|
||||||
every.dump();
|
every.dump();
|
||||||
Serial << endl;
|
Serial << endl;
|
||||||
cmd="";
|
cmd.clear();
|
||||||
}
|
}
|
||||||
else if (ms==0 and compare(cmd, "list"))
|
}
|
||||||
|
else if (compare(cmd, "off") or compare(cmd, "on"))
|
||||||
|
{
|
||||||
|
bool active=getword(cmd)=="on";
|
||||||
|
uint8_t ever=getint(cmd, 100);
|
||||||
|
uint8_t count=0;
|
||||||
|
for(auto& every: everies)
|
||||||
|
{
|
||||||
|
if (count==ever or (ever==100))
|
||||||
|
{
|
||||||
|
if (every.active != active)
|
||||||
|
{
|
||||||
|
Serial << "every #" << count << (active ? " on" :" off") << endl;
|
||||||
|
every.active = active;
|
||||||
|
every.underrun = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
count++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (compare(cmd, "list") or cmd.length()==0)
|
||||||
{
|
{
|
||||||
getword(cmd);
|
getword(cmd);
|
||||||
Serial << "List of everies (ms=" << millis() << ")" << endl;
|
Serial << "List of everies (ms=" << millis() << ")" << endl;
|
||||||
@@ -510,11 +628,17 @@ void eval(std::string& cmd)
|
|||||||
count++;
|
count++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (ms==0 and compare(cmd, "remove"))
|
else if (compare(cmd, "remove"))
|
||||||
{
|
{
|
||||||
|
Serial << "Removing..." << endl;
|
||||||
getword(cmd);
|
getword(cmd);
|
||||||
int8_t every=getint(cmd, -1);
|
int8_t every=getint(cmd, -1);
|
||||||
if (every==-1 and compare(cmd, "all"))
|
if (every==-1 and compare(cmd, "last") and everies.size())
|
||||||
|
{
|
||||||
|
getword(cmd);
|
||||||
|
everies.erase(everies.begin()+everies.size()-1);
|
||||||
|
}
|
||||||
|
else if (every==-1 and compare(cmd, "all"))
|
||||||
{
|
{
|
||||||
getword(cmd);
|
getword(cmd);
|
||||||
everies.clear();
|
everies.clear();
|
||||||
@@ -523,7 +647,11 @@ void eval(std::string& cmd)
|
|||||||
{
|
{
|
||||||
everies.erase(everies.begin()+every);
|
everies.erase(everies.begin()+every);
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
Serial << "Bad colmmand" << endl;
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
Serial << "Bad command" << endl;
|
||||||
}
|
}
|
||||||
else if (compare(s, "blink"))
|
else if (compare(s, "blink"))
|
||||||
{
|
{
|
||||||
@@ -552,7 +680,12 @@ void eval(std::string& cmd)
|
|||||||
else if (compare(s, "broker"))
|
else if (compare(s, "broker"))
|
||||||
{
|
{
|
||||||
std::string id=getword(cmd);
|
std::string id=getword(cmd);
|
||||||
if (id.length() or brokers.find(id)!=brokers.end())
|
if (clients.find(id) != clients.end())
|
||||||
|
{
|
||||||
|
Serial << "A client already have that name" << endl;
|
||||||
|
cmd.clear();
|
||||||
|
}
|
||||||
|
else if (id.length() or brokers.find(id)!=brokers.end())
|
||||||
{
|
{
|
||||||
int port=getint(cmd, 0);
|
int port=getint(cmd, 0);
|
||||||
if (port)
|
if (port)
|
||||||
@@ -564,16 +697,26 @@ void eval(std::string& cmd)
|
|||||||
Serial << "new broker (" << id.c_str() << ")" << endl;
|
Serial << "new broker (" << id.c_str() << ")" << endl;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
{
|
||||||
Serial << "Missing port" << endl;
|
Serial << "Missing port" << endl;
|
||||||
|
cmd.clear();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
{
|
||||||
Serial << "Missing or existing broker name (" << id.c_str() << ")" << endl;
|
Serial << "Missing or existing broker name (" << id.c_str() << ")" << endl;
|
||||||
cmd+=" ls";
|
cmd.clear();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else if (compare(s, "client"))
|
else if (compare(s, "client"))
|
||||||
{
|
{
|
||||||
std::string id=getword(cmd);
|
std::string id=getword(cmd);
|
||||||
if (id.length() or clients.find(id)!=clients.end())
|
if (brokers.find(id) != brokers.end())
|
||||||
|
{
|
||||||
|
Serial << "A broker have that name" << endl;
|
||||||
|
cmd.clear();
|
||||||
|
}
|
||||||
|
else if (id.length() or clients.find(id)!=clients.end())
|
||||||
{
|
{
|
||||||
s=getword(cmd); // broker name
|
s=getword(cmd); // broker name
|
||||||
if (s=="" or brokers.find(s) != brokers.end())
|
if (s=="" or brokers.find(s) != brokers.end())
|
||||||
@@ -590,11 +733,14 @@ void eval(std::string& cmd)
|
|||||||
else if (s.length())
|
else if (s.length())
|
||||||
{
|
{
|
||||||
Serial << " not found." << endl;
|
Serial << " not found." << endl;
|
||||||
|
cmd.clear();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
{
|
||||||
Serial << "Missing or existing client name" << endl;
|
Serial << "Missing or existing client name" << endl;
|
||||||
cmd+=" ls";
|
cmd.clear();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else if (compare(s, "set"))
|
else if (compare(s, "set"))
|
||||||
{
|
{
|
||||||
@@ -627,14 +773,14 @@ void eval(std::string& cmd)
|
|||||||
Serial << "--< " << clients.size() << " client/s. >--" << endl;
|
Serial << "--< " << clients.size() << " client/s. >--" << endl;
|
||||||
for(auto it: clients)
|
for(auto it: clients)
|
||||||
{
|
{
|
||||||
Serial << " "; it.second->dump();
|
it.second->dump(" ");
|
||||||
}
|
}
|
||||||
|
|
||||||
Serial << "--< " << brokers.size() << " brokers/s. >--" << endl;
|
Serial << "--< " << brokers.size() << " brokers/s. >--" << endl;
|
||||||
for(auto it: brokers)
|
for(auto it: brokers)
|
||||||
{
|
{
|
||||||
Serial << " ==[ Broker: " << it.first.c_str() << " ]== ";
|
Serial << " +-- '" << it.first.c_str() << "' " << it.second->clientsCount() << " client/s."<< endl;
|
||||||
it.second->dump();
|
it.second->dump(" ");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (compare(s, "reset"))
|
else if (compare(s, "reset"))
|
||||||
@@ -645,7 +791,9 @@ void eval(std::string& cmd)
|
|||||||
{
|
{
|
||||||
Serial << "syntax:" << endl;
|
Serial << "syntax:" << endl;
|
||||||
Serial << " MqttBroker:" << endl;
|
Serial << " MqttBroker:" << endl;
|
||||||
Serial << " broker {name} {port} : create a new broker" << endl;
|
Serial << " broker {broker_name} {port} : create a new broker" << endl;
|
||||||
|
Serial << " broker_name.delete : delete a broker (buggy)" << endl;
|
||||||
|
Serial << " broker_name.view : dump a broker" << endl;
|
||||||
Serial << endl;
|
Serial << endl;
|
||||||
Serial << " MqttClient:" << endl;
|
Serial << " MqttClient:" << endl;
|
||||||
Serial << " client {name} {parent broker} : create a client then" << endl;
|
Serial << " client {name} {parent broker} : create a client then" << endl;
|
||||||
@@ -654,18 +802,21 @@ void eval(std::string& cmd)
|
|||||||
Serial << " name.publish [topic][payload]" << endl;
|
Serial << " name.publish [topic][payload]" << endl;
|
||||||
Serial << " name.view" << endl;
|
Serial << " name.view" << endl;
|
||||||
Serial << " name.delete" << endl;
|
Serial << " name.delete" << endl;
|
||||||
|
Serial << endl;
|
||||||
|
|
||||||
automatic::help();
|
automatic::help();
|
||||||
Serial << endl;
|
Serial << endl;
|
||||||
Serial << " help" << endl;
|
Serial << " help" << endl;
|
||||||
Serial << " blink [Dx on_ms off_ms]" << endl;
|
Serial << " blink [Dx on_ms off_ms] : make pin blink" << endl;
|
||||||
Serial << " ls / ip / reset" << endl;
|
Serial << " ls / ip / reset" << endl;
|
||||||
Serial << " set [name][value]" << endl;
|
Serial << " set [name][value]" << endl;
|
||||||
Serial << " ! repeat last command" << endl;
|
Serial << " ! repeat last command" << endl;
|
||||||
Serial << endl;
|
Serial << endl;
|
||||||
Serial << " every ms [command]; every list; every remove [nr|all]" << endl;
|
Serial << " echo [on|off] or strings" << endl;
|
||||||
|
Serial << " every ms [command]; every list; every remove [nr|all]; every (on|off) [#]" << endl;
|
||||||
Serial << " on {output}; off {output}" << endl;
|
Serial << " on {output}; off {output}" << endl;
|
||||||
Serial << " $id : name of the client." << endl;
|
Serial << " $id : name of the client." << endl;
|
||||||
|
Serial << " rnd[(min[,max])] random number." << endl;
|
||||||
Serial << " default topic is '" << topic.c_str() << "'" << endl;
|
Serial << " default topic is '" << topic.c_str() << "'" << endl;
|
||||||
Serial << endl;
|
Serial << endl;
|
||||||
}
|
}
|
||||||
@@ -678,7 +829,7 @@ void eval(std::string& cmd)
|
|||||||
|
|
||||||
if (retval != MqttOk)
|
if (retval != MqttOk)
|
||||||
{
|
{
|
||||||
Serial << "## ERROR " << retval << endl;
|
Serial << "# MQTT ERROR " << retval << endl;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -688,16 +839,24 @@ 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;
|
||||||
|
|
||||||
for(auto& every: everies)
|
for(auto& every: everies)
|
||||||
{
|
{
|
||||||
|
if (not every.active) continue;
|
||||||
if (every.ms && every.cmd.length() && ms > every.next)
|
if (every.ms && every.cmd.length() && ms > every.next)
|
||||||
{
|
{
|
||||||
std::string cmd(every.cmd);
|
std::string cmd(every.cmd);
|
||||||
eval(cmd);
|
eval(cmd);
|
||||||
every.next += every.ms;
|
every.next += every.ms;
|
||||||
|
if (ms > every.next and ms > every.underrun)
|
||||||
|
{
|
||||||
|
Serial << "Underrun every #" << e << ", " << (ms - every.next) << "ms late" << endl;
|
||||||
|
every.underrun = ms+5000;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
e++;
|
||||||
|
}
|
||||||
|
|
||||||
while(blink_bits)
|
while(blink_bits)
|
||||||
{
|
{
|
||||||
@@ -720,13 +879,10 @@ void loop()
|
|||||||
}
|
}
|
||||||
|
|
||||||
static long count;
|
static long count;
|
||||||
|
#if defined(ESP9266)
|
||||||
MDNS.update();
|
MDNS.update();
|
||||||
|
#endif
|
||||||
|
|
||||||
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();
|
||||||
|
|
||||||
@@ -739,8 +895,10 @@ void loop()
|
|||||||
{
|
{
|
||||||
static std::string cmd;
|
static std::string cmd;
|
||||||
char c=Serial.read();
|
char c=Serial.read();
|
||||||
|
if (echo_on)
|
||||||
|
Serial << c;
|
||||||
|
|
||||||
if (c==10 or c==14)
|
if (c==10 or c==13)
|
||||||
{
|
{
|
||||||
Serial << "----------------[ " << cmd.c_str() << " ]--------------" << endl;
|
Serial << "----------------[ " << cmd.c_str() << " ]--------------" << endl;
|
||||||
static std::string last_cmd;
|
static std::string last_cmd;
|
||||||
|
|||||||
18
library.json
18
library.json
@@ -1,18 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "TinyMqtt",
|
|
||||||
"keywords": "ethernet, mqtt, m2m, iot",
|
|
||||||
"description": "MQTT is a lightweight messaging protocol ideal for small devices. This library allows to send and receive MQTT messages. It does support MQTT 3.1.1 with QOS=0.",
|
|
||||||
"repository": {
|
|
||||||
"type": "git",
|
|
||||||
"url": "https://github.com/hsaturn/TinyMqtt.git"
|
|
||||||
},
|
|
||||||
"version": "0.7.3",
|
|
||||||
"exclude": "",
|
|
||||||
"examples": "examples/*/*.ino",
|
|
||||||
"frameworks": "arduino",
|
|
||||||
"platforms": [
|
|
||||||
"atmelavr",
|
|
||||||
"espressif8266",
|
|
||||||
"espressif32"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -1,11 +1,10 @@
|
|||||||
name=TinyMqtt
|
name=TinyMqtt
|
||||||
version=0.7.3
|
version=0.9.5
|
||||||
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.
|
||||||
paragraph=MQTT is a lightweight messaging protocol ideal for small devices. This library allows to send and receive MQTT messages and to host a broker in your ESP. It does support MQTT 3.1.1 with QoS=0.
|
paragraph=MQTT is a lightweight messaging protocol ideal for small devices. This library allows to send and receive MQTT messages and to host a broker in your ESP 8266 and 32 WROOM. It does support MQTT 3.1.1 with QoS=0.
|
||||||
category=Communication
|
category=Communication
|
||||||
url=https://github.com/hsaturn/TinyMqtt
|
url=https://github.com/hsaturn/TinyMqtt
|
||||||
architectures=*
|
architectures=*
|
||||||
includes=TinyMqtt.h
|
includes=TinyMqtt.h
|
||||||
depends=AsyncTCP
|
|
||||||
|
|||||||
@@ -38,11 +38,12 @@
|
|||||||
4. Simple _FMT mechanism ala printf, but without the typeunsafetyness
|
4. Simple _FMT mechanism ala printf, but without the typeunsafetyness
|
||||||
and no internal buffers for replaceable stream printing
|
and no internal buffers for replaceable stream printing
|
||||||
*/
|
*/
|
||||||
|
#pragma once
|
||||||
|
|
||||||
#ifndef ARDUINO_STREAMING
|
#ifndef ARDUINO_STREAMING
|
||||||
#define ARDUINO_STREAMING
|
#define ARDUINO_STREAMING
|
||||||
|
|
||||||
#if defined(ARDUINO) && ARDUINO >= 100
|
#if (defined(ARDUINO) && ARDUINO >= 100) || defined(EPOXY_DUINO)
|
||||||
#include "Arduino.h"
|
#include "Arduino.h"
|
||||||
#else
|
#else
|
||||||
#ifndef STREAMING_CONSOLE
|
#ifndef STREAMING_CONSOLE
|
||||||
@@ -154,7 +155,7 @@ template<typename T>
|
|||||||
inline Print &operator <<(Print &obj, const _BASED<T> &arg)
|
inline Print &operator <<(Print &obj, const _BASED<T> &arg)
|
||||||
{ obj.print(arg.val, arg.base); return obj; }
|
{ obj.print(arg.val, arg.base); return obj; }
|
||||||
|
|
||||||
#if ARDUINO >= 18
|
#if ARDUINO >= 18 || defined(EPOXY_DUINO)
|
||||||
// Specialization for class _FLOAT
|
// Specialization for class _FLOAT
|
||||||
// Thanks to Michael Margolis for suggesting a way
|
// Thanks to Michael Margolis for suggesting a way
|
||||||
// to accommodate Arduino 0018's floating point precision
|
// to accommodate Arduino 0018's floating point precision
|
||||||
|
|||||||
@@ -28,29 +28,6 @@ class StringIndexer
|
|||||||
public:
|
public:
|
||||||
using index_t = uint8_t;
|
using index_t = uint8_t;
|
||||||
|
|
||||||
static index_t strToIndex(const char* str, uint8_t len)
|
|
||||||
{
|
|
||||||
for(auto it=strings.begin(); it!=strings.end(); it++)
|
|
||||||
{
|
|
||||||
if (strncmp(it->second.str.c_str(), str, len)==0)
|
|
||||||
{
|
|
||||||
it->second.used++;
|
|
||||||
return it->first;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for(index_t index=1; index; index++)
|
|
||||||
{
|
|
||||||
if (strings.find(index)==strings.end())
|
|
||||||
{
|
|
||||||
strings[index].str = std::string(str, len);
|
|
||||||
strings[index].used++;
|
|
||||||
// Serial << "Creating index " << index << " for (" << strings[index].str.c_str() << ") len=" << len << endl;
|
|
||||||
return index;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return 0; // TODO out of indexes
|
|
||||||
}
|
|
||||||
|
|
||||||
static const std::string& str(const index_t& index)
|
static const std::string& str(const index_t& index)
|
||||||
{
|
{
|
||||||
static std::string dummy;
|
static std::string dummy;
|
||||||
@@ -82,6 +59,32 @@ class StringIndexer
|
|||||||
static uint16_t count() { return strings.size(); }
|
static uint16_t count() { return strings.size(); }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
friend class IndexedString;
|
||||||
|
|
||||||
|
// increment use of str or create a new index
|
||||||
|
static index_t strToIndex(const char* str, uint8_t len)
|
||||||
|
{
|
||||||
|
for(auto it=strings.begin(); it!=strings.end(); it++)
|
||||||
|
{
|
||||||
|
if (it->second.str.length() == len && strcmp(it->second.str.c_str(), str)==0)
|
||||||
|
{
|
||||||
|
it->second.used++;
|
||||||
|
return it->first;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for(index_t index=1; index; index++)
|
||||||
|
{
|
||||||
|
if (strings.find(index)==strings.end())
|
||||||
|
{
|
||||||
|
strings[index].str = std::string(str, len);
|
||||||
|
strings[index].used++;
|
||||||
|
// Serial << "Creating index " << index << " for (" << strings[index].str.c_str() << ") len=" << len << endl;
|
||||||
|
return index;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0; // TODO out of indexes
|
||||||
|
}
|
||||||
|
|
||||||
static std::map<index_t, StringCounter> strings;
|
static std::map<index_t, StringCounter> strings;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
220
src/TinyMqtt.cpp
220
src/TinyMqtt.cpp
@@ -1,18 +1,14 @@
|
|||||||
#include "TinyMqtt.h"
|
#include "TinyMqtt.h"
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
|
|
||||||
void outstring(const char* prefix, const char*p, uint16_t len)
|
#ifdef EPOXY_DUINO
|
||||||
{
|
std::map<MqttMessage::Type, int> MqttClient::counters;
|
||||||
return;
|
#endif
|
||||||
Serial << prefix << "='";
|
|
||||||
while(len--) Serial << (char)*p++;
|
|
||||||
Serial << '\'' << endl;
|
|
||||||
}
|
|
||||||
|
|
||||||
MqttBroker::MqttBroker(uint16_t port)
|
MqttBroker::MqttBroker(uint16_t port)
|
||||||
{
|
{
|
||||||
server = new TcpServer(port);
|
server = new TcpServer(port);
|
||||||
#ifdef TCP_ASYNC
|
#ifdef TINY_MQTT_ASYNC
|
||||||
server->onClient(onClient, this);
|
server->onClient(onClient, this);
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
@@ -30,7 +26,7 @@ MqttBroker::~MqttBroker()
|
|||||||
MqttClient::MqttClient(MqttBroker* parent, TcpClient* new_client)
|
MqttClient::MqttClient(MqttBroker* parent, TcpClient* new_client)
|
||||||
: parent(parent)
|
: parent(parent)
|
||||||
{
|
{
|
||||||
#ifdef TCP_ASYNC
|
#ifdef TINY_MQTT_ASYNC
|
||||||
client = new_client;
|
client = new_client;
|
||||||
client->onData(onData, this);
|
client->onData(onData, this);
|
||||||
// client->onConnect() TODO
|
// client->onConnect() TODO
|
||||||
@@ -38,7 +34,11 @@ MqttClient::MqttClient(MqttBroker* parent, TcpClient* new_client)
|
|||||||
#else
|
#else
|
||||||
client = new WiFiClient(*new_client);
|
client = new WiFiClient(*new_client);
|
||||||
#endif
|
#endif
|
||||||
alive = millis()+5000; // client expires after 5s if no CONNECT msg
|
#ifdef EPOXY_DUINO
|
||||||
|
alive = millis()+500000;
|
||||||
|
#else
|
||||||
|
alive = millis()+5000; // TODO MAGIC client expires after 5s if no CONNECT msg
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
MqttClient::MqttClient(MqttBroker* parent, const std::string& id)
|
MqttClient::MqttClient(MqttBroker* parent, const std::string& id)
|
||||||
@@ -59,7 +59,7 @@ 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) // connected to a remote broker
|
||||||
{
|
{
|
||||||
if (bSendDisconnect and client->connected())
|
if (bSendDisconnect and client->connected())
|
||||||
{
|
{
|
||||||
@@ -76,16 +76,22 @@ void MqttClient::close(bool bSendDisconnect)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void MqttClient::connect(MqttBroker* parentBroker)
|
||||||
|
{
|
||||||
|
close();
|
||||||
|
parent = parentBroker;
|
||||||
|
}
|
||||||
|
|
||||||
void MqttClient::connect(std::string broker, uint16_t port, uint16_t ka)
|
void MqttClient::connect(std::string broker, uint16_t port, uint16_t ka)
|
||||||
{
|
{
|
||||||
debug("cnx: closing");
|
debug("MqttClient::connect");
|
||||||
keep_alive = ka;
|
keep_alive = ka;
|
||||||
close();
|
close();
|
||||||
if (client) delete client;
|
if (client) delete client;
|
||||||
client = new TcpClient;
|
client = new TcpClient;
|
||||||
|
|
||||||
debug("Trying to connect to " << broker.c_str() << ':' << port);
|
debug("Trying to connect to " << broker.c_str() << ':' << port);
|
||||||
#ifdef TCP_ASYNC
|
#ifdef TINY_MQTT_ASYNC
|
||||||
client->onData(onData, this);
|
client->onData(onData, this);
|
||||||
client->onConnect(onConnect, this);
|
client->onConnect(onConnect, this);
|
||||||
client->connect(broker.c_str(), port);
|
client->connect(broker.c_str(), port);
|
||||||
@@ -140,7 +146,7 @@ void MqttBroker::onClient(void* broker_ptr, TcpClient* client)
|
|||||||
|
|
||||||
void MqttBroker::loop()
|
void MqttBroker::loop()
|
||||||
{
|
{
|
||||||
#ifndef TCP_ASYNC
|
#ifndef TINY_MQTT_ASYNC
|
||||||
WiFiClient client = server->available();
|
WiFiClient client = server->available();
|
||||||
|
|
||||||
if (client)
|
if (client)
|
||||||
@@ -184,7 +190,7 @@ MqttError MqttBroker::subscribe(const Topic& topic, uint8_t qos)
|
|||||||
return MqttNowhereToSend;
|
return MqttNowhereToSend;
|
||||||
}
|
}
|
||||||
|
|
||||||
MqttError MqttBroker::publish(const MqttClient* source, const Topic& topic, const MqttMessage& msg) const
|
MqttError MqttBroker::publish(const MqttClient* source, const Topic& topic, MqttMessage& msg) const
|
||||||
{
|
{
|
||||||
MqttError retval = MqttOk;
|
MqttError retval = MqttOk;
|
||||||
|
|
||||||
@@ -244,7 +250,11 @@ void MqttClient::clientAlive(uint32_t more_seconds)
|
|||||||
{
|
{
|
||||||
if (keep_alive)
|
if (keep_alive)
|
||||||
{
|
{
|
||||||
|
#ifdef EPOXY_DUINO
|
||||||
|
alive=millis()+500000;
|
||||||
|
#else
|
||||||
alive=millis()+1000*(keep_alive+more_seconds);
|
alive=millis()+1000*(keep_alive+more_seconds);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
alive=0;
|
alive=0;
|
||||||
@@ -271,7 +281,7 @@ void MqttClient::loop()
|
|||||||
// there is no need to send one PingReq per instance.
|
// there is no need to send one PingReq per instance.
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#ifndef TCP_ASYNC
|
#ifndef TINY_MQTT_ASYNC
|
||||||
while(client && client->available()>0)
|
while(client && client->available()>0)
|
||||||
{
|
{
|
||||||
message.incoming(client->read());
|
message.incoming(client->read());
|
||||||
@@ -304,7 +314,7 @@ void MqttClient::onConnect(void *mqttclient_ptr, TcpClient*)
|
|||||||
mqtt->clientAlive(0);
|
mqtt->clientAlive(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef TCP_ASYNC
|
#ifdef TINY_MQTT_ASYNC
|
||||||
void MqttClient::onData(void* client_ptr, TcpClient*, void* data, size_t len)
|
void MqttClient::onData(void* client_ptr, TcpClient*, void* data, size_t len)
|
||||||
{
|
{
|
||||||
char* char_ptr = static_cast<char*>(data);
|
char* char_ptr = static_cast<char*>(data);
|
||||||
@@ -389,16 +399,18 @@ MqttError MqttClient::sendTopic(const Topic& topic, MqttMessage::Type type, uint
|
|||||||
return msg.sendTo(this);
|
return msg.sendTo(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
long MqttClient::counter=0;
|
void MqttClient::processMessage(MqttMessage* mesg)
|
||||||
|
|
||||||
void MqttClient::processMessage(const MqttMessage* mesg)
|
|
||||||
{
|
{
|
||||||
counter++;
|
|
||||||
#ifdef TINY_MQTT_DEBUG
|
#ifdef TINY_MQTT_DEBUG
|
||||||
if (mesg->type() != MqttMessage::Type::PingReq && mesg->type() != MqttMessage::Type::PingResp)
|
if (mesg->type() != MqttMessage::Type::PingReq && mesg->type() != MqttMessage::Type::PingResp)
|
||||||
{
|
{
|
||||||
|
#ifdef NOT_ESP_CORE
|
||||||
|
Serial << "---> INCOMING " << _HEX(mesg->type()) << " client(" << (dbg_ptr)client << ':' << clientId << ") mem=" << " ESP.getFreeHeap() "<< endl;
|
||||||
|
#else
|
||||||
Serial << "---> INCOMING " << _HEX(mesg->type()) << " client(" << (dbg_ptr)client << ':' << clientId << ") mem=" << ESP.getFreeHeap() << endl;
|
Serial << "---> INCOMING " << _HEX(mesg->type()) << " client(" << (dbg_ptr)client << ':' << clientId << ") mem=" << ESP.getFreeHeap() << endl;
|
||||||
|
#endif
|
||||||
// mesg->hexdump("Incoming");
|
// mesg->hexdump("Incoming");
|
||||||
|
mesg->hexdump("Incoming");
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
auto header = mesg->getVHeader();
|
auto header = mesg->getVHeader();
|
||||||
@@ -406,7 +418,11 @@ if (mesg->type() != MqttMessage::Type::PingReq && mesg->type() != MqttMessage::T
|
|||||||
uint16_t len;
|
uint16_t len;
|
||||||
bool bclose=true;
|
bool bclose=true;
|
||||||
|
|
||||||
switch(mesg->type() & 0XF0)
|
#ifdef EPOXY_DUINO
|
||||||
|
counters[mesg->type()]++;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
switch(mesg->type())
|
||||||
{
|
{
|
||||||
case MqttMessage::Type::Connect:
|
case MqttMessage::Type::Connect:
|
||||||
if (mqtt_connected)
|
if (mqtt_connected)
|
||||||
@@ -436,11 +452,9 @@ if (mesg->type() != MqttMessage::Type::PingReq && mesg->type() != MqttMessage::T
|
|||||||
if (mqtt_flags & FlagWill) // Will topic
|
if (mqtt_flags & FlagWill) // Will topic
|
||||||
{
|
{
|
||||||
mesg->getString(payload, len); // Will Topic
|
mesg->getString(payload, len); // Will Topic
|
||||||
outstring("WillTopic", payload, len);
|
|
||||||
payload += len;
|
payload += len;
|
||||||
|
|
||||||
mesg->getString(payload, len); // Will Message
|
mesg->getString(payload, len); // Will Message
|
||||||
outstring("WillMessage", payload, len);
|
|
||||||
payload += len;
|
payload += len;
|
||||||
}
|
}
|
||||||
// FIXME forgetting credential is allowed (security hole)
|
// FIXME forgetting credential is allowed (security hole)
|
||||||
@@ -457,7 +471,9 @@ if (mesg->type() != MqttMessage::Type::PingReq && mesg->type() != MqttMessage::T
|
|||||||
payload += len;
|
payload += len;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef TINY_MQTT_DEBUG
|
||||||
Serial << "Connected client:" << clientId.c_str() << ", keep alive=" << keep_alive << '.' << endl;
|
Serial << "Connected client:" << clientId.c_str() << ", keep alive=" << keep_alive << '.' << endl;
|
||||||
|
#endif
|
||||||
bclose = false;
|
bclose = false;
|
||||||
mqtt_connected=true;
|
mqtt_connected=true;
|
||||||
{
|
{
|
||||||
@@ -507,18 +523,25 @@ if (mesg->type() != MqttMessage::Type::PingReq && mesg->type() != MqttMessage::T
|
|||||||
payload = header+2;
|
payload = header+2;
|
||||||
|
|
||||||
debug("un/subscribe loop");
|
debug("un/subscribe loop");
|
||||||
|
std::string qoss;
|
||||||
while(payload < mesg->end())
|
while(payload < mesg->end())
|
||||||
{
|
{
|
||||||
mesg->getString(payload, len); // Topic
|
mesg->getString(payload, len); // Topic
|
||||||
debug( " topic (" << std::string(payload, len) << ')');
|
debug( " topic (" << std::string(payload, len) << ')');
|
||||||
outstring(" un/subscribes", payload, len);
|
|
||||||
// subscribe(Topic(payload, len));
|
// subscribe(Topic(payload, len));
|
||||||
Topic topic(payload, len);
|
Topic topic(payload, len);
|
||||||
|
|
||||||
payload += len;
|
payload += len;
|
||||||
if ((mesg->type() & 0XF0) == MqttMessage::Type::Subscribe)
|
if (mesg->type() == MqttMessage::Type::Subscribe)
|
||||||
{
|
{
|
||||||
uint8_t qos = *payload++;
|
uint8_t qos = *payload++;
|
||||||
if (qos != 0) debug("Unsupported QOS" << qos << endl);
|
if (qos != 0)
|
||||||
|
{
|
||||||
|
debug("Unsupported QOS" << qos << endl);
|
||||||
|
qoss.push_back(0x80);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
qoss.push_back(qos);
|
||||||
subscriptions.insert(topic);
|
subscriptions.insert(topic);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@@ -530,7 +553,12 @@ if (mesg->type() != MqttMessage::Type::PingReq && mesg->type() != MqttMessage::T
|
|||||||
}
|
}
|
||||||
debug("end loop");
|
debug("end loop");
|
||||||
bclose = false;
|
bclose = false;
|
||||||
// TODO SUBACK
|
|
||||||
|
MqttMessage ack(mesg->type() == MqttMessage::Type::Subscribe ? MqttMessage::Type::SubAck : MqttMessage::Type::UnSuback);
|
||||||
|
ack.add(header[0]);
|
||||||
|
ack.add(header[1]);
|
||||||
|
ack.add(qoss.c_str(), qoss.size(), false);
|
||||||
|
ack.sendTo(this);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@@ -540,9 +568,12 @@ if (mesg->type() != MqttMessage::Type::PingReq && mesg->type() != MqttMessage::T
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case MqttMessage::Type::Publish:
|
case MqttMessage::Type::Publish:
|
||||||
|
#ifdef TINY_MQTT_DEBUG
|
||||||
|
Serial << "publish " << mqtt_connected << '/' << (long) client << endl;
|
||||||
|
#endif
|
||||||
if (mqtt_connected or client == nullptr)
|
if (mqtt_connected or client == nullptr)
|
||||||
{
|
{
|
||||||
uint8_t qos = mesg->type() & 0x6;
|
uint8_t qos = mesg->flags();
|
||||||
payload = header;
|
payload = header;
|
||||||
mesg->getString(payload, len);
|
mesg->getString(payload, len);
|
||||||
Topic published(payload, len);
|
Topic published(payload, len);
|
||||||
@@ -554,8 +585,12 @@ if (mesg->type() != MqttMessage::Type::PingReq && mesg->type() != MqttMessage::T
|
|||||||
// TODO reset DUP
|
// TODO reset DUP
|
||||||
// TODO reset RETAIN
|
// TODO reset RETAIN
|
||||||
|
|
||||||
if (client==nullptr) // internal MqttClient receives publish
|
if (parent==nullptr or client==nullptr) // internal MqttClient receives publish
|
||||||
{
|
{
|
||||||
|
#ifdef TINY_MQTT_DEBUG
|
||||||
|
Serial << (isSubscribedTo(published) ? "not" : "") << " subscribed.\n";
|
||||||
|
Serial << "has " << (callback ? "" : "no ") << " callback.\n";
|
||||||
|
#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); // TODO send the real payload
|
||||||
@@ -584,10 +619,12 @@ if (mesg->type() != MqttMessage::Type::PingReq && mesg->type() != MqttMessage::T
|
|||||||
};
|
};
|
||||||
if (bclose)
|
if (bclose)
|
||||||
{
|
{
|
||||||
|
#ifdef TINY_MQTT_DEBUG
|
||||||
Serial << "*************** Error msg 0x" << _HEX(mesg->type());
|
Serial << "*************** Error msg 0x" << _HEX(mesg->type());
|
||||||
mesg->hexdump("-------ERROR ------");
|
mesg->hexdump("-------ERROR ------");
|
||||||
dump();
|
dump();
|
||||||
Serial << endl;
|
Serial << endl;
|
||||||
|
#endif
|
||||||
close();
|
close();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@@ -599,9 +636,65 @@ if (mesg->type() != MqttMessage::Type::PingReq && mesg->type() != MqttMessage::T
|
|||||||
bool Topic::matches(const Topic& topic) const
|
bool Topic::matches(const Topic& topic) const
|
||||||
{
|
{
|
||||||
if (getIndex() == topic.getIndex()) return true;
|
if (getIndex() == topic.getIndex()) return true;
|
||||||
if (str() == topic.str()) return true;
|
const char* p1 = c_str();
|
||||||
|
const char* p2 = topic.c_str();
|
||||||
|
|
||||||
|
if (p1 == p2) return true;
|
||||||
|
if (*p2 == '$' and *p1 != '$') return false;
|
||||||
|
|
||||||
|
while(*p1 and *p2)
|
||||||
|
{
|
||||||
|
if (*p1 == '+')
|
||||||
|
{
|
||||||
|
++p1;
|
||||||
|
if (*p1 and *p1!='/') return false;
|
||||||
|
if (*p1) ++p1;
|
||||||
|
while(*p2 and *p2++!='/');
|
||||||
|
}
|
||||||
|
else if (*p1 == '#')
|
||||||
|
{
|
||||||
|
if (*++p1==0) return true;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
else if (*p1 == '*')
|
||||||
|
{
|
||||||
|
const char c=*(p1+1);
|
||||||
|
if (c==0) return true;
|
||||||
|
if (c!='/') return false;
|
||||||
|
const char*p = p1+2;
|
||||||
|
while(*p and *p2)
|
||||||
|
{
|
||||||
|
if (*p == *p2)
|
||||||
|
{
|
||||||
|
if (*p==0) return true;
|
||||||
|
if (*p=='/')
|
||||||
|
{
|
||||||
|
p1=p;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
while(*p2 and *p2++!='/');
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
++p;
|
||||||
|
++p2;
|
||||||
|
}
|
||||||
|
if (*p==0) return true;
|
||||||
|
}
|
||||||
|
else if (*p1 == *p2)
|
||||||
|
{
|
||||||
|
++p1;
|
||||||
|
++p2;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (*p1=='/' and p1[1]=='#' and p1[2]==0) return true;
|
||||||
|
return *p1==0 and *p2==0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// 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)
|
||||||
@@ -610,6 +703,7 @@ MqttError MqttClient::publish(const Topic& topic, const char* payload, size_t pa
|
|||||||
msg.add(topic);
|
msg.add(topic);
|
||||||
msg.add(payload, pay_length, false);
|
msg.add(payload, pay_length, false);
|
||||||
msg.complete();
|
msg.complete();
|
||||||
|
|
||||||
if (parent)
|
if (parent)
|
||||||
{
|
{
|
||||||
return parent->publish(this, topic, msg);
|
return parent->publish(this, topic, msg);
|
||||||
@@ -621,7 +715,7 @@ MqttError MqttClient::publish(const Topic& topic, const char* payload, size_t pa
|
|||||||
}
|
}
|
||||||
|
|
||||||
// republish a received publish if it matches any in subscriptions
|
// republish a received publish if it matches any in subscriptions
|
||||||
MqttError MqttClient::publishIfSubscribed(const Topic& topic, const MqttMessage& msg)
|
MqttError MqttClient::publishIfSubscribed(const Topic& topic, MqttMessage& msg)
|
||||||
{
|
{
|
||||||
MqttError retval=MqttOk;
|
MqttError retval=MqttOk;
|
||||||
|
|
||||||
@@ -633,6 +727,10 @@ MqttError MqttClient::publishIfSubscribed(const Topic& topic, const MqttMessage&
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
processMessage(&msg);
|
processMessage(&msg);
|
||||||
|
|
||||||
|
#ifdef TINY_MQTT_DEBUG
|
||||||
|
Serial << "Should call the callback ?\n";
|
||||||
|
#endif
|
||||||
// callback(this, topic, nullptr, 0); // TODO Payload
|
// callback(this, topic, nullptr, 0); // TODO Payload
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -661,24 +759,23 @@ void MqttMessage::incoming(char in_byte)
|
|||||||
switch(state)
|
switch(state)
|
||||||
{
|
{
|
||||||
case FixedHeader:
|
case FixedHeader:
|
||||||
size=0;
|
size=MaxBufferLength;
|
||||||
state = Length;
|
state = Length;
|
||||||
break;
|
break;
|
||||||
case Length:
|
case Length:
|
||||||
size = (size<<7) + (in_byte & 0x3F);
|
|
||||||
|
if (size==MaxBufferLength)
|
||||||
|
size = in_byte & 0x7F;
|
||||||
|
else
|
||||||
|
size += static_cast<uint16_t>(in_byte & 0x7F)<<7;
|
||||||
|
|
||||||
if (size > MaxBufferLength)
|
if (size > MaxBufferLength)
|
||||||
{
|
|
||||||
state = Error;
|
state = Error;
|
||||||
}
|
|
||||||
else if ((in_byte & 0x80) == 0)
|
else if ((in_byte & 0x80) == 0)
|
||||||
{
|
{
|
||||||
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);
|
buffer.reserve(size);
|
||||||
@@ -700,12 +797,14 @@ void MqttMessage::incoming(char in_byte)
|
|||||||
break;
|
break;
|
||||||
case Complete:
|
case Complete:
|
||||||
default:
|
default:
|
||||||
|
#ifdef TINY_MQTT_DEBUG
|
||||||
Serial << "Spurious " << _HEX(in_byte) << endl;
|
Serial << "Spurious " << _HEX(in_byte) << endl;
|
||||||
hexdump("spurious");
|
hexdump("spurious");
|
||||||
|
#endif
|
||||||
reset();
|
reset();
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
if (buffer.length() > MaxBufferLength) // TODO magic 256 ?
|
if (buffer.length() > MaxBufferLength)
|
||||||
{
|
{
|
||||||
debug("Too long " << state);
|
debug("Too long " << state);
|
||||||
reset();
|
reset();
|
||||||
@@ -716,36 +815,42 @@ void MqttMessage::add(const char* p, size_t len, bool addLength)
|
|||||||
{
|
{
|
||||||
if (addLength)
|
if (addLength)
|
||||||
{
|
{
|
||||||
buffer.reserve(buffer.length()+addLength+2);
|
buffer.reserve(buffer.length()+2);
|
||||||
incoming(len>>8);
|
incoming(len>>8);
|
||||||
incoming(len & 0xFF);
|
incoming(len & 0xFF);
|
||||||
}
|
}
|
||||||
while(len--) incoming(*p++);
|
while(len--) incoming(*p++);
|
||||||
}
|
}
|
||||||
|
|
||||||
void MqttMessage::encodeLength(char* msb, int length) const
|
void MqttMessage::encodeLength()
|
||||||
{
|
{
|
||||||
do
|
if (state != Complete)
|
||||||
{
|
{
|
||||||
uint8_t encoded(length & 0x7F);
|
int length = buffer.size()-3; // 3 = 1 byte for header + 2 bytes for pre-reserved length field.
|
||||||
length >>=7;
|
if (length <= 0x7F)
|
||||||
if (length) encoded |= 0x80;
|
|
||||||
*msb++ = encoded;
|
|
||||||
} while (length);
|
|
||||||
};
|
|
||||||
|
|
||||||
void MqttMessage::complete()
|
|
||||||
{
|
{
|
||||||
encodeLength(&buffer[1], buffer.size()-2);
|
buffer.erase(1,1);
|
||||||
state = Complete;
|
buffer[1] = length;
|
||||||
|
vheader = 2;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
buffer[1] = 0x80 | (length & 0x7F);
|
||||||
|
buffer[2] = (length >> 7);
|
||||||
|
vheader = 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
MqttError MqttMessage::sendTo(MqttClient* client) const
|
// We could check that buffer[2] < 128 (end of length encoding)
|
||||||
|
state = Complete;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
MqttError MqttMessage::sendTo(MqttClient* client)
|
||||||
{
|
{
|
||||||
if (buffer.size())
|
if (buffer.size())
|
||||||
{
|
{
|
||||||
debug("sending " << buffer.size() << " bytes");
|
debug("sending " << buffer.size() << " bytes");
|
||||||
encodeLength(&buffer[1], buffer.size()-2);
|
encodeLength();
|
||||||
// hexdump("snd");
|
// hexdump("snd");
|
||||||
client->write(&buffer[0], buffer.size());
|
client->write(&buffer[0], buffer.size());
|
||||||
}
|
}
|
||||||
@@ -759,6 +864,8 @@ MqttError MqttMessage::sendTo(MqttClient* client) const
|
|||||||
|
|
||||||
void MqttMessage::hexdump(const char* prefix) const
|
void MqttMessage::hexdump(const char* prefix) const
|
||||||
{
|
{
|
||||||
|
(void)prefix;
|
||||||
|
#ifdef TINY_MQTT_DEBUG
|
||||||
uint16_t addr=0;
|
uint16_t addr=0;
|
||||||
const int bytes_per_row = 8;
|
const int bytes_per_row = 8;
|
||||||
const char* hex_to_str = " | ";
|
const char* hex_to_str = " | ";
|
||||||
@@ -794,4 +901,5 @@ void MqttMessage::hexdump(const char* prefix) const
|
|||||||
}
|
}
|
||||||
|
|
||||||
Serial << endl;
|
Serial << endl;
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|||||||
115
src/TinyMqtt.h
115
src/TinyMqtt.h
@@ -1,28 +1,33 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
// TODO Should add a AUnit with both TCP_ASYNC and not TCP_ASYNC
|
// TODO Should add a AUnit with both TINY_MQTT_ASYNC and not TINY_MQTT_ASYNC
|
||||||
// #define TCP_ASYNC // Uncomment this to use ESPAsyncTCP instead of normal cnx
|
// #define TINY_MQTT_ASYNC // Uncomment this to use ESPAsyncTCP instead of normal cnx
|
||||||
|
|
||||||
#if defined(ESP8266) || defined(EPOXY_DUINO)
|
#if defined(ESP8266) || defined(EPOXY_DUINO)
|
||||||
#ifdef TCP_ASYNC
|
#ifdef TINY_MQTT_ASYNC
|
||||||
#include <ESPAsyncTCP.h>
|
#include <ESPAsyncTCP.h>
|
||||||
#else
|
#else
|
||||||
#include <ESP8266WiFi.h>
|
#include <ESP8266WiFi.h>
|
||||||
#endif
|
#endif
|
||||||
#elif defined(ESP32)
|
#elif defined(ESP32)
|
||||||
#ifdef TCP_ASYNC
|
|
||||||
#include <AsyncTCP.h> // https://github.com/me-no-dev/AsyncTCP
|
|
||||||
#else
|
|
||||||
#include <WiFi.h>
|
#include <WiFi.h>
|
||||||
|
#ifdef TINY_MQTT_ASYNC
|
||||||
|
#include <AsyncTCP.h> // https://github.com/me-no-dev/AsyncTCP
|
||||||
#endif
|
#endif
|
||||||
#else
|
|
||||||
#error "Unsupported platform"
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef EPOXY_DUINO
|
#ifdef EPOXY_DUINO
|
||||||
#define dbg_ptr uint64_t
|
#define dbg_ptr uint64_t
|
||||||
#else
|
#else
|
||||||
#define dbg_ptr uint32_t
|
#define dbg_ptr uint32_t
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef WIO_TERMINAL
|
||||||
|
// Uncommon board handling
|
||||||
|
// If you have a problem with this line, just remove it.
|
||||||
|
// Note: https://github.com/hsaturn/TinyMqtt/issues/41
|
||||||
|
#include <rpcWiFi.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <set>
|
#include <set>
|
||||||
#include <string>
|
#include <string>
|
||||||
@@ -32,12 +37,12 @@
|
|||||||
// #define TINY_MQTT_DEBUG
|
// #define TINY_MQTT_DEBUG
|
||||||
|
|
||||||
#ifdef TINY_MQTT_DEBUG
|
#ifdef TINY_MQTT_DEBUG
|
||||||
#define debug(what) { Serial << __LINE__ << ' ' << what << endl; delay(100); }
|
#define debug(what) { Serial << (int)__LINE__ << ' ' << what << endl; delay(100); }
|
||||||
#else
|
#else
|
||||||
#define debug(what) {}
|
#define debug(what) {}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef TCP_ASYNC
|
#ifdef TINY_MQTT_ASYNC
|
||||||
using TcpClient = AsyncClient;
|
using TcpClient = AsyncClient;
|
||||||
using TcpServer = AsyncServer;
|
using TcpServer = AsyncServer;
|
||||||
#else
|
#else
|
||||||
@@ -45,7 +50,7 @@
|
|||||||
using TcpServer = WiFiServer;
|
using TcpServer = WiFiServer;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
enum MqttError
|
enum __attribute__((packed)) MqttError
|
||||||
{
|
{
|
||||||
MqttOk = 0,
|
MqttOk = 0,
|
||||||
MqttNowhereToSend=1,
|
MqttNowhereToSend=1,
|
||||||
@@ -67,9 +72,9 @@ class Topic : public IndexedString
|
|||||||
class MqttClient;
|
class MqttClient;
|
||||||
class MqttMessage
|
class MqttMessage
|
||||||
{
|
{
|
||||||
const uint16_t MaxBufferLength = 255;
|
const uint16_t MaxBufferLength = 4096; //hard limit: 16k due to size decoding
|
||||||
public:
|
public:
|
||||||
enum Type
|
enum __attribute__((packed)) Type
|
||||||
{
|
{
|
||||||
Unknown = 0,
|
Unknown = 0,
|
||||||
Connect = 0x10,
|
Connect = 0x10,
|
||||||
@@ -84,7 +89,7 @@ class MqttMessage
|
|||||||
PingResp = 0xD0,
|
PingResp = 0xD0,
|
||||||
Disconnect = 0xE0
|
Disconnect = 0xE0
|
||||||
};
|
};
|
||||||
enum State
|
enum __attribute__((packed)) State
|
||||||
{
|
{
|
||||||
FixedHeader=0,
|
FixedHeader=0,
|
||||||
Length=1,
|
Length=1,
|
||||||
@@ -104,8 +109,7 @@ class MqttMessage
|
|||||||
void add(const Topic& t) { add(t.str()); }
|
void add(const Topic& t) { add(t.str()); }
|
||||||
const char* end() const { return &buffer[0]+buffer.size(); }
|
const char* end() const { return &buffer[0]+buffer.size(); }
|
||||||
const char* getVHeader() const { return &buffer[vheader]; }
|
const char* getVHeader() const { return &buffer[vheader]; }
|
||||||
uint16_t length() const { return buffer.size(); }
|
void complete() { encodeLength(); }
|
||||||
void complete();
|
|
||||||
|
|
||||||
void reset();
|
void reset();
|
||||||
|
|
||||||
@@ -115,24 +119,27 @@ class MqttMessage
|
|||||||
|
|
||||||
Type type() const
|
Type type() const
|
||||||
{
|
{
|
||||||
return state == Complete ? static_cast<Type>(buffer[0]) : Unknown;
|
return state == Complete ? static_cast<Type>(buffer[0] & 0xF0) : Unknown;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
uint8_t flags() const { return static_cast<uint8_t>(buffer[0] & 0x0F); }
|
||||||
|
|
||||||
void create(Type type)
|
void create(Type type)
|
||||||
{
|
{
|
||||||
buffer=(char)type;
|
buffer=(decltype(buffer)::value_type)type;
|
||||||
buffer+='\0'; // reserved for msg length
|
buffer+='\0'; // reserved for msg length byte 1/2
|
||||||
vheader=2;
|
buffer+='\0'; // reserved for msg length byte 2/2 (fixed)
|
||||||
|
vheader=3; // Should never change
|
||||||
size=0;
|
size=0;
|
||||||
state=Create;
|
state=Create;
|
||||||
}
|
}
|
||||||
MqttError sendTo(MqttClient*) const;
|
MqttError sendTo(MqttClient*);
|
||||||
void hexdump(const char* prefix=nullptr) const;
|
void hexdump(const char* prefix=nullptr) const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void encodeLength(char* msb, int length) const;
|
void encodeLength();
|
||||||
|
|
||||||
mutable std::string buffer; // mutable -> sendTo()
|
std::string buffer;
|
||||||
uint8_t vheader;
|
uint8_t vheader;
|
||||||
uint16_t size; // bytes left to receive
|
uint16_t size; // bytes left to receive
|
||||||
State state;
|
State state;
|
||||||
@@ -142,7 +149,7 @@ class MqttBroker;
|
|||||||
class MqttClient
|
class MqttClient
|
||||||
{
|
{
|
||||||
using CallBack = void (*)(const MqttClient* source, const Topic& topic, const char* payload, size_t payload_length);
|
using CallBack = void (*)(const MqttClient* source, const Topic& topic, const char* payload, size_t payload_length);
|
||||||
enum Flags
|
enum __attribute__((packed)) Flags
|
||||||
{
|
{
|
||||||
FlagUserName = 128,
|
FlagUserName = 128,
|
||||||
FlagPassword = 64,
|
FlagPassword = 64,
|
||||||
@@ -163,6 +170,8 @@ class MqttClient
|
|||||||
void connect(MqttBroker* parent);
|
void connect(MqttBroker* parent);
|
||||||
void connect(std::string broker, uint16_t port, uint16_t keep_alive = 10);
|
void connect(std::string broker, uint16_t port, uint16_t keep_alive = 10);
|
||||||
|
|
||||||
|
// TODO it seems that connected returns true in tcp mode even if
|
||||||
|
// no negociation occured (only if tcp link is established)
|
||||||
bool connected() { return
|
bool connected() { return
|
||||||
(parent!=nullptr and client==nullptr) or
|
(parent!=nullptr and client==nullptr) or
|
||||||
(client and client->connected()); }
|
(client and client->connected()); }
|
||||||
@@ -175,10 +184,18 @@ class MqttClient
|
|||||||
/** Should be called in main loop() */
|
/** Should be called in main loop() */
|
||||||
void loop();
|
void loop();
|
||||||
void close(bool bSendDisconnect=true);
|
void close(bool bSendDisconnect=true);
|
||||||
void setCallback(CallBack fun) {callback=fun; };
|
void setCallback(CallBack fun)
|
||||||
|
{
|
||||||
|
callback=fun;
|
||||||
|
#ifdef TINY_MQTT_DEBUG
|
||||||
|
Serial << "Callback set to " << (long)fun << endl;
|
||||||
|
if (callback) callback(this, "test/topic", "value", 5);
|
||||||
|
#endif
|
||||||
|
};
|
||||||
|
|
||||||
// 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);
|
||||||
|
MqttError publish(const Topic& t, const char* payload) { return publish(t, payload, strlen(payload)); }
|
||||||
MqttError publish(const Topic& t, const String& s) { return publish(t, s.c_str(), s.length()); }
|
MqttError publish(const Topic& t, const String& s) { return publish(t, s.c_str(), s.length()); }
|
||||||
MqttError publish(const Topic& t, const std::string& s) { return publish(t,s.c_str(),s.length());}
|
MqttError publish(const Topic& t, const std::string& s) { return publish(t,s.c_str(),s.length());}
|
||||||
MqttError publish(const Topic& t) { return publish(t, nullptr, 0);};
|
MqttError publish(const Topic& t) { return publish(t, nullptr, 0);};
|
||||||
@@ -191,31 +208,39 @@ class MqttClient
|
|||||||
// TODO seems to be useless
|
// TODO seems to be useless
|
||||||
bool isLocal() const { return client == nullptr; }
|
bool isLocal() const { return client == nullptr; }
|
||||||
|
|
||||||
void dump()
|
void dump(std::string indent="")
|
||||||
{
|
{
|
||||||
|
(void)indent;
|
||||||
|
#ifdef TINY_MQTT_DEBUG
|
||||||
uint32_t ms=millis();
|
uint32_t ms=millis();
|
||||||
Serial << "MqttClient (" << clientId.c_str() << ") " << (connected() ? " ON " : " OFF");
|
Serial << indent << "+-- " << '\'' << clientId.c_str() << "' " << (connected() ? " ON " : " OFF");
|
||||||
Serial << ", alive=" << alive << '/' << ms << ", ka=" << keep_alive;
|
Serial << ", alive=" << alive << '/' << ms << ", ka=" << keep_alive << ' ';
|
||||||
Serial << (client && client->connected() ? "" : "dis") << "connected";
|
Serial << (client && client->connected() ? "" : "dis") << "connected";
|
||||||
message.hexdump("entrant msg");
|
if (subscriptions.size())
|
||||||
|
{
|
||||||
bool c = false;
|
bool c = false;
|
||||||
Serial << " [";
|
Serial << " [";
|
||||||
for(auto s: subscriptions)
|
for(auto s: subscriptions)
|
||||||
{
|
{
|
||||||
Serial << (c?", ": "")<< s.str().c_str();
|
if (c) Serial << ", ";
|
||||||
|
Serial << s.str().c_str();
|
||||||
c=true;
|
c=true;
|
||||||
}
|
}
|
||||||
|
Serial << ']';
|
||||||
|
}
|
||||||
Serial << "]" << endl;
|
Serial << endl;
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Count the number of messages that have been sent **/
|
#ifdef EPOXY_DUINO
|
||||||
static long counter;
|
static std::map<MqttMessage::Type, int> counters; // Number of processed messages
|
||||||
|
#endif
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
|
// event when tcp/ip link established (real or fake)
|
||||||
static void onConnect(void * client_ptr, TcpClient*);
|
static void onConnect(void * client_ptr, TcpClient*);
|
||||||
#ifdef TCP_ASYNC
|
#ifdef TINY_MQTT_ASYNC
|
||||||
static void onData(void* client_ptr, TcpClient*, void* data, size_t len);
|
static void onData(void* client_ptr, TcpClient*, void* data, size_t len);
|
||||||
#endif
|
#endif
|
||||||
MqttError sendTopic(const Topic& topic, MqttMessage::Type type, uint8_t qos);
|
MqttError sendTopic(const Topic& topic, MqttMessage::Type type, uint8_t qos);
|
||||||
@@ -224,10 +249,10 @@ class MqttClient
|
|||||||
friend class MqttBroker;
|
friend class MqttBroker;
|
||||||
MqttClient(MqttBroker* parent, TcpClient* client);
|
MqttClient(MqttBroker* parent, TcpClient* client);
|
||||||
// republish a received publish if topic matches any in subscriptions
|
// republish a received publish if topic matches any in subscriptions
|
||||||
MqttError publishIfSubscribed(const Topic& topic, const MqttMessage& msg);
|
MqttError publishIfSubscribed(const Topic& topic, MqttMessage& msg);
|
||||||
|
|
||||||
void clientAlive(uint32_t more_seconds);
|
void clientAlive(uint32_t more_seconds);
|
||||||
void processMessage(const MqttMessage* message);
|
void processMessage(MqttMessage* message);
|
||||||
|
|
||||||
bool mqtt_connected = false;
|
bool mqtt_connected = false;
|
||||||
char mqtt_flags;
|
char mqtt_flags;
|
||||||
@@ -240,7 +265,7 @@ class MqttClient
|
|||||||
// (this is the case when MqttBroker isn't used except here)
|
// (this is the case when MqttBroker isn't used except here)
|
||||||
MqttBroker* parent=nullptr; // connection to local broker
|
MqttBroker* parent=nullptr; // connection to local broker
|
||||||
|
|
||||||
TcpClient* client=nullptr; // connection to mqtt client or to remote broker
|
TcpClient* client=nullptr; // connection to remote broker
|
||||||
std::set<Topic> subscriptions;
|
std::set<Topic> subscriptions;
|
||||||
std::string clientId;
|
std::string clientId;
|
||||||
CallBack callback = nullptr;
|
CallBack callback = nullptr;
|
||||||
@@ -248,7 +273,7 @@ class MqttClient
|
|||||||
|
|
||||||
class MqttBroker
|
class MqttBroker
|
||||||
{
|
{
|
||||||
enum State
|
enum __attribute__((packed)) State
|
||||||
{
|
{
|
||||||
Disconnected, // Also the initial state
|
Disconnected, // Also the initial state
|
||||||
Connecting, // connect and sends a fake publish to avoid circular cnx
|
Connecting, // connect and sends a fake publish to avoid circular cnx
|
||||||
@@ -267,14 +292,10 @@ class MqttBroker
|
|||||||
|
|
||||||
size_t clientsCount() const { return clients.size(); }
|
size_t clientsCount() const { return clients.size(); }
|
||||||
|
|
||||||
void dump()
|
void dump(std::string indent="")
|
||||||
{
|
{
|
||||||
Serial << clients.size() << " client/s" << endl;
|
|
||||||
for(auto client: clients)
|
for(auto client: clients)
|
||||||
{
|
client->dump(indent);
|
||||||
Serial << " ";
|
|
||||||
client->dump();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
@@ -288,7 +309,7 @@ class MqttBroker
|
|||||||
{ return compareString(auth_password, password, len); }
|
{ return compareString(auth_password, password, len); }
|
||||||
|
|
||||||
|
|
||||||
MqttError publish(const MqttClient* source, const Topic& topic, const MqttMessage& msg) const;
|
MqttError publish(const MqttClient* source, const Topic& topic, MqttMessage& msg) const;
|
||||||
|
|
||||||
MqttError subscribe(const Topic& topic, uint8_t qos);
|
MqttError subscribe(const Topic& topic, uint8_t qos);
|
||||||
|
|
||||||
|
|||||||
@@ -5,7 +5,15 @@ tests:
|
|||||||
$(MAKE) -C $$(dirname $$i) -j; \
|
$(MAKE) -C $$(dirname $$i) -j; \
|
||||||
done
|
done
|
||||||
|
|
||||||
runtests: tests
|
debugtest:
|
||||||
|
set -e; \
|
||||||
|
$(MAKE) clean; \
|
||||||
|
$(MAKE) -C debug-mode -j; \
|
||||||
|
debug-mode/debug-tests.out
|
||||||
|
|
||||||
|
runtests: debugtest
|
||||||
|
$(MAKE) clean
|
||||||
|
$(MAKE) tests
|
||||||
set -e; \
|
set -e; \
|
||||||
for i in *-tests/Makefile; do \
|
for i in *-tests/Makefile; do \
|
||||||
echo '==== Running:' $$(dirname $$i); \
|
echo '==== Running:' $$(dirname $$i); \
|
||||||
|
|||||||
13
tests/debug-mode/Makefile
Normal file
13
tests/debug-mode/Makefile
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
||||||
|
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
||||||
|
|
||||||
|
EXTRA_CXXFLAGS=-g3 -O0 -DTINY_MQTT_DEBUG
|
||||||
|
|
||||||
|
# Remove flto flag from EpoxyDuino (too many <optimized out>)
|
||||||
|
CXXFLAGS = -Wextra -Wall -std=gnu++11 -fno-exceptions -fno-threadsafe-statics
|
||||||
|
|
||||||
|
APP_NAME := debug-tests
|
||||||
|
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsyncTCP
|
||||||
|
ARDUINO_LIB_DIRS := ../../../EspMock/libraries
|
||||||
|
EPOXY_CORE := EPOXY_CORE_ESP8266
|
||||||
|
include ../../../EpoxyDuino/EpoxyDuino.mk
|
||||||
15
tests/debug-mode/debug-tests.ino
Normal file
15
tests/debug-mode/debug-tests.ino
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
#include <Arduino.h>
|
||||||
|
#include <AUnit.h>
|
||||||
|
#include <TinyMqtt.h>
|
||||||
|
#include <map>
|
||||||
|
#include <iostream>
|
||||||
|
#include <iomanip>
|
||||||
|
#include <sstream>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
// Only compilation check, so do nothing
|
||||||
|
|
||||||
|
void setup() {}
|
||||||
|
void loop() {
|
||||||
|
aunit::TestRunner::run();
|
||||||
|
}
|
||||||
7
tests/howto
Normal file
7
tests/howto
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
cd TinyMqtt/tests/../..
|
||||||
|
git clone https://github.com/hsaturn/EspMock.git
|
||||||
|
git clone https://github.com/bxparks/AUnit.git
|
||||||
|
git clone https://github.com/bxparks/EpoxyDuino.git
|
||||||
|
cd TinyMqtt/tests
|
||||||
|
make
|
||||||
|
make runtests
|
||||||
10
tests/length-tests.todo/Makefile
Normal file
10
tests/length-tests.todo/Makefile
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
||||||
|
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
||||||
|
|
||||||
|
EXTRA_CXXFLAGS=-g3 -O0
|
||||||
|
|
||||||
|
APP_NAME := length-tests
|
||||||
|
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsyncTCP
|
||||||
|
ARDUINO_LIB_DIRS := ../../../EspMock/libraries
|
||||||
|
EPOXY_CORE := EPOXY_CORE_ESP8266
|
||||||
|
include ../../../EpoxyDuino/EpoxyDuino.mk
|
||||||
53
tests/length-tests.todo/length-tests.ino
Normal file
53
tests/length-tests.todo/length-tests.ino
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
#include <AUnit.h>
|
||||||
|
#include <TinyMqtt.h>
|
||||||
|
#include <map>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* TinyMqtt local unit tests.
|
||||||
|
*
|
||||||
|
* Clients are connected to pseudo remote broker
|
||||||
|
* The remote should be 127.0.0.1:1883 <--- But this does not work due to Esp network limitations
|
||||||
|
* 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
|
||||||
|
|
||||||
|
const char* lastPayload;
|
||||||
|
size_t lastLength;
|
||||||
|
|
||||||
|
void onPublish(const MqttClient* srce, const Topic& topic, const char* payload, size_t length)
|
||||||
|
{
|
||||||
|
if (srce)
|
||||||
|
published[srce->id()][topic]++;
|
||||||
|
lastPayload = payload;
|
||||||
|
lastLength = length;
|
||||||
|
}
|
||||||
|
|
||||||
|
test(length_decode_greater_than_127)
|
||||||
|
{
|
||||||
|
// TODO WRITE THIS TEST
|
||||||
|
// The test should verify than a mqtt message with more than 127 bytes is correctly decoded
|
||||||
|
assertEqual(1,2);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
//----------------------------------------------------------------------------
|
||||||
|
// 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();
|
||||||
|
}
|
||||||
@@ -1,7 +1,13 @@
|
|||||||
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
||||||
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
||||||
|
|
||||||
|
EXTRA_CXXFLAGS=-g3 -O0
|
||||||
|
|
||||||
|
# Remove flto flag from EpoxyDuino (too many <optimized out>)
|
||||||
|
CXXFLAGS = -Wextra -Wall -std=gnu++11 -fno-exceptions -fno-threadsafe-statics
|
||||||
|
|
||||||
APP_NAME := local-tests
|
APP_NAME := local-tests
|
||||||
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock
|
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsyncTCP
|
||||||
ESP_LIBS = ESP8266WiFi ESPAsyncTCP
|
ARDUINO_LIB_DIRS := ../../../EspMock/libraries
|
||||||
include ../../../EspMock/EspMock.mk
|
EPOXY_CORE := EPOXY_CORE_ESP8266
|
||||||
|
include ../../../EpoxyDuino/EpoxyDuino.mk
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
#include <Arduino.h>
|
||||||
#include <AUnit.h>
|
#include <AUnit.h>
|
||||||
#include <TinyMqtt.h>
|
#include <TinyMqtt.h>
|
||||||
#include <map>
|
#include <map>
|
||||||
@@ -30,12 +31,10 @@ void onPublish(const MqttClient* srce, const Topic& topic, const char* payload,
|
|||||||
|
|
||||||
test(local_client_should_unregister_when_destroyed)
|
test(local_client_should_unregister_when_destroyed)
|
||||||
{
|
{
|
||||||
return;
|
|
||||||
assertEqual(broker.clientsCount(), (size_t)0);
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
{
|
{
|
||||||
MqttClient client;
|
|
||||||
assertEqual(broker.clientsCount(), (size_t)0); // Ensure client is not yet connected
|
assertEqual(broker.clientsCount(), (size_t)0); // Ensure client is not yet connected
|
||||||
client.connect("127.0.0.1", 1883);
|
MqttClient client(&broker);
|
||||||
assertEqual(broker.clientsCount(), (size_t)1); // Ensure client is now connected
|
assertEqual(broker.clientsCount(), (size_t)1); // Ensure client is now connected
|
||||||
}
|
}
|
||||||
assertEqual(broker.clientsCount(), (size_t)0);
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
@@ -67,8 +66,8 @@ test(local_publish_should_be_dispatched)
|
|||||||
publisher.publish("a/c");
|
publisher.publish("a/c");
|
||||||
|
|
||||||
assertEqual(published.size(), (size_t)1); // 1 client has received something
|
assertEqual(published.size(), (size_t)1); // 1 client has received something
|
||||||
assertTrue(published[""]["a/b"] == 1);
|
assertEqual(published[""]["a/b"], 1);
|
||||||
assertTrue(published[""]["a/c"] == 2);
|
assertEqual(published[""]["a/c"], 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
test(local_publish_should_be_dispatched_to_local_clients)
|
test(local_publish_should_be_dispatched_to_local_clients)
|
||||||
@@ -90,10 +89,10 @@ test(local_publish_should_be_dispatched_to_local_clients)
|
|||||||
publisher.publish("a/c");
|
publisher.publish("a/c");
|
||||||
|
|
||||||
assertEqual(published.size(), (size_t)2); // 2 clients have received something
|
assertEqual(published.size(), (size_t)2); // 2 clients have received something
|
||||||
assertTrue(published["A"]["a/b"] == 1);
|
assertEqual(published["A"]["a/b"], 1);
|
||||||
assertTrue(published["A"]["a/c"] == 1);
|
assertEqual(published["A"]["a/c"], 1);
|
||||||
assertTrue(published["B"]["a/b"] == 1);
|
assertEqual(published["B"]["a/b"], 1);
|
||||||
assertTrue(published["B"]["a/c"] == 0);
|
assertEqual(published["B"]["a/c"], 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
test(local_unsubscribe)
|
test(local_unsubscribe)
|
||||||
@@ -113,7 +112,7 @@ test(local_unsubscribe)
|
|||||||
publisher.publish("a/b");
|
publisher.publish("a/b");
|
||||||
publisher.publish("a/b");
|
publisher.publish("a/b");
|
||||||
|
|
||||||
assertTrue(published[""]["a/b"] == 1); // Only one publish has been received
|
assertEqual(published[""]["a/b"], 1); // Only one publish has been received
|
||||||
}
|
}
|
||||||
|
|
||||||
test(local_nocallback_when_destroyed)
|
test(local_nocallback_when_destroyed)
|
||||||
@@ -142,7 +141,7 @@ void setup() {
|
|||||||
Serial.begin(115200);
|
Serial.begin(115200);
|
||||||
while(!Serial);
|
while(!Serial);
|
||||||
|
|
||||||
Serial.println("=============[ NO WIFI CONNECTION TinyMqtt TESTS ]========================");
|
Serial.println("=============[ LOCAL TinyMqtt TESTS ]========================");
|
||||||
}
|
}
|
||||||
|
|
||||||
void loop() {
|
void loop() {
|
||||||
|
|||||||
13
tests/network-tests/Makefile
Normal file
13
tests/network-tests/Makefile
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
||||||
|
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
||||||
|
|
||||||
|
EXTRA_CXXFLAGS=-g3 -O0
|
||||||
|
|
||||||
|
# Remove flto flag from EpoxyDuino (too many <optimized out>)
|
||||||
|
CXXFLAGS = -Wextra -Wall -std=gnu++11 -fno-exceptions -fno-threadsafe-statics
|
||||||
|
|
||||||
|
APP_NAME := network-tests
|
||||||
|
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsyncTCP
|
||||||
|
ARDUINO_LIB_DIRS := ../../../EspMock/libraries
|
||||||
|
EPOXY_CORE := EPOXY_CORE_ESP8266
|
||||||
|
include ../../../EpoxyDuino/EpoxyDuino.mk
|
||||||
429
tests/network-tests/network-tests.ino
Normal file
429
tests/network-tests/network-tests.ino
Normal file
@@ -0,0 +1,429 @@
|
|||||||
|
#include <Arduino.h>
|
||||||
|
#include <AUnit.h>
|
||||||
|
#include <TinyMqtt.h>
|
||||||
|
#include <map>
|
||||||
|
#include <iostream>
|
||||||
|
#include <iomanip>
|
||||||
|
#include <sstream>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* TinyMqtt network unit tests.
|
||||||
|
*
|
||||||
|
* No wifi connection unit tests.
|
||||||
|
* Checks with a local broker. Clients must connect to the local broker
|
||||||
|
**/
|
||||||
|
|
||||||
|
// if ascii_pos = 0, no ascii dump, else ascii dump starts after column ascii_pos
|
||||||
|
std::string bufferToHexa(const uint8_t* buffer, size_t length, char sep = 0, size_t ascii_pos = 0)
|
||||||
|
{
|
||||||
|
std::stringstream out;
|
||||||
|
std::string ascii;
|
||||||
|
std::string h("0123456789ABCDEF");
|
||||||
|
for(size_t i=0; i<length; i++)
|
||||||
|
{
|
||||||
|
uint8_t c = buffer[i];
|
||||||
|
out << h[ c >> 4] << h[ c & 0x0F ];
|
||||||
|
if (sep) out << sep;
|
||||||
|
if (ascii_pos)
|
||||||
|
{
|
||||||
|
if (c>=32)
|
||||||
|
ascii += c;
|
||||||
|
else
|
||||||
|
ascii +='.';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
std::string ret(out.str());
|
||||||
|
if (ascii_pos)
|
||||||
|
{
|
||||||
|
while(ret.length() < ascii_pos)
|
||||||
|
ret += ' ';
|
||||||
|
ret +='[' + ascii + ']';
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
void dumpMqttMessage(const uint8_t* buffer, size_t length)
|
||||||
|
{
|
||||||
|
std::map<int, std::string> pkt =
|
||||||
|
{ { MqttMessage::Unknown , "Unknown " },
|
||||||
|
{ MqttMessage::Connect , "Connect " },
|
||||||
|
{ MqttMessage::ConnAck , "ConnAck " },
|
||||||
|
{ MqttMessage::Publish , "Publish " },
|
||||||
|
{ MqttMessage::PubAck , "PubAck " },
|
||||||
|
{ MqttMessage::Subscribe , "Subscribe " },
|
||||||
|
{ MqttMessage::SubAck , "SubAck " },
|
||||||
|
{ MqttMessage::UnSubscribe , "Unsubscribe " },
|
||||||
|
{ MqttMessage::UnSuback , "UnSubAck " },
|
||||||
|
{ MqttMessage::PingReq , "PingReq " },
|
||||||
|
{ MqttMessage::PingResp , "PingResp " },
|
||||||
|
{ MqttMessage::Disconnect , "Disconnect " } };
|
||||||
|
|
||||||
|
std::cout << " | data sent " << std::setw(3) << length << " : ";
|
||||||
|
auto it = pkt.find(buffer[0] & 0xF0);
|
||||||
|
if (it == pkt.end())
|
||||||
|
std::cout << pkt[MqttMessage::Unknown];
|
||||||
|
else
|
||||||
|
std::cout << it->second;
|
||||||
|
|
||||||
|
std::cout << bufferToHexa(buffer, length, ' ', 60) << std::endl;
|
||||||
|
}
|
||||||
|
|
||||||
|
String toString(const IPAddress& ip)
|
||||||
|
{
|
||||||
|
return String(ip[0])+'.'+String(ip[1])+'.'+String(ip[2])+'.'+String(ip[3]);
|
||||||
|
}
|
||||||
|
|
||||||
|
MqttBroker broker(1883);
|
||||||
|
|
||||||
|
std::map<std::string, std::map<Topic, int>> published; // map[client_id] => map[topic] = count
|
||||||
|
|
||||||
|
char* lastPayload = nullptr;
|
||||||
|
size_t lastLength;
|
||||||
|
|
||||||
|
void start_servers(int n, bool early_accept = true)
|
||||||
|
{
|
||||||
|
ESP8266WiFiClass::resetInstances();
|
||||||
|
ESP8266WiFiClass::earlyAccept = early_accept;
|
||||||
|
while(n)
|
||||||
|
{
|
||||||
|
ESP8266WiFiClass::selectInstance(n--);
|
||||||
|
WiFi.mode(WIFI_STA);
|
||||||
|
WiFi.begin("fake_ssid", "fake_pwd");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void onPublish(const MqttClient* srce, const Topic& topic, const char* payload, size_t length)
|
||||||
|
{
|
||||||
|
if (srce)
|
||||||
|
published[srce->id()][topic]++;
|
||||||
|
|
||||||
|
if (lastPayload) free(lastPayload);
|
||||||
|
lastPayload = strdup(payload);
|
||||||
|
lastLength = length;
|
||||||
|
}
|
||||||
|
|
||||||
|
test(network_single_broker_begin)
|
||||||
|
{
|
||||||
|
assertEqual(WiFi.status(), WL_CONNECTED);
|
||||||
|
|
||||||
|
MqttBroker broker(1883);
|
||||||
|
broker.begin();
|
||||||
|
|
||||||
|
// TODO Nothing is tested here !
|
||||||
|
}
|
||||||
|
|
||||||
|
test(suback)
|
||||||
|
{
|
||||||
|
start_servers(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.toString().c_str(), 1883);
|
||||||
|
broker.loop();
|
||||||
|
|
||||||
|
assertTrue(broker.clientsCount() == 1);
|
||||||
|
assertTrue(client.connected());
|
||||||
|
|
||||||
|
MqttClient::counters[MqttMessage::Type::SubAck] = 0;
|
||||||
|
client.subscribe("a/b");
|
||||||
|
|
||||||
|
// TODO how to avoid these loops ???
|
||||||
|
broker.loop();
|
||||||
|
client.loop();
|
||||||
|
|
||||||
|
assertEqual(MqttClient::counters[MqttMessage::Type::SubAck], 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(network_client_to_broker_connexion)
|
||||||
|
{
|
||||||
|
start_servers(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.toString().c_str(), 1883);
|
||||||
|
broker.loop();
|
||||||
|
|
||||||
|
assertTrue(broker.clientsCount() == 1);
|
||||||
|
assertTrue(client.connected());
|
||||||
|
}
|
||||||
|
|
||||||
|
test(network_one_client_one_broker_publish_and_subscribe_through_network)
|
||||||
|
{
|
||||||
|
start_servers(2, true);
|
||||||
|
published.clear();
|
||||||
|
assertEqual(WiFi.status(), WL_CONNECTED);
|
||||||
|
|
||||||
|
MqttBroker broker(1883);
|
||||||
|
broker.begin();
|
||||||
|
IPAddress ip_broker = WiFi.localIP();
|
||||||
|
|
||||||
|
// We have a 2nd ESP in order to test through wifi (opposed to local)
|
||||||
|
ESP8266WiFiClass::selectInstance(2);
|
||||||
|
MqttClient client;
|
||||||
|
client.connect(ip_broker.toString().c_str(), 1883);
|
||||||
|
broker.loop();
|
||||||
|
assertTrue(client.connected());
|
||||||
|
|
||||||
|
client.setCallback(onPublish);
|
||||||
|
client.subscribe("a/b");
|
||||||
|
client.publish("a/b", "ab");
|
||||||
|
|
||||||
|
for (int i =0; i<2; i++)
|
||||||
|
{
|
||||||
|
client.loop();
|
||||||
|
broker.loop();
|
||||||
|
}
|
||||||
|
|
||||||
|
assertEqual(published.size(), (size_t)1);
|
||||||
|
assertEqual((int)lastLength, (int)2); // sizeof(ab)
|
||||||
|
}
|
||||||
|
|
||||||
|
test(network_one_client_one_broker_hudge_publish_and_subscribe_through_network)
|
||||||
|
{
|
||||||
|
start_servers(2, true);
|
||||||
|
published.clear();
|
||||||
|
assertEqual(WiFi.status(), WL_CONNECTED);
|
||||||
|
|
||||||
|
MqttBroker broker(1883);
|
||||||
|
broker.begin();
|
||||||
|
IPAddress ip_broker = WiFi.localIP();
|
||||||
|
|
||||||
|
// We have a 2nd ESP in order to test through wifi (opposed to local)
|
||||||
|
ESP8266WiFiClass::selectInstance(2);
|
||||||
|
MqttClient client;
|
||||||
|
client.connect(ip_broker.toString().c_str(), 1883);
|
||||||
|
broker.loop();
|
||||||
|
assertTrue(client.connected());
|
||||||
|
|
||||||
|
std::string sent;
|
||||||
|
|
||||||
|
for(int i=0; i<200; i++)
|
||||||
|
sent += char('0'+i%10);
|
||||||
|
|
||||||
|
client.setCallback(onPublish);
|
||||||
|
client.subscribe("a/b");
|
||||||
|
client.publish("a/b", sent.c_str());
|
||||||
|
|
||||||
|
for (int i =0; i<2; i++)
|
||||||
|
{
|
||||||
|
client.loop();
|
||||||
|
broker.loop();
|
||||||
|
}
|
||||||
|
|
||||||
|
assertEqual(published.size(), (size_t)1);
|
||||||
|
assertEqual((unsigned int)lastLength, (unsigned int)sent.size());
|
||||||
|
}
|
||||||
|
|
||||||
|
test(network_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);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// THESE TESTS ARE IN LOCAL MODE
|
||||||
|
// WE HAVE TO CONVERT THEM TO WIFI MODE (pass through virtual TCP link)
|
||||||
|
|
||||||
|
test(network_connect)
|
||||||
|
{
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
|
MqttClient client(&broker);
|
||||||
|
assertTrue(client.connected());
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)1);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(network_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
|
||||||
|
assertEqual(published[""]["a/b"], 1);
|
||||||
|
assertEqual(published[""]["a/c"], 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(network_publish_should_be_dispatched_to_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"); // A and B should receive this
|
||||||
|
publisher.publish("a/c"); // A should receive this
|
||||||
|
|
||||||
|
assertEqual(published.size(), (size_t)2); // 2 clients have received something
|
||||||
|
assertEqual(published["A"]["a/b"], 1);
|
||||||
|
assertEqual(published["A"]["a/c"], 1);
|
||||||
|
assertEqual(published["B"]["a/b"], 1);
|
||||||
|
assertEqual(published["B"]["a/c"], 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(network_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"); // This publish is received
|
||||||
|
|
||||||
|
subscriber.unsubscribe("a/b");
|
||||||
|
|
||||||
|
publisher.publish("a/b"); // Those one, no (unsubscribed)
|
||||||
|
publisher.publish("a/b");
|
||||||
|
|
||||||
|
assertEqual(published[""]["a/b"], 1); // Only one publish has been received
|
||||||
|
}
|
||||||
|
|
||||||
|
test(network_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
|
||||||
|
}
|
||||||
|
|
||||||
|
test(network_small_payload)
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
|
||||||
|
const char* payload="abcd";
|
||||||
|
|
||||||
|
MqttClient subscriber(&broker);
|
||||||
|
subscriber.setCallback(onPublish);
|
||||||
|
subscriber.subscribe("a/b");
|
||||||
|
|
||||||
|
MqttClient publisher(&broker);
|
||||||
|
publisher.publish("a/b", payload, strlen(payload)); // This publish is received
|
||||||
|
|
||||||
|
// coming from MqttClient::publish(...)
|
||||||
|
assertEqual(payload, lastPayload);
|
||||||
|
assertEqual(lastLength, (size_t)4);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(network_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 !";
|
||||||
|
|
||||||
|
MqttClient subscriber(&broker);
|
||||||
|
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);
|
||||||
|
assertEqual(lastLength, strlen(payload));
|
||||||
|
assertEqual(strcmp(payload, lastPayload), 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(connack)
|
||||||
|
{
|
||||||
|
const bool view = false;
|
||||||
|
|
||||||
|
NetworkObserver check(
|
||||||
|
[this](const WiFiClient*, const uint8_t* buffer, size_t length)
|
||||||
|
{
|
||||||
|
if (view) dumpMqttMessage(buffer, length);
|
||||||
|
if (buffer[0] == MqttMessage::ConnAck)
|
||||||
|
{
|
||||||
|
std::string hex = bufferToHexa(buffer, length);
|
||||||
|
assertStringCaseEqual(hex.c_str(), "20020000");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
start_servers(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.toString().c_str(), 1883);
|
||||||
|
broker.loop();
|
||||||
|
|
||||||
|
assertTrue(broker.clientsCount() == 1);
|
||||||
|
assertTrue(client.connected());
|
||||||
|
|
||||||
|
MqttClient::counters[MqttMessage::Type::SubAck] = 0;
|
||||||
|
client.subscribe("a/b");
|
||||||
|
|
||||||
|
// TODO how to avoid these loops ???
|
||||||
|
broker.loop();
|
||||||
|
client.loop();
|
||||||
|
|
||||||
|
assertEqual(MqttClient::counters[MqttMessage::Type::SubAck], 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
//----------------------------------------------------------------------------
|
||||||
|
// setup() and loop()
|
||||||
|
void setup() {
|
||||||
|
/* delay(1000);
|
||||||
|
Serial.begin(115200);
|
||||||
|
while(!Serial);
|
||||||
|
*/
|
||||||
|
|
||||||
|
Serial.println("=============[ FAKE NETWORK TinyMqtt TESTS ]========================");
|
||||||
|
|
||||||
|
WiFi.mode(WIFI_STA);
|
||||||
|
WiFi.begin("network", "password");
|
||||||
|
}
|
||||||
|
|
||||||
|
void loop() {
|
||||||
|
aunit::TestRunner::run();
|
||||||
|
|
||||||
|
if (Serial.available()) ESP.reset();
|
||||||
|
}
|
||||||
@@ -1,7 +1,13 @@
|
|||||||
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
||||||
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
||||||
|
|
||||||
|
EXTRA_CXXFLAGS=-g3 -O0
|
||||||
|
|
||||||
|
# Remove flto flag from EpoxyDuino (too many <optimized out>)
|
||||||
|
CXXFLAGS = -Wextra -Wall -std=gnu++11 -fno-exceptions -fno-threadsafe-statics
|
||||||
|
|
||||||
APP_NAME := nowifi-tests
|
APP_NAME := nowifi-tests
|
||||||
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock
|
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsyncTCP
|
||||||
ESP_LIBS = ESP8266WiFi ESPAsyncTCP
|
ARDUINO_LIB_DIRS := ../../../EspMock/libraries
|
||||||
include ../../../EspMock/EspMock.mk
|
EPOXY_CORE := EPOXY_CORE_ESP8266
|
||||||
|
include ../../../EpoxyDuino/EpoxyDuino.mk
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
#include <Arduino.h>
|
||||||
#include <AUnit.h>
|
#include <AUnit.h>
|
||||||
#include <TinyMqtt.h>
|
#include <TinyMqtt.h>
|
||||||
#include <map>
|
#include <map>
|
||||||
@@ -92,6 +93,104 @@ test(nowifi_publish_should_be_dispatched_to_clients)
|
|||||||
assertEqual(published["B"]["a/c"], 0);
|
assertEqual(published["B"]["a/c"], 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
test(nowifi_subscribe_with_star_wildcard)
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
|
MqttClient subscriber(&broker, "A");
|
||||||
|
subscriber.setCallback(onPublish);
|
||||||
|
subscriber.subscribe("house/*/temp");
|
||||||
|
|
||||||
|
MqttClient publisher(&broker);
|
||||||
|
publisher.publish("house/bedroom/temp");
|
||||||
|
publisher.publish("house/kitchen/temp");
|
||||||
|
publisher.publish("house/living_room/tv/temp");
|
||||||
|
publisher.publish("building/location1/bedroom/temp");
|
||||||
|
|
||||||
|
assertEqual(published["A"]["house/bedroom/temp"], 1);
|
||||||
|
assertEqual(published["A"]["house/kitchen/temp"], 1);
|
||||||
|
assertEqual(published["A"]["house/living_room/tv/temp"], 1);
|
||||||
|
assertEqual(published["A"]["building/location1/bedroom/temp"], 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(nowifi_subscribe_with_plus_wildcard)
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
|
MqttClient subscriber(&broker, "A");
|
||||||
|
subscriber.setCallback(onPublish);
|
||||||
|
subscriber.subscribe("house/+/temp");
|
||||||
|
|
||||||
|
MqttClient publisher(&broker);
|
||||||
|
publisher.publish("house/bedroom/temp");
|
||||||
|
publisher.publish("house/kitchen/temp");
|
||||||
|
publisher.publish("house/living_room/tv/temp");
|
||||||
|
publisher.publish("building/location1/bedroom/temp");
|
||||||
|
|
||||||
|
assertEqual(published["A"]["house/bedroom/temp"], 1);
|
||||||
|
assertEqual(published["A"]["house/kitchen/temp"], 1);
|
||||||
|
assertEqual(published["A"]["house/living_room/tv/temp"], 0);
|
||||||
|
assertEqual(published["A"]["building/location1/bedroom/temp"], 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(nowifi_should_not_receive_sys_msg)
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
|
MqttClient subscriber(&broker, "A");
|
||||||
|
subscriber.setCallback(onPublish);
|
||||||
|
subscriber.subscribe("+/data");
|
||||||
|
|
||||||
|
MqttClient publisher(&broker);
|
||||||
|
publisher.publish("$SYS/data");
|
||||||
|
|
||||||
|
assertEqual(published["A"]["$SYS/data"], 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(nowifi_subscribe_with_mixed_wildcards)
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
|
MqttClient subscriber(&broker, "A");
|
||||||
|
subscriber.setCallback(onPublish);
|
||||||
|
subscriber.subscribe("+/data/#");
|
||||||
|
|
||||||
|
MqttClient publisher(&broker);
|
||||||
|
publisher.publish("node1/data/update");
|
||||||
|
publisher.publish("node2/data/delta");
|
||||||
|
publisher.publish("node3/data");
|
||||||
|
|
||||||
|
assertEqual(published["A"]["node1/data/update"], 1);
|
||||||
|
assertEqual(published["A"]["node2/data/delta"], 1);
|
||||||
|
assertEqual(published["A"]["node3/data"], 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
test(nowifi_unsubscribe_with_wildcards)
|
||||||
|
{
|
||||||
|
published.clear();
|
||||||
|
assertEqual(broker.clientsCount(), (size_t)0);
|
||||||
|
|
||||||
|
MqttClient subscriber(&broker, "A");
|
||||||
|
subscriber.setCallback(onPublish);
|
||||||
|
subscriber.subscribe("one/two/+");
|
||||||
|
subscriber.subscribe("one/two/three");
|
||||||
|
|
||||||
|
MqttClient publisher(&broker);
|
||||||
|
publisher.publish("one/two/three");
|
||||||
|
publisher.publish("one/two/four");
|
||||||
|
|
||||||
|
subscriber.unsubscribe("one/two/+");
|
||||||
|
publisher.publish("one/two/five");
|
||||||
|
|
||||||
|
assertEqual(published["A"]["one/two/three"], 1);
|
||||||
|
assertEqual(published["A"]["one/two/four"], 1);
|
||||||
|
assertEqual(published["A"]["one/two/five"], 0);
|
||||||
|
}
|
||||||
|
|
||||||
test(nowifi_unsubscribe)
|
test(nowifi_unsubscribe)
|
||||||
{
|
{
|
||||||
published.clear();
|
published.clear();
|
||||||
@@ -131,7 +230,7 @@ test(nowifi_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(nowifi_payload_nullptr)
|
test(nowifi_small_payload)
|
||||||
{
|
{
|
||||||
published.clear();
|
published.clear();
|
||||||
|
|
||||||
@@ -149,6 +248,22 @@ test(nowifi_payload_nullptr)
|
|||||||
assertEqual(lastLength, (size_t)4);
|
assertEqual(lastLength, (size_t)4);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
test(nowifi_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 !";
|
||||||
|
|
||||||
|
MqttClient subscriber(&broker);
|
||||||
|
subscriber.setCallback(onPublish);
|
||||||
|
subscriber.subscribe("a/b");
|
||||||
|
|
||||||
|
MqttClient publisher(&broker);
|
||||||
|
publisher.publish("a/b", payload); // This publish is received
|
||||||
|
|
||||||
|
// onPublish should have filled lastPayload and lastLength
|
||||||
|
assertEqual(payload, lastPayload);
|
||||||
|
assertEqual(lastLength, strlen(payload));
|
||||||
|
}
|
||||||
|
|
||||||
//----------------------------------------------------------------------------
|
//----------------------------------------------------------------------------
|
||||||
// setup() and loop()
|
// setup() and loop()
|
||||||
void setup() {
|
void setup() {
|
||||||
|
|||||||
@@ -1,7 +1,10 @@
|
|||||||
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
||||||
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
||||||
|
|
||||||
|
EXTRA_CXXFLAGS=-g3 -O0
|
||||||
|
|
||||||
APP_NAME := string-indexer-tests
|
APP_NAME := string-indexer-tests
|
||||||
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock
|
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsync
|
||||||
ESP_LIBS = ESP8266WiFi ESPAsyncTCP
|
ARDUINO_LIB_DIRS := ../../../EspMock/libraries
|
||||||
include ../../../EspMock/EspMock.mk
|
EPOXY_CORE := EPOXY_CORE_ESP8266
|
||||||
|
include ../../../EpoxyDuino/EpoxyDuino.mk
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
#include <Arduino.h>
|
||||||
#include <AUnit.h>
|
#include <AUnit.h>
|
||||||
#include <StringIndexer.h>
|
#include <StringIndexer.h>
|
||||||
#include <map>
|
#include <map>
|
||||||
@@ -61,6 +62,14 @@ test(indexer_same_strings_should_equal)
|
|||||||
assertTrue(one == two);
|
assertTrue(one == two);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
test(indexer_compare_strings_with_same_beginning)
|
||||||
|
{
|
||||||
|
IndexedString two("one_two");
|
||||||
|
IndexedString one("one");
|
||||||
|
|
||||||
|
assertNotEqual(one.getIndex(), two.getIndex());
|
||||||
|
}
|
||||||
|
|
||||||
test(indexer_indexed_operator_eq)
|
test(indexer_indexed_operator_eq)
|
||||||
{
|
{
|
||||||
IndexedString one("one");
|
IndexedString one("one");
|
||||||
@@ -105,5 +114,5 @@ void setup() {
|
|||||||
void loop() {
|
void loop() {
|
||||||
aunit::TestRunner::run();
|
aunit::TestRunner::run();
|
||||||
|
|
||||||
if (Serial.available()) ESP.reset();
|
// if (Serial.available()) ESP.reset();
|
||||||
}
|
}
|
||||||
|
|||||||
10
tests/topic-tests/Makefile
Normal file
10
tests/topic-tests/Makefile
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
# See https://github.com/bxparks/EpoxyDuino for documentation about this
|
||||||
|
# Makefile to compile and run Arduino programs natively on Linux or MacOS.
|
||||||
|
|
||||||
|
EXTRA_CXXFLAGS=-g3 -O0
|
||||||
|
|
||||||
|
APP_NAME := topic-tests
|
||||||
|
ARDUINO_LIBS := AUnit AceCommon AceTime TinyMqtt EspMock ESP8266WiFi ESPAsync
|
||||||
|
ARDUINO_LIB_DIRS := ../../../EspMock/libraries
|
||||||
|
EPOXY_CORE := EPOXY_CORE_ESP8266
|
||||||
|
include ../../../EpoxyDuino/EpoxyDuino.mk
|
||||||
85
tests/topic-tests/topic-tests.ino
Normal file
85
tests/topic-tests/topic-tests.ino
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
#include <Arduino.h>
|
||||||
|
#include <AUnit.h>
|
||||||
|
#include <TinyMqtt.h>
|
||||||
|
#include <map>
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
#define endl "\n"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* TinyMqtt / StringIndexer unit tests.
|
||||||
|
*
|
||||||
|
**/
|
||||||
|
|
||||||
|
using namespace std;
|
||||||
|
|
||||||
|
bool testTopicMatch(const char* a, const char* b, bool expected)
|
||||||
|
{
|
||||||
|
Topic ta(a);
|
||||||
|
Topic tb(b);
|
||||||
|
bool match(ta.matches(tb));
|
||||||
|
cout << " " << ta.c_str() << ' ';
|
||||||
|
if (match != expected)
|
||||||
|
cout << (expected ? " should match " : " should not match ");
|
||||||
|
else
|
||||||
|
cout << (expected ? " matches " : " unmatches ");
|
||||||
|
cout << tb.c_str() << endl;
|
||||||
|
return expected == match;
|
||||||
|
}
|
||||||
|
|
||||||
|
test(topic_matches)
|
||||||
|
{
|
||||||
|
// matching
|
||||||
|
assertTrue(testTopicMatch("a/b/c" , "a/b/c" , true));
|
||||||
|
assertTrue(testTopicMatch("a/*/c" , "a/xyz/c" , true));
|
||||||
|
assertTrue(testTopicMatch("a/*/e" , "a/b/c/d/e" , true));
|
||||||
|
assertTrue(testTopicMatch("a/*" , "a/b/c/d/e" , true));
|
||||||
|
assertTrue(testTopicMatch("*/c" , "a/b/c" , true));
|
||||||
|
assertTrue(testTopicMatch("/*/c" , "/a/b/c" , true));
|
||||||
|
assertTrue(testTopicMatch("a/*" , "a/b/c/d" , true));
|
||||||
|
assertTrue(testTopicMatch("a/+/c" , "a/b/c" , true));
|
||||||
|
assertTrue(testTopicMatch("a/+/c/+/e", "a/b/c/d/e" , true));
|
||||||
|
assertTrue(testTopicMatch("a/*/c/+/e", "a/b/c/d/e" , true));
|
||||||
|
assertTrue(testTopicMatch("/+/b" , "/a/b" , true));
|
||||||
|
assertTrue(testTopicMatch("+" , "a" , true));
|
||||||
|
assertTrue(testTopicMatch("a/b/#" , "a/b/c/d" , true));
|
||||||
|
assertTrue(testTopicMatch("a/b/#" , "a/b" , true));
|
||||||
|
assertTrue(testTopicMatch("a/*/c" , "a/*/c" , true));
|
||||||
|
|
||||||
|
// not matching
|
||||||
|
assertTrue(testTopicMatch("a/b/c" , "a/b/d" , false));
|
||||||
|
assertTrue(testTopicMatch("a/b/c" , "a/b/d" , false));
|
||||||
|
assertTrue(testTopicMatch("a/*/e" , "a/b/c/d/f" , false));
|
||||||
|
assertTrue(testTopicMatch("a/+" , "a" , false));
|
||||||
|
assertTrue(testTopicMatch("a/+" , "a/b/d" , false));
|
||||||
|
assertTrue(testTopicMatch("a/+/" , "a/" , false));
|
||||||
|
|
||||||
|
// $SYS topics
|
||||||
|
assertTrue(testTopicMatch("+/any" , "$SYS/any" , false));
|
||||||
|
assertTrue(testTopicMatch("*/any" , "$SYS/any" , false));
|
||||||
|
assertTrue(testTopicMatch("$SYS/any" , "$SYS/any" , true));
|
||||||
|
assertTrue(testTopicMatch("$SYS/+/y" , "$SYS/a/y" , true));
|
||||||
|
assertTrue(testTopicMatch("$SYS/#" , "$SYS/a/y" , true));
|
||||||
|
|
||||||
|
// not valid
|
||||||
|
assertTrue(testTopicMatch("a/#/b" , "a/x/b" , false));
|
||||||
|
assertTrue(testTopicMatch("a+" , "a/b/d" , false));
|
||||||
|
assertTrue(testTopicMatch("a/b/#/d" , "a/b/c/d" , false));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
//----------------------------------------------------------------------------
|
||||||
|
// setup() and loop()
|
||||||
|
void setup() {
|
||||||
|
delay(1000);
|
||||||
|
Serial.begin(115200);
|
||||||
|
while(!Serial);
|
||||||
|
|
||||||
|
Serial.println("=============[ TinyMqtt StringIndexer TESTS ]========================");
|
||||||
|
}
|
||||||
|
|
||||||
|
void loop() {
|
||||||
|
aunit::TestRunner::run();
|
||||||
|
|
||||||
|
// if (Serial.available()) ESP.reset();
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user