* simple_sensor: added alert send queue, with retries, checks for ACKs, etc. Low pri alerts only 1 send attempt, otherwise 4 attempts
This commit is contained in:
@@ -58,6 +58,8 @@
|
|||||||
|
|
||||||
#define LAZY_CONTACTS_WRITE_DELAY 5000
|
#define LAZY_CONTACTS_WRITE_DELAY 5000
|
||||||
|
|
||||||
|
#define ALERT_ACK_EXPIRY_MILLIS 6000 // wait 6 secs for ACKs to alert messages
|
||||||
|
|
||||||
static File openAppend(FILESYSTEM* _fs, const char* fname) {
|
static File openAppend(FILESYSTEM* _fs, const char* fname) {
|
||||||
#if defined(NRF52_PLATFORM) || defined(STM32_PLATFORM)
|
#if defined(NRF52_PLATFORM) || defined(STM32_PLATFORM)
|
||||||
return _fs->open(fname, FILE_O_WRITE);
|
return _fs->open(fname, FILE_O_WRITE);
|
||||||
@@ -163,6 +165,7 @@ static uint8_t getDataSize(uint8_t type) {
|
|||||||
case LPP_TEMPERATURE:
|
case LPP_TEMPERATURE:
|
||||||
case LPP_CONCENTRATION:
|
case LPP_CONCENTRATION:
|
||||||
case LPP_BAROMETRIC_PRESSURE:
|
case LPP_BAROMETRIC_PRESSURE:
|
||||||
|
case LPP_RELATIVE_HUMIDITY:
|
||||||
case LPP_ALTITUDE:
|
case LPP_ALTITUDE:
|
||||||
case LPP_VOLTAGE:
|
case LPP_VOLTAGE:
|
||||||
case LPP_CURRENT:
|
case LPP_CURRENT:
|
||||||
@@ -185,6 +188,7 @@ static uint32_t getMultiplier(uint8_t type) {
|
|||||||
return 100;
|
return 100;
|
||||||
case LPP_TEMPERATURE:
|
case LPP_TEMPERATURE:
|
||||||
case LPP_BAROMETRIC_PRESSURE:
|
case LPP_BAROMETRIC_PRESSURE:
|
||||||
|
case LPP_RELATIVE_HUMIDITY:
|
||||||
return 10;
|
return 10;
|
||||||
}
|
}
|
||||||
return 1;
|
return 1;
|
||||||
@@ -332,46 +336,54 @@ void SensorMesh::applyContactPermissions(const uint8_t* pubkey, uint16_t perms)
|
|||||||
dirty_contacts_expiry = futureMillis(LAZY_CONTACTS_WRITE_DELAY); // trigger saveContacts()
|
dirty_contacts_expiry = futureMillis(LAZY_CONTACTS_WRITE_DELAY); // trigger saveContacts()
|
||||||
}
|
}
|
||||||
|
|
||||||
void SensorMesh::sendAlert(AlertPriority pri, const char* text) {
|
void SensorMesh::sendAlert(ContactInfo* c, Trigger* t) {
|
||||||
int text_len = strlen(text);
|
int text_len = strlen(t->text);
|
||||||
uint16_t pri_mask = (pri == HIGH_PRI_ALERT) ? PERM_RECV_ALERTS_HI : PERM_RECV_ALERTS_LO;
|
|
||||||
|
|
||||||
// send text message to all contacts with RECV_ALERT permission
|
uint8_t data[MAX_PACKET_PAYLOAD];
|
||||||
for (int i = 0; i < num_contacts; i++) {
|
memcpy(data, &t->timestamp, 4);
|
||||||
auto c = &contacts[i];
|
data[4] = (TXT_TYPE_PLAIN << 2) | t->attempt; // attempt and flags
|
||||||
if ((c->permissions & pri_mask) == 0) continue; // contact does NOT want alert
|
memcpy(&data[5], t->text, text_len);
|
||||||
|
|
||||||
uint8_t data[MAX_PACKET_PAYLOAD];
|
// calc expected ACK reply
|
||||||
uint32_t now = getRTCClock()->getCurrentTimeUnique(); // need different timestamp per packet
|
mesh::Utils::sha256((uint8_t *)&t->expected_acks[t->attempt], 4, data, 5 + text_len, self_id.pub_key, PUB_KEY_SIZE);
|
||||||
memcpy(data, &now, 4);
|
t->attempt++;
|
||||||
data[4] = (TXT_TYPE_PLAIN << 2); // attempt and flags
|
|
||||||
memcpy(&data[5], text, text_len);
|
|
||||||
// calc expected ACK reply
|
|
||||||
// uint32_t expected_ack;
|
|
||||||
// mesh::Utils::sha256((uint8_t *)&expected_ack, 4, data, 5 + text_len, self_id.pub_key, PUB_KEY_SIZE);
|
|
||||||
|
|
||||||
auto pkt = createDatagram(PAYLOAD_TYPE_TXT_MSG, c->id, c->shared_secret, data, 5 + text_len);
|
auto pkt = createDatagram(PAYLOAD_TYPE_TXT_MSG, c->id, c->shared_secret, data, 5 + text_len);
|
||||||
if (pkt) {
|
if (pkt) {
|
||||||
if (c->out_path_len >= 0) { // we have an out_path, so send DIRECT
|
if (c->out_path_len >= 0) { // we have an out_path, so send DIRECT
|
||||||
sendDirect(pkt, c->out_path, c->out_path_len);
|
sendDirect(pkt, c->out_path, c->out_path_len);
|
||||||
} else {
|
} else {
|
||||||
sendFlood(pkt);
|
sendFlood(pkt);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
t->send_expiry = futureMillis(ALERT_ACK_EXPIRY_MILLIS);
|
||||||
}
|
}
|
||||||
|
|
||||||
void SensorMesh::alertIf(bool condition, Trigger& t, AlertPriority pri, const char* text) {
|
void SensorMesh::alertIf(bool condition, Trigger& t, AlertPriority pri, const char* text) {
|
||||||
if (condition) {
|
if (condition) {
|
||||||
if (!t.triggered) {
|
if (!t.isTriggered() && num_alert_tasks < MAX_CONCURRENT_ALERTS) {
|
||||||
t.triggered = true;
|
StrHelper::strncpy(t.text, text, sizeof(t.text));
|
||||||
t.time = getRTCClock()->getCurrentTime();
|
t.pri = pri;
|
||||||
sendAlert(pri, text);
|
t.send_expiry = 0; // signal that initial send is needed
|
||||||
|
t.attempt = 4;
|
||||||
|
t.curr_contact_idx = -1; // start iterating thru contacts[]
|
||||||
|
|
||||||
|
alert_tasks[num_alert_tasks++] = &t; // add to queue
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (t.triggered) {
|
if (t.isTriggered()) {
|
||||||
t.triggered = false;
|
t.text[0] = 0;
|
||||||
// TODO: apply debounce logic
|
// remove 't' from alert queue
|
||||||
|
int i = 0;
|
||||||
|
while (i < num_alert_tasks && alert_tasks[i] != &t) i++;
|
||||||
|
|
||||||
|
if (i < num_alert_tasks) { // found, now delete from array
|
||||||
|
num_alert_tasks--;
|
||||||
|
while (i < num_alert_tasks) {
|
||||||
|
alert_tasks[i] = alert_tasks[i + 1];
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -629,6 +641,20 @@ bool SensorMesh::onPeerPathRecv(mesh::Packet* packet, int sender_idx, const uint
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void SensorMesh::onAckRecv(mesh::Packet* packet, uint32_t ack_crc) {
|
||||||
|
if (num_alert_tasks > 0) {
|
||||||
|
auto t = alert_tasks[0]; // check current alert task
|
||||||
|
for (int i = 0; i < t->attempt; i++) {
|
||||||
|
if (ack_crc == t->expected_acks[i]) { // matching ACK!
|
||||||
|
t->attempt = 4; // signal to move to next contact
|
||||||
|
t->send_expiry = 0;
|
||||||
|
packet->markDoNotRetransmit(); // ACK was for this node, so don't retransmit
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
SensorMesh::SensorMesh(mesh::MainBoard& board, mesh::Radio& radio, mesh::MillisecondClock& ms, mesh::RNG& rng, mesh::RTCClock& rtc, mesh::MeshTables& tables)
|
SensorMesh::SensorMesh(mesh::MainBoard& board, mesh::Radio& radio, mesh::MillisecondClock& ms, mesh::RNG& rng, mesh::RTCClock& rtc, mesh::MeshTables& tables)
|
||||||
: mesh::Mesh(radio, ms, rng, rtc, *new StaticPoolPacketManager(32), tables),
|
: mesh::Mesh(radio, ms, rng, rtc, *new StaticPoolPacketManager(32), tables),
|
||||||
_cli(board, rtc, &_prefs, this), telemetry(MAX_PACKET_PAYLOAD - 4)
|
_cli(board, rtc, &_prefs, this), telemetry(MAX_PACKET_PAYLOAD - 4)
|
||||||
@@ -637,6 +663,7 @@ SensorMesh::SensorMesh(mesh::MainBoard& board, mesh::Radio& radio, mesh::Millise
|
|||||||
next_local_advert = next_flood_advert = 0;
|
next_local_advert = next_flood_advert = 0;
|
||||||
dirty_contacts_expiry = 0;
|
dirty_contacts_expiry = 0;
|
||||||
last_read_time = 0;
|
last_read_time = 0;
|
||||||
|
num_alert_tasks = 0;
|
||||||
|
|
||||||
// defaults
|
// defaults
|
||||||
memset(&_prefs, 0, sizeof(_prefs));
|
memset(&_prefs, 0, sizeof(_prefs));
|
||||||
@@ -736,7 +763,14 @@ float SensorMesh::getTelemValue(uint8_t channel, uint8_t type) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool SensorMesh::getGPS(uint8_t channel, float& lat, float& lon, float& alt) {
|
bool SensorMesh::getGPS(uint8_t channel, float& lat, float& lon, float& alt) {
|
||||||
return false; // TODO
|
if (channel == TELEM_CHANNEL_SELF) {
|
||||||
|
lat = sensors.node_lat;
|
||||||
|
lon = sensors.node_lon;
|
||||||
|
alt = sensors.node_altitude;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
// REVISIT: custom GPS channels??
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void SensorMesh::loop() {
|
void SensorMesh::loop() {
|
||||||
@@ -767,6 +801,42 @@ void SensorMesh::loop() {
|
|||||||
last_read_time = curr;
|
last_read_time = curr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// check the alert send queue
|
||||||
|
if (num_alert_tasks > 0) {
|
||||||
|
auto t = alert_tasks[0]; // process head of queue
|
||||||
|
|
||||||
|
if (millisHasNowPassed(t->send_expiry)) { // next send needed?
|
||||||
|
if (t->attempt >= 4) { // max attempts reached, try next contact
|
||||||
|
t->curr_contact_idx++;
|
||||||
|
if (t->curr_contact_idx >= num_contacts) { // no more contacts to try?
|
||||||
|
num_alert_tasks--; // remove t from queue
|
||||||
|
for (int i = 0; i < num_alert_tasks; i++) {
|
||||||
|
alert_tasks[i] = alert_tasks[i + 1];
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
auto c = &contacts[t->curr_contact_idx];
|
||||||
|
uint16_t pri_mask = (t->pri == HIGH_PRI_ALERT) ? PERM_RECV_ALERTS_HI : PERM_RECV_ALERTS_LO;
|
||||||
|
|
||||||
|
if (c->permissions & pri_mask) { // contact wants alert
|
||||||
|
// reset attempts
|
||||||
|
t->attempt = (t->pri == LOW_PRI_ALERT) ? 3 : 0; // Low pri alerts, start at attempt #3 (ie. only make ONE attempt)
|
||||||
|
t->timestamp = getRTCClock()->getCurrentTimeUnique(); // need unique timestamp per contact
|
||||||
|
|
||||||
|
sendAlert(c, t); // NOTE: modifies attempt, expected_acks[] and send_expiry
|
||||||
|
} else {
|
||||||
|
// next contact tested in next ::loop()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if (t->curr_contact_idx < num_contacts) {
|
||||||
|
auto c = &contacts[t->curr_contact_idx]; // send next attempt
|
||||||
|
sendAlert(c, t); // NOTE: modifies attempt, expected_acks[] and send_expiry
|
||||||
|
} else {
|
||||||
|
// contact list has likely been modified while waiting for alert ACK, cancel this task
|
||||||
|
t->attempt = 4; // next ::loop() will remove t from queue
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// is there are pending dirty contacts write needed?
|
// is there are pending dirty contacts write needed?
|
||||||
if (dirty_contacts_expiry && millisHasNowPassed(dirty_contacts_expiry)) {
|
if (dirty_contacts_expiry && millisHasNowPassed(dirty_contacts_expiry)) {
|
||||||
saveContacts();
|
saveContacts();
|
||||||
|
|||||||
@@ -55,7 +55,8 @@ struct ContactInfo {
|
|||||||
#define MAX_CONTACTS 32
|
#define MAX_CONTACTS 32
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#define MAX_SEARCH_RESULTS 8
|
#define MAX_SEARCH_RESULTS 8
|
||||||
|
#define MAX_CONCURRENT_ALERTS 4
|
||||||
|
|
||||||
class SensorMesh : public mesh::Mesh, public CommonCLICallbacks {
|
class SensorMesh : public mesh::Mesh, public CommonCLICallbacks {
|
||||||
public:
|
public:
|
||||||
@@ -99,13 +100,20 @@ protected:
|
|||||||
bool getGPS(uint8_t channel, float& lat, float& lon, float& alt);
|
bool getGPS(uint8_t channel, float& lat, float& lon, float& alt);
|
||||||
|
|
||||||
// alerts
|
// alerts
|
||||||
struct Trigger {
|
|
||||||
bool triggered;
|
|
||||||
uint32_t time;
|
|
||||||
|
|
||||||
Trigger() { triggered = false; time = 0; }
|
|
||||||
};
|
|
||||||
enum AlertPriority { LOW_PRI_ALERT, HIGH_PRI_ALERT };
|
enum AlertPriority { LOW_PRI_ALERT, HIGH_PRI_ALERT };
|
||||||
|
|
||||||
|
struct Trigger {
|
||||||
|
uint32_t timestamp;
|
||||||
|
AlertPriority pri;
|
||||||
|
uint32_t expected_acks[4];
|
||||||
|
int8_t curr_contact_idx;
|
||||||
|
uint8_t attempt;
|
||||||
|
unsigned long send_expiry;
|
||||||
|
char text[MAX_PACKET_PAYLOAD];
|
||||||
|
|
||||||
|
Trigger() { text[0] = 0; }
|
||||||
|
bool isTriggered() const { return text[0] != 0; }
|
||||||
|
};
|
||||||
void alertIf(bool condition, Trigger& t, AlertPriority pri, const char* text);
|
void alertIf(bool condition, Trigger& t, AlertPriority pri, const char* text);
|
||||||
|
|
||||||
virtual void onSensorDataRead() = 0; // for app to implement
|
virtual void onSensorDataRead() = 0; // for app to implement
|
||||||
@@ -124,6 +132,7 @@ protected:
|
|||||||
void getPeerSharedSecret(uint8_t* dest_secret, int peer_idx) override;
|
void getPeerSharedSecret(uint8_t* dest_secret, int peer_idx) override;
|
||||||
void onPeerDataRecv(mesh::Packet* packet, uint8_t type, int sender_idx, const uint8_t* secret, uint8_t* data, size_t len) override;
|
void onPeerDataRecv(mesh::Packet* packet, uint8_t type, int sender_idx, const uint8_t* secret, uint8_t* data, size_t len) override;
|
||||||
bool onPeerPathRecv(mesh::Packet* packet, int sender_idx, const uint8_t* secret, uint8_t* path, uint8_t path_len, uint8_t extra_type, uint8_t* extra, uint8_t extra_len) override;
|
bool onPeerPathRecv(mesh::Packet* packet, int sender_idx, const uint8_t* secret, uint8_t* path, uint8_t path_len, uint8_t extra_type, uint8_t* extra, uint8_t extra_len) override;
|
||||||
|
void onAckRecv(mesh::Packet* packet, uint32_t ack_crc) override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
FILESYSTEM* _fs;
|
FILESYSTEM* _fs;
|
||||||
@@ -137,6 +146,8 @@ private:
|
|||||||
CayenneLPP telemetry;
|
CayenneLPP telemetry;
|
||||||
uint32_t last_read_time;
|
uint32_t last_read_time;
|
||||||
int matching_peer_indexes[MAX_SEARCH_RESULTS];
|
int matching_peer_indexes[MAX_SEARCH_RESULTS];
|
||||||
|
int num_alert_tasks;
|
||||||
|
Trigger* alert_tasks[MAX_CONCURRENT_ALERTS];
|
||||||
|
|
||||||
void loadContacts();
|
void loadContacts();
|
||||||
void saveContacts();
|
void saveContacts();
|
||||||
@@ -146,6 +157,6 @@ private:
|
|||||||
ContactInfo* putContact(const mesh::Identity& id);
|
ContactInfo* putContact(const mesh::Identity& id);
|
||||||
void applyContactPermissions(const uint8_t* pubkey, uint16_t perms);
|
void applyContactPermissions(const uint8_t* pubkey, uint16_t perms);
|
||||||
|
|
||||||
void sendAlert(AlertPriority pri, const char* text);
|
void sendAlert(ContactInfo* c, Trigger* t);
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ void TimeSeriesData::recordData(mesh::RTCClock* clock, float value) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void TimeSeriesData::calcDataMinMaxAvg(mesh::RTCClock* clock, uint32_t start_secs_ago, uint32_t end_secs_ago, MinMaxAvg* dest, uint8_t channel, uint8_t lpp_type) const {
|
void TimeSeriesData::calcMinMaxAvg(mesh::RTCClock* clock, uint32_t start_secs_ago, uint32_t end_secs_ago, MinMaxAvg* dest, uint8_t channel, uint8_t lpp_type) const {
|
||||||
int i = next, n = num_slots;
|
int i = next, n = num_slots;
|
||||||
uint32_t ago = clock->getCurrentTime() - last_timestamp;
|
uint32_t ago = clock->getCurrentTime() - last_timestamp;
|
||||||
int num_values = 0;
|
int num_values = 0;
|
||||||
@@ -40,6 +40,6 @@ void TimeSeriesData::calcDataMinMaxAvg(mesh::RTCClock* clock, uint32_t start_sec
|
|||||||
if (num_values > 0) {
|
if (num_values > 0) {
|
||||||
dest->_avg = total / num_values;
|
dest->_avg = total / num_values;
|
||||||
} else {
|
} else {
|
||||||
dest->_avg = NAN;
|
dest->_max = dest->_min = dest->_avg = NAN;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -24,6 +24,6 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
void recordData(mesh::RTCClock* clock, float value);
|
void recordData(mesh::RTCClock* clock, float value);
|
||||||
void calcDataMinMaxAvg(mesh::RTCClock* clock, uint32_t start_secs_ago, uint32_t end_secs_ago, MinMaxAvg* dest, uint8_t channel, uint8_t lpp_type) const;
|
void calcMinMaxAvg(mesh::RTCClock* clock, uint32_t start_secs_ago, uint32_t end_secs_ago, MinMaxAvg* dest, uint8_t channel, uint8_t lpp_type) const;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -15,18 +15,19 @@ public:
|
|||||||
|
|
||||||
protected:
|
protected:
|
||||||
/* ========================== custom logic here ========================== */
|
/* ========================== custom logic here ========================== */
|
||||||
Trigger low_batt;
|
Trigger low_batt, critical_batt;
|
||||||
TimeSeriesData battery_data;
|
TimeSeriesData battery_data;
|
||||||
|
|
||||||
void onSensorDataRead() override {
|
void onSensorDataRead() override {
|
||||||
float batt_voltage = getVoltage(TELEM_CHANNEL_SELF);
|
float batt_voltage = getVoltage(TELEM_CHANNEL_SELF);
|
||||||
|
|
||||||
battery_data.recordData(getRTCClock(), batt_voltage); // record battery
|
battery_data.recordData(getRTCClock(), batt_voltage); // record battery
|
||||||
alertIf(batt_voltage < 3.4f, low_batt, HIGH_PRI_ALERT, "Battery low!");
|
alertIf(batt_voltage < 3.4f, critical_batt, HIGH_PRI_ALERT, "Battery is critical!");
|
||||||
|
alertIf(batt_voltage < 3.6f, low_batt, LOW_PRI_ALERT, "Battery is low");
|
||||||
}
|
}
|
||||||
|
|
||||||
int querySeriesData(uint32_t start_secs_ago, uint32_t end_secs_ago, MinMaxAvg dest[], int max_num) override {
|
int querySeriesData(uint32_t start_secs_ago, uint32_t end_secs_ago, MinMaxAvg dest[], int max_num) override {
|
||||||
battery_data.calcDataMinMaxAvg(getRTCClock(), start_secs_ago, end_secs_ago, &dest[0], TELEM_CHANNEL_SELF, LPP_VOLTAGE);
|
battery_data.calcMinMaxAvg(getRTCClock(), start_secs_ago, end_secs_ago, &dest[0], TELEM_CHANNEL_SELF, LPP_VOLTAGE);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
/* ======================================================================= */
|
/* ======================================================================= */
|
||||||
|
|||||||
Reference in New Issue
Block a user