mirror of
https://github.com/1technophile/OpenMQTTGateway.git
synced 2026-02-26 03:31:42 +01:00
By building only the log level required into LOG_LEVEL Co-authored-by: Florian <1technophile@users.noreply.github.com>
233 lines
7.5 KiB
C++
233 lines
7.5 KiB
C++
/*
|
|
Theengs OpenMQTTGateway - We Unite Sensors in One Open-Source Interface
|
|
|
|
Act as a wifi or ethernet gateway between your RF/infrared IR signal and a MQTT broker
|
|
Send and receiving command by MQTT
|
|
|
|
This gateway enables to:
|
|
- receive MQTT data from a topic and send RF signal corresponding to the received MQTT data with an RFM69 module
|
|
- publish MQTT data to a different topic related to received signal from an RFM69 module
|
|
|
|
Copyright: (c)Felix Rusu LowPowerLab.com
|
|
Library and code by Felix Rusu - felix@lowpowerlab.com
|
|
Modification of the code nanohab from bbx10 https://github.com/bbx10/nanohab
|
|
Copyright: (c)Florian ROBERT
|
|
|
|
This file is part of OpenMQTTGateway.
|
|
|
|
OpenMQTTGateway is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
OpenMQTTGateway 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 General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
#include "User_config.h"
|
|
|
|
#ifdef ZgatewayRFM69
|
|
# include <EEPROM.h>
|
|
# include <RFM69.h> //https://www.github.com/lowpowerlab/rfm69
|
|
|
|
# include "TheengsCommon.h"
|
|
# include "config_RFM69.h"
|
|
|
|
static char RadioConfig[128];
|
|
static const char* PROGMEM ENCRYPTKEY = RFM69_ENCRYPTKEY;
|
|
static const char* PROGMEM MDNS_NAME = RFM69_MDNS_NAME;
|
|
static const char* PROGMEM MQTT_BROKER = RFM69_MQTT_BROKER;
|
|
static const char* PROGMEM RFM69AP_NAME = RFM69_RFM69AP_NAME
|
|
|
|
// vvvvvvvvv Global Configuration vvvvvvvvvvv
|
|
|
|
struct _GLOBAL_CONFIG {
|
|
uint32_t checksum;
|
|
char rfmapname[32];
|
|
char encryptkey[16 + 1];
|
|
uint8_t networkid;
|
|
uint8_t nodeid;
|
|
uint8_t powerlevel; // bits 0..4 power leve, bit 7 RFM69HCW 1=true
|
|
uint8_t rfmfrequency;
|
|
};
|
|
|
|
# define GC_POWER_LEVEL (pGC->powerlevel & 0x1F)
|
|
# define GC_IS_RFM69HCW ((pGC->powerlevel & 0x80) != 0)
|
|
# define SELECTED_FREQ(f) ((pGC->rfmfrequency == f) ? "selected" : "")
|
|
|
|
struct _GLOBAL_CONFIG* pGC;
|
|
|
|
// vvvvvvvvv Global Configuration vvvvvvvvvvv
|
|
uint32_t gc_checksum() {
|
|
uint8_t* p = (uint8_t*)pGC;
|
|
uint32_t checksum = 0;
|
|
p += sizeof(pGC->checksum);
|
|
for (size_t i = 0; i < (sizeof(*pGC) - 4); i++) {
|
|
checksum += *p++;
|
|
}
|
|
return checksum;
|
|
}
|
|
|
|
void eeprom_setup() {
|
|
EEPROM.begin(4096);
|
|
pGC = (struct _GLOBAL_CONFIG*)EEPROM.getDataPtr();
|
|
// if checksum bad init GC else use GC values
|
|
if (gc_checksum() != pGC->checksum) {
|
|
THEENGS_LOG_TRACE(F("Factory reset" CR));
|
|
memset(pGC, 0, sizeof(*pGC));
|
|
strcpy_P(pGC->encryptkey, ENCRYPTKEY);
|
|
strcpy_P(pGC->rfmapname, RFM69AP_NAME);
|
|
pGC->networkid = RFM69_NETWORKID;
|
|
pGC->nodeid = RFM69_NODEID;
|
|
pGC->powerlevel = ((IS_RFM69HCW) ? 0x80 : 0x00) | POWER_LEVEL;
|
|
pGC->rfmfrequency = FREQUENCY;
|
|
pGC->checksum = gc_checksum();
|
|
EEPROM.commit();
|
|
}
|
|
}
|
|
|
|
RFM69 radio;
|
|
|
|
void setupRFM69(void) {
|
|
eeprom_setup();
|
|
int freq;
|
|
static const char PROGMEM JSONtemplate[] =
|
|
R"({"msgType":"config","freq":%d,"rfm69hcw":%d,"netid":%d,"power":%d})";
|
|
char payload[128];
|
|
|
|
radio = RFM69(RFM69_CS, RFM69_IRQ, GC_IS_RFM69HCW, RFM69_IRQN);
|
|
|
|
// Initialize radio
|
|
if (!radio.initialize(pGC->rfmfrequency, pGC->nodeid, pGC->networkid)) {
|
|
THEENGS_LOG_ERROR(F("gatewayRFM69 initialization failed" CR));
|
|
}
|
|
|
|
if (GC_IS_RFM69HCW) {
|
|
radio.setHighPower(); // Only for RFM69HCW & HW!
|
|
}
|
|
radio.setPowerLevel(GC_POWER_LEVEL); // power output ranges from 0 (5dBm) to 31 (20dBm)
|
|
|
|
if (pGC->encryptkey[0] != '\0')
|
|
radio.encrypt(pGC->encryptkey);
|
|
|
|
switch (pGC->rfmfrequency) {
|
|
case RF69_433MHZ:
|
|
freq = 433;
|
|
break;
|
|
case RF69_868MHZ:
|
|
freq = 868;
|
|
break;
|
|
case RF69_915MHZ:
|
|
freq = 915;
|
|
break;
|
|
case RF69_315MHZ:
|
|
freq = 315;
|
|
break;
|
|
default:
|
|
freq = -1;
|
|
break;
|
|
}
|
|
THEENGS_LOG_NOTICE(F("gatewayRFM69 Listening and transmitting at: %d" CR), freq);
|
|
|
|
size_t len = snprintf_P(RadioConfig, sizeof(RadioConfig), JSONtemplate,
|
|
freq, GC_IS_RFM69HCW, pGC->networkid, GC_POWER_LEVEL);
|
|
if (len >= sizeof(RadioConfig)) {
|
|
THEENGS_LOG_TRACE(F("\n\n*** RFM69 config truncated ***\n" CR));
|
|
}
|
|
}
|
|
|
|
bool RFM69toX(void) {
|
|
//check if something was received (could be an interrupt from the radio)
|
|
if (radio.receiveDone()) {
|
|
StaticJsonDocument<JSON_MSG_BUFFER> RFM69dataBuffer;
|
|
JsonObject RFM69data = RFM69dataBuffer.to<JsonObject>();
|
|
uint8_t data[RF69_MAX_DATA_LEN + 1]; // For the null character
|
|
uint8_t SENDERID = radio.SENDERID;
|
|
uint8_t DATALEN = radio.DATALEN;
|
|
uint16_t RSSI = radio.RSSI;
|
|
|
|
//save packet because it may be overwritten
|
|
memcpy(data, (void*)radio.DATA, DATALEN);
|
|
data[DATALEN] = '\0'; // Terminate the string
|
|
|
|
// Ack as soon as possible
|
|
//check if sender wanted an ACK
|
|
if (radio.ACKRequested()) {
|
|
radio.sendACK();
|
|
}
|
|
//updateClients(senderId, rssi, (const char *)data);
|
|
|
|
THEENGS_LOG_TRACE(F("Data received: %s" CR), (const char*)data);
|
|
|
|
char buff[sizeof(subjectRFM69toMQTT) + 4];
|
|
sprintf(buff, "%s/%d", subjectRFM69toMQTT, SENDERID);
|
|
RFM69data["data"] = (char*)data;
|
|
RFM69data["rssi"] = (int)radio.RSSI;
|
|
RFM69data["senderid"] = (int)radio.SENDERID;
|
|
RFM69data["origin"] = buff;
|
|
enqueueJsonObject(RFM69data);
|
|
|
|
return true;
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
# if simpleReceiving
|
|
void XtoRFM69(const char* topicOri, const char* datacallback) {
|
|
if (cmpToMainTopic(topicOri, subjectMQTTtoRFM69)) {
|
|
THEENGS_LOG_TRACE(F("MQTTtoRFM69 data analysis" CR));
|
|
char data[RF69_MAX_DATA_LEN + 1];
|
|
memcpy(data, (void*)datacallback, RF69_MAX_DATA_LEN);
|
|
data[RF69_MAX_DATA_LEN] = '\0';
|
|
|
|
//We look into the subject to see if a special RF protocol is defined
|
|
String topic = topicOri;
|
|
int valueRCV = defaultRFM69ReceiverId; //default receiver id value
|
|
int pos = topic.lastIndexOf(RFM69receiverKey);
|
|
if (pos != -1) {
|
|
pos = pos + +strlen(RFM69receiverKey);
|
|
valueRCV = (topic.substring(pos, pos + 3)).toInt();
|
|
THEENGS_LOG_NOTICE(F("RFM69 receiver ID: %d" CR), valueRCV);
|
|
}
|
|
if (radio.sendWithRetry(valueRCV, data, strlen(data), 10)) {
|
|
THEENGS_LOG_NOTICE(F(" OK " CR));
|
|
// Acknowledgement to the GTWRF topic
|
|
char buff[sizeof(subjectGTWRFM69toMQTT) + 4];
|
|
sprintf(buff, "%s/%d", subjectGTWRFM69toMQTT, radio.SENDERID);
|
|
pub(buff, data);
|
|
} else {
|
|
THEENGS_LOG_ERROR(F("RFM69 sending failed" CR));
|
|
}
|
|
}
|
|
}
|
|
# endif
|
|
# if jsonReceiving
|
|
void XtoRFM69(const char* topicOri, JsonObject& RFM69data) {
|
|
if (cmpToMainTopic(topicOri, subjectMQTTtoRFM69)) {
|
|
const char* data = RFM69data["data"];
|
|
THEENGS_LOG_TRACE(F("MQTTtoRFM69 json data analysis" CR));
|
|
if (data) {
|
|
THEENGS_LOG_TRACE(F("MQTTtoRFM69 data ok" CR));
|
|
int valueRCV = RFM69data["receiverid"] | defaultRFM69ReceiverId; //default receiver id value
|
|
THEENGS_LOG_NOTICE(F("RFM69 receiver ID: %d" CR), valueRCV);
|
|
if (radio.sendWithRetry(valueRCV, data, strlen(data), 10)) {
|
|
THEENGS_LOG_NOTICE(F(" OK " CR));
|
|
// Acknowledgement to the GTWRF topic
|
|
RFM69data["origin"] = subjectGTWRFM69toMQTT;
|
|
enqueueJsonObject(RFM69data);
|
|
} else {
|
|
THEENGS_LOG_ERROR(F("MQTTtoRFM69 sending failed" CR));
|
|
}
|
|
} else {
|
|
THEENGS_LOG_ERROR(F("MQTTtoRFM69 failed json read" CR));
|
|
}
|
|
}
|
|
}
|
|
# endif
|
|
#endif
|