Modular PR1: NetMgr, HttpClient, SmsMgr with working attach/HTTP/SMS

This commit is contained in:
2025-09-02 09:20:23 +10:00
parent aba2234f53
commit aac68aa77c
8 changed files with 229 additions and 3 deletions

View File

@ -1,9 +1,26 @@
#include "app/App.hpp"
#include "src/core/NetMgr.hpp"
#include "src/core/HttpClient.hpp"
#include "src/core/SmsMgr.hpp"
static const char* APN = "hologram";
static const char* BASE_URL = "http://laravel-server.lab.audasmedia.com.au";
static const char* PATH = "/api/gps";
void setup() {
App::instance().setup();
Serial.begin(115200);
Serial2.begin(9600, SERIAL_8N1, 16, 17); // RX=16, TX=17
NetMgr::powerOnSIM7080(4); // PWRKEY pin 4
NetMgr::attachAndPdp(APN); // LTE reg + CNACT + DNS
SmsMgr::setup(); // enable text mode + URCs
String json = "{\"device_id\":\"sim7080g-01\",\"lat\":-33.865143,"
"\"lng\":151.2099,\"speed\":12.5,\"altitude\":30.2}";
HttpClient::postJson(BASE_URL, PATH, json);
}
void loop() {
App::instance().loop();
SmsMgr::pollUrc(); // nonblocking URC scan; handles +CMTI
SmsMgr::pollUnread(); // optional safety poll
delay(250);
}

View File

@ -1,5 +1,6 @@
#include "App.hpp"
#include "../core/HttpClient.hpp"
App& App::instance() {
static App inst;

View File

@ -0,0 +1,34 @@
#include "HttpClient.hpp"
#include "NetMgr.hpp"
namespace {
bool ensureHttpReady(const String& baseUrl) {
NetMgr::sendAT("AT+SHCONF=\"URL\",\"" + baseUrl + "\"", 5000);
NetMgr::sendAT("AT+SHCONF=\"BODYLEN\",1024", 2000);
NetMgr::sendAT("AT+SHCONF=\"HEADERLEN\",350", 2000);
String r = NetMgr::sendAT("AT+SHCONN", 15000);
if (r.indexOf("OK") == -1) return false;
NetMgr::sendAT("AT+SHCHEAD", 2000);
NetMgr::sendAT("AT+SHAHEAD=\"Content-Type\",\"application/json\"", 2000);
return true;
}
}
namespace HttpClient {
bool postJson(const String& baseUrl, const String& path, const String& jsonBody) {
if (!ensureHttpReady(baseUrl)) { Serial.println("SHCONN failed"); return false; }
NetMgr::sendAT("AT+SHBOD=" + String(jsonBody.length()) + ",10000", 2000);
Serial2.print(jsonBody);
delay(200);
String r = NetMgr::sendAT("AT+SHREQ=\"" + path + "\",3", 20000); // 3=POST
int p = r.indexOf("+SHREQ:");
if (p != -1) Serial.println("Status/URC: " + r.substring(p));
NetMgr::sendAT("AT+SHREAD=0,2048", 4000); // non-fatal if CME 3
NetMgr::sendAT("AT+SHDISC", 3000);
return true;
}
} // namespace HttpClient

View File

@ -0,0 +1,6 @@
#pragma once
#include <Arduino.h>
namespace HttpClient {
bool postJson(const String& baseUrl, const String& path, const String& jsonBody);
}

View File

@ -0,0 +1,61 @@
#include "NetMgr.hpp"
namespace {
String at(const String& cmd, uint32_t timeoutMs) {
Serial.print(">> "); Serial.println(cmd);
while (Serial2.available()) (void)Serial2.read();
Serial2.println(cmd);
String resp; uint32_t t0 = millis();
while (millis() - t0 < timeoutMs) {
while (Serial2.available()) {
char c = (char)Serial2.read();
resp += c; Serial.print(c);
t0 = millis();
}
}
resp.trim();
Serial.println("\n[Resp]\n" + resp);
return resp;
}
}
namespace NetMgr {
String sendAT(const String& cmd, uint32_t timeoutMs) { return at(cmd, timeoutMs); }
void powerOnSIM7080(uint8_t pwrkeyPin) {
pinMode(pwrkeyPin, OUTPUT);
digitalWrite(pwrkeyPin, LOW); delay(2000);
digitalWrite(pwrkeyPin, HIGH); delay(1000);
digitalWrite(pwrkeyPin, LOW); delay(30000);
}
bool waitLTE(uint32_t ms) {
uint32_t t0 = millis();
while (millis() - t0 < ms) {
String r = at("AT+CEREG?", 4000);
if (r.indexOf("+CEREG: 0,1") != -1 || r.indexOf("+CEREG: 0,5") != -1) return true;
delay(3000);
}
return false;
}
void attachAndPdp(const char* apn) {
at("AT", 2000);
at("AT+CMEE=1", 2000);
at("AT+CPIN?", 2000);
at("AT+CSQ", 2000);
at("AT+CNMP=38", 5000);
at("AT+CMNB=1", 5000);
at("AT+CGDCONT=1,\"IP\",\"" + String(apn) + "\"", 5000);
at("AT+CGATT=1", 10000);
if (!waitLTE()) { Serial.println("No LTE reg"); while (true) delay(1000); }
at("AT+CNCFG=0,1,\"" + String(apn) + "\"", 5000);
at("AT+CNACT=0,1", 45000); // may 767 if already active
at("AT+CNACT?", 3000);
at("AT+CDNSCFG=\"8.8.8.8\",\"1.1.1.1\"", 3000);
}
} // namespace NetMgr

View File

@ -0,0 +1,9 @@
#pragma once
#include <Arduino.h>
namespace NetMgr {
void powerOnSIM7080(uint8_t pwrkeyPin);
bool waitLTE(uint32_t ms = 120000);
void attachAndPdp(const char* apn);
String sendAT(const String& cmd, uint32_t timeoutMs);
}

View File

@ -0,0 +1,89 @@
#include "SmsMgr.hpp"
#include "NetMgr.hpp"
namespace {
String readAndDelete(int idx) {
String r = NetMgr::sendAT("AT+CMGR=" + String(idx), 4000);
int okPos = r.lastIndexOf("\nOK");
String s = (okPos>0) ? r.substring(0, okPos) : r;
int lastLf = s.lastIndexOf('\n');
if (lastLf != -1) s = s.substring(lastLf + 1);
s.trim();
NetMgr::sendAT("AT+CMGD=" + String(idx), 2000);
return s;
}
void parseSimpleJson(const String& msg) {
auto extract = [&](const String& key)->String {
String pat = "\"" + key + "\":";
int k = msg.indexOf(pat);
if (k < 0) return "";
k += pat.length();
while (k < (int)msg.length() && (msg[k]==' '||msg[k]=='\"')) k++;
if (k>0 && msg[k-1]=='\"') {
int e = msg.indexOf('\"', k);
return (e>k) ? msg.substring(k, e) : "";
} else {
int e=k; while (e<(int)msg.length() && (isDigit(msg[e])||msg[e]=='.'||msg[e]=='-')) e++;
return msg.substring(k, e);
}
};
String deviceId = extract("device_id");
String speed = extract("speed");
Serial.println("SMS JSON parsed: device_id=" + deviceId + " speed=" + speed);
}
} // anon
namespace SmsMgr {
void setup() {
NetMgr::sendAT("AT+CMGF=1", 2000);
NetMgr::sendAT("AT+CPMS=\"SM\",\"SM\",\"SM\"", 2000);
NetMgr::sendAT("AT+CNMI=2,1,0,0,0", 2000);
}
void handleBody(const String& body) {
Serial.println("SMS body: " + body);
if (body.startsWith("{")) parseSimpleJson(body);
// else map plain-text commands here
}
void pollUrc() {
static String urc;
while (Serial2.available()) {
char c = (char)Serial2.read();
urc += c;
if (c=='\n') {
urc.trim();
if (urc.startsWith("+CMTI:")) {
int comma = urc.lastIndexOf(',');
if (comma != -1) {
int idx = urc.substring(comma+1).toInt();
String body = readAndDelete(idx);
handleBody(body);
}
}
urc = "";
}
}
}
void pollUnread() {
String list = NetMgr::sendAT("AT+CMGL=\"REC UNREAD\"", 3000);
int pos = 0;
while (true) {
int p = list.indexOf("+CMGL:", pos);
if (p == -1) break;
int comma = list.indexOf(',', p);
int idxStart = list.indexOf(' ', p);
if (comma==-1 || idxStart==-1 || idxStart>comma) break;
int idx = list.substring(idxStart+1, comma).toInt();
String body = readAndDelete(idx);
handleBody(body);
pos = comma + 1;
}
}
} // namespace SmsMgr

View File

@ -0,0 +1,9 @@
#pragma once
#include <Arduino.h>
namespace SmsMgr {
void setup();
void pollUrc(); // handle +CMTI
void pollUnread(); // optional periodic safety poll
void handleBody(const String& body); // your command handler entry
}