mirror of
https://github.com/daniviga/bite.git
synced 2025-01-11 16:21:00 +01:00
198 lines
5.2 KiB
C++
198 lines
5.2 KiB
C++
/* -*- coding: utf-8 -*-
|
|
* vim: tabstop=2 shiftwidth=2 softtabstop=2
|
|
*
|
|
* BITE - A Basic/IoT/Example
|
|
* Copyright (C) 2020-2021 Daniele Viganò <daniele@vigano.me>
|
|
*
|
|
* BITE is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Affero General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* BITE is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Affero General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <Preferences.h>
|
|
#include <WiFi.h>
|
|
#include <WiFiUdp.h>
|
|
#include <PubSubClient.h>
|
|
#include <NTPClient.h>
|
|
#include <ArduinoJson.h>
|
|
|
|
#define DEBUG_TO_SERIAL 1 // debug on serial port
|
|
#define USE_MQTT 1 // use mqtt protocol instead of http post
|
|
#define USE_INTERNAL_NTP 1 // use default ntp server or the internal one
|
|
#define TELEMETRY_DELAY 10 // second between telemetry samples
|
|
|
|
// const String serverName = "sensor.server.domain";
|
|
const size_t capacity = 2 * JSON_OBJECT_SIZE(3) + JSON_OBJECT_SIZE(2) + 20;
|
|
|
|
Preferences preferences;
|
|
|
|
DynamicJsonDocument telemetry(capacity);
|
|
JsonObject payload = telemetry.createNestedObject("payload");
|
|
|
|
unsigned int counter = 0;
|
|
|
|
WiFiUDP ntpUDP;
|
|
NTPClient timeClient(ntpUDP);
|
|
bool NTPValid = false;
|
|
|
|
WiFiClient ethClient;
|
|
PubSubClient clientMQTT(ethClient);
|
|
|
|
struct netConfig {
|
|
IPAddress address;
|
|
unsigned int port;
|
|
} config;
|
|
|
|
char* serial;
|
|
const String apiURL = "/api/device/subscribe/";
|
|
const String telemetryURL = "/telemetry/";
|
|
|
|
void setup(void) {
|
|
Serial.begin(115200);
|
|
|
|
StaticJsonDocument<64> api;
|
|
|
|
preferences.begin("iot");
|
|
// Get the serial number from flash
|
|
serial = strdup(preferences.getString("serial").c_str());
|
|
|
|
// Get network configuration
|
|
size_t _len = preferences.getBytesLength("config");
|
|
char _buffer[_len];
|
|
preferences.getBytes("config", &_buffer, _len);
|
|
memcpy(&config, _buffer, _len);
|
|
preferences.end();
|
|
|
|
// Get WiFi parameters
|
|
preferences.begin("wifi");
|
|
const char* _ssid = strdup(preferences.getString("ssid").c_str());
|
|
const char* _password = strdup(preferences.getString("password").c_str());
|
|
preferences.end();
|
|
|
|
Serial.print("Starting connecting WiFi to ");
|
|
Serial.print(_ssid);
|
|
delay(10);
|
|
WiFi.begin(_ssid, _password);
|
|
while (WiFi.status() != WL_CONNECTED) {
|
|
delay(500);
|
|
Serial.print(".");
|
|
}
|
|
Serial.println("\nWiFi connected");
|
|
|
|
Serial.print("IoT #");
|
|
Serial.print(serial);
|
|
Serial.print(" at address: ");
|
|
Serial.println(WiFi.localIP());
|
|
Serial.println();
|
|
Serial.print("Connecting to: ");
|
|
Serial.print(config.address.toString());
|
|
Serial.print(":");
|
|
Serial.print(config.port);
|
|
Serial.print(" every ");
|
|
Serial.print(TELEMETRY_DELAY);
|
|
Serial.println("s");
|
|
|
|
#if USE_INTERNAL_NTP
|
|
timeClient.setPoolServerIP(config.address);
|
|
#endif
|
|
timeClient.begin();
|
|
if (timeClient.update()) {
|
|
NTPValid = true;
|
|
}
|
|
|
|
#if DEBUG_TO_SERIAL
|
|
Serial.println("DEBUG: clock updated via NTP.");
|
|
#endif
|
|
|
|
api["serial"] = serial;
|
|
postData(config, apiURL, api);
|
|
|
|
telemetry["device"] = serial;
|
|
// payload["id"] = serverName;
|
|
|
|
#if USE_MQTT
|
|
clientMQTT.setServer(config.address, 1883);
|
|
#endif
|
|
}
|
|
|
|
void loop(void) {
|
|
const int postDelay = TELEMETRY_DELAY * 1000;
|
|
|
|
if (NTPValid) {
|
|
telemetry["clock"] = timeClient.getEpochTime();
|
|
} else {
|
|
telemetry["clock"] = NULL; // converted into 0
|
|
}
|
|
|
|
payload["hall"] = hallRead();
|
|
payload["wifi-rssi"] = WiFi.RSSI();
|
|
|
|
#if USE_MQTT
|
|
publishData(config, telemetry);
|
|
#else
|
|
postData(config, telemetryURL, telemetry);
|
|
#endif
|
|
|
|
if (counter == 6 * 120) { // Update clock every 6 times * 10 sec * 120 minutes = 2 hrs
|
|
timeClient.update();
|
|
counter = 0;
|
|
#if DEBUG_TO_SERIAL
|
|
Serial.println("DEBUG: clock updated via NTP.");
|
|
#endif
|
|
} else {
|
|
counter++;
|
|
}
|
|
|
|
delay(postDelay);
|
|
}
|
|
|
|
#if USE_MQTT
|
|
void publishData(const netConfig &mqtt, const DynamicJsonDocument &json) {
|
|
if (clientMQTT.connect(serial)) {
|
|
char buffer[256];
|
|
serializeJson(json, buffer);
|
|
clientMQTT.publish(serial, buffer);
|
|
|
|
#if DEBUG_TO_SERIAL
|
|
Serial.println("DEBUG: MQTT PUBLISH>>>");
|
|
serializeJsonPretty(json, Serial);
|
|
Serial.println("\n<<<");
|
|
#endif
|
|
}
|
|
}
|
|
#endif
|
|
|
|
void postData(const netConfig &postAPI, const String &URL, const DynamicJsonDocument &json) {
|
|
if (ethClient.connect(postAPI.address, postAPI.port)) {
|
|
ethClient.print("POST ");
|
|
ethClient.print(URL);
|
|
ethClient.println(" HTTP/1.1");
|
|
ethClient.print("Host: ");
|
|
ethClient.print(postAPI.address.toString());
|
|
ethClient.print(":");
|
|
ethClient.println(postAPI.port);
|
|
ethClient.println("Content-Type: application/json");
|
|
ethClient.print("Content-Length: ");
|
|
ethClient.println(measureJson(json));
|
|
ethClient.println("Connection: close");
|
|
ethClient.println();
|
|
serializeJson(json, ethClient);
|
|
ethClient.stop();
|
|
|
|
#if DEBUG_TO_SERIAL
|
|
Serial.println("DEBUG: HTTP POST>>>");
|
|
serializeJsonPretty(json, Serial);
|
|
Serial.println("\n<<<");
|
|
#endif
|
|
}
|
|
}
|