2020-06-02 22:38:20 +02:00
|
|
|
#include <EEPROM.h>
|
2020-06-02 00:57:20 +02:00
|
|
|
#include <Ethernet.h>
|
2020-06-03 18:53:33 +02:00
|
|
|
#include <EthernetUdp.h>
|
2020-06-08 23:17:16 +02:00
|
|
|
#include <PubSubClient.h>
|
2020-06-03 18:53:33 +02:00
|
|
|
#include <NTPClient.h>
|
2020-06-02 00:57:20 +02:00
|
|
|
#include <ArduinoJson.h>
|
|
|
|
|
2020-06-08 23:17:16 +02:00
|
|
|
#define DEBUG_TO_SERIAL 1 // debug on serial port
|
|
|
|
#define USE_MQTT 0 // use mqtt protocol instead of http post
|
2020-06-03 18:53:33 +02:00
|
|
|
#define USE_INTERNAL_NTP 0 // use default ntp server or the internal one
|
2020-06-08 23:17:16 +02:00
|
|
|
#define AREF_VOLTAGE 3.3 // set aref voltage to 3.3v instead of default 5v
|
|
|
|
|
|
|
|
char serial[9];
|
2020-06-02 00:57:20 +02:00
|
|
|
|
2020-06-03 18:53:33 +02:00
|
|
|
// const String serverName = "sensor.server.domain";
|
2020-06-03 23:53:25 +02:00
|
|
|
const size_t capacity = 2 * JSON_OBJECT_SIZE(3) + JSON_OBJECT_SIZE(2) + 20;
|
2020-06-02 00:57:20 +02:00
|
|
|
|
2020-06-03 23:53:25 +02:00
|
|
|
DynamicJsonDocument telemetry(capacity);
|
|
|
|
JsonObject payload = telemetry.createNestedObject("payload");
|
2020-06-02 00:57:20 +02:00
|
|
|
JsonObject temp = payload.createNestedObject("temperature");
|
|
|
|
|
2020-06-03 18:53:33 +02:00
|
|
|
unsigned int counter = 0;
|
2020-06-05 13:24:12 +02:00
|
|
|
|
2020-06-03 18:53:33 +02:00
|
|
|
EthernetUDP ntpUDP;
|
|
|
|
NTPClient timeClient(ntpUDP);
|
|
|
|
bool NTPValid = false;
|
|
|
|
|
2020-06-08 23:17:16 +02:00
|
|
|
EthernetClient ethClient;
|
|
|
|
PubSubClient clientMQTT(ethClient);
|
|
|
|
|
2020-06-02 22:38:20 +02:00
|
|
|
struct netConfig {
|
|
|
|
IPAddress address;
|
2020-06-03 18:53:33 +02:00
|
|
|
unsigned int port;
|
2020-06-02 22:38:20 +02:00
|
|
|
};
|
|
|
|
netConfig config;
|
2020-06-02 00:57:20 +02:00
|
|
|
|
2020-06-05 13:24:12 +02:00
|
|
|
const String apiURL = "/api/device/subscribe/";
|
2020-06-03 23:53:25 +02:00
|
|
|
const String telemetryURL = "/telemetry/";
|
2020-06-02 00:57:20 +02:00
|
|
|
|
|
|
|
void setup(void) {
|
|
|
|
Serial.begin(9600);
|
2020-06-03 18:53:33 +02:00
|
|
|
|
2020-06-02 00:57:20 +02:00
|
|
|
analogReference(EXTERNAL);
|
2020-06-02 22:38:20 +02:00
|
|
|
|
2020-06-03 23:53:25 +02:00
|
|
|
StaticJsonDocument<20> api;
|
|
|
|
|
2020-06-02 22:38:20 +02:00
|
|
|
byte mac[6];
|
|
|
|
int eeAddress = 0;
|
|
|
|
|
|
|
|
EEPROM.get(eeAddress, mac);
|
|
|
|
eeAddress += sizeof(mac);
|
|
|
|
EEPROM.get(eeAddress, serial);
|
|
|
|
eeAddress += sizeof(serial);
|
2020-06-03 18:53:33 +02:00
|
|
|
|
2020-06-02 00:57:20 +02:00
|
|
|
if (Ethernet.begin(mac) == 0) {
|
2020-06-02 22:38:20 +02:00
|
|
|
if (Ethernet.hardwareStatus() == EthernetNoHardware) {
|
2020-06-03 18:53:33 +02:00
|
|
|
Serial.println("ERROR: ethernet shield was not found.");
|
2020-06-02 22:38:20 +02:00
|
|
|
}
|
|
|
|
while (true) {
|
|
|
|
delay(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
EEPROM.get(eeAddress, config);
|
2020-06-03 18:53:33 +02:00
|
|
|
|
2020-06-02 22:38:20 +02:00
|
|
|
Serial.print("IoT #");
|
|
|
|
Serial.print(serial);
|
2020-06-08 23:17:16 +02:00
|
|
|
Serial.print(" at address: ");
|
2020-06-02 22:38:20 +02:00
|
|
|
Serial.println(Ethernet.localIP());
|
2020-06-02 00:57:20 +02:00
|
|
|
Serial.println();
|
2020-06-08 23:17:16 +02:00
|
|
|
Serial.print("Connecting to: ");
|
2020-06-02 22:38:20 +02:00
|
|
|
Serial.print(config.address);
|
|
|
|
Serial.print(":");
|
|
|
|
Serial.println(config.port);
|
2020-06-02 00:57:20 +02:00
|
|
|
|
2020-06-03 18:53:33 +02:00
|
|
|
#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
|
|
|
|
|
2020-06-03 23:53:25 +02:00
|
|
|
api["serial"] = serial;
|
|
|
|
postData(config, apiURL, api);
|
|
|
|
|
|
|
|
telemetry["device"] = serial;
|
2020-06-03 18:53:33 +02:00
|
|
|
// payload["id"] = serverName;
|
2020-06-08 23:17:16 +02:00
|
|
|
|
|
|
|
#if USE_MQTT
|
|
|
|
clientMQTT.setServer(config.address, 1883);
|
|
|
|
#endif
|
2020-06-02 00:57:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void loop(void) {
|
2020-06-05 13:24:12 +02:00
|
|
|
const int postDelay = 10 * 1000;
|
2020-06-03 18:53:33 +02:00
|
|
|
|
2020-06-05 13:24:12 +02:00
|
|
|
unsigned int tempReading = analogRead(A0);
|
|
|
|
unsigned int photocellReading = analogRead(A1);
|
2020-06-02 00:57:20 +02:00
|
|
|
|
|
|
|
float tempVoltage = tempReading * AREF_VOLTAGE / 1024.0;
|
|
|
|
float tempC = (tempVoltage - 0.5) * 100 ;
|
|
|
|
|
2020-06-03 18:53:33 +02:00
|
|
|
if (NTPValid) {
|
2020-06-03 23:53:25 +02:00
|
|
|
telemetry["clock"] = timeClient.getEpochTime();
|
2020-06-03 18:53:33 +02:00
|
|
|
} else {
|
2020-06-03 23:53:25 +02:00
|
|
|
telemetry["clock"] = NULL; // converted into 0
|
2020-06-03 18:53:33 +02:00
|
|
|
}
|
2020-06-02 00:57:20 +02:00
|
|
|
payload["light"] = photocellReading;
|
|
|
|
|
|
|
|
temp["celsius"] = tempC;
|
|
|
|
temp["raw"] = tempReading;
|
|
|
|
temp["volts"] = tempVoltage;
|
2020-06-03 18:53:33 +02:00
|
|
|
|
2020-06-08 23:17:16 +02:00
|
|
|
#if USE_MQTT
|
|
|
|
publishData(config, telemetry);
|
|
|
|
#else
|
2020-06-03 23:53:25 +02:00
|
|
|
postData(config, telemetryURL, telemetry);
|
2020-06-08 23:17:16 +02:00
|
|
|
#endif
|
2020-06-03 23:53:25 +02:00
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2020-06-08 23:17:16 +02:00
|
|
|
#if USE_MQTT
|
|
|
|
void publishData(const netConfig &mqtt, const DynamicJsonDocument &json) {
|
|
|
|
if (clientMQTT.connect(serial, "freedcs", "password")) {
|
|
|
|
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
|
|
|
|
|
2020-06-03 23:53:25 +02:00
|
|
|
void postData(const netConfig &postAPI, const String &URL, const DynamicJsonDocument &json) {
|
|
|
|
if (EthernetClient client = client.connect(postAPI.address, postAPI.port)) {
|
2020-06-02 00:57:20 +02:00
|
|
|
client.print("POST ");
|
|
|
|
client.print(URL);
|
|
|
|
client.println(" HTTP/1.1");
|
|
|
|
client.print("Host: ");
|
2020-06-03 23:53:25 +02:00
|
|
|
client.print(postAPI.address);
|
2020-06-02 00:57:20 +02:00
|
|
|
client.print(":");
|
2020-06-03 23:53:25 +02:00
|
|
|
client.println(postAPI.port);
|
2020-06-02 00:57:20 +02:00
|
|
|
client.println("Content-Type: application/json");
|
|
|
|
client.print("Content-Length: ");
|
2020-06-03 18:53:33 +02:00
|
|
|
client.println(measureJsonPretty(json));
|
2020-06-02 00:57:20 +02:00
|
|
|
client.println("Connection: close");
|
|
|
|
client.println();
|
2020-06-03 18:53:33 +02:00
|
|
|
serializeJson(json, client);
|
2020-06-02 00:57:20 +02:00
|
|
|
client.stop();
|
2020-06-03 18:53:33 +02:00
|
|
|
|
|
|
|
#if DEBUG_TO_SERIAL
|
2020-06-08 23:17:16 +02:00
|
|
|
Serial.println("DEBUG: HTTP POST>>>");
|
2020-06-03 18:53:33 +02:00
|
|
|
serializeJsonPretty(json, Serial);
|
|
|
|
Serial.println("\n<<<");
|
|
|
|
#endif
|
|
|
|
}
|
2020-06-02 00:57:20 +02:00
|
|
|
}
|