muizenval

Observe mouse traps remotely
Log | Files | Refs

commit 3e07f277c2e1aa39dbc20d997e963660a41b3059
parent fe9df280727c571f78d01b63ccddccf6852f17b6
Author: NibbaNamedHassan <[email protected]>
Date:   Mon, 27 Jun 2022 14:43:26 +0200

Merge branch 'master' of https://github.com/friedelschoen/muizenval.tk

Diffstat:
M.vscode/arduino.json | 4++--
M.vscode/c_cpp_properties.json | 1+
M.vscode/settings.json | 3++-
D5g-client/5g-client.ino | 172-------------------------------------------------------------------------------
D5g-client/include/config.h | 36------------------------------------
Aclient/client.ino | 86+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Aclient/config.ino | 15+++++++++++++++
Aclient/include/config.h | 65+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Aclient/include/interface.h | 51+++++++++++++++++++++++++++++++++++++++++++++++++++
Aclient/include/led.h | 14++++++++++++++
Aclient/interface.ino | 152+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Aclient/led.ino | 19+++++++++++++++++++
R5g-client/include/modem.h -> dump/modem.h | 0
R5g-client/modem.ino -> dump/modem.ino | 0
Adump/old_client.ino | 175+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
R5g-client/include/remote.h -> dump/remote.h | 0
R5g-client/remote.ino -> dump/remote.ino | 0
Rremote.md -> dump/remote.md | 0
Aled-codes.txt | 8++++++++
Mremote.py | 68+++++++++++++++++++++++++-------------------------------------------
Aremote.txt | 7+++++++
Mtest-server.py | 42++++++++++++++++++++++--------------------
Dtest.txt | 2--
23 files changed, 644 insertions(+), 276 deletions(-)

diff --git a/.vscode/arduino.json b/.vscode/arduino.json @@ -1,6 +1,6 @@ { - "sketch": "5g-client/5g-client.ino", + "sketch": "client/client.ino", "board": "SODAQ:samd:sodaq_sara", - "port": "/dev/tty.usbmodem1D11101", + "port": "/dev/tty.usbmodem14201", "output": "build" } \ No newline at end of file diff --git a/.vscode/c_cpp_properties.json b/.vscode/c_cpp_properties.json @@ -23,6 +23,7 @@ "/Users/friedel/Library/Arduino15/packages/arduino/tools/CMSIS-Atmel/1.2.0/CMSIS/Device/ATMEL/", "/Users/friedel/Library/Arduino15/packages/SODAQ/hardware/samd/1.8.9/cores/arduino", "/Users/friedel/Library/Arduino15/packages/SODAQ/hardware/samd/1.8.9/variants/sodaq_sara", + "/Users/friedel/Documents/Arduino/libraries/FlashStorage/src", "/Users/friedel/Documents/Arduino/libraries/Arduino_JSON/src", "/Users/friedel/Documents/Arduino/libraries/Sodaq_LSM303AGR/src", "/Users/friedel/Library/Arduino15/packages/SODAQ/hardware/samd/1.8.9/libraries/Wire", diff --git a/.vscode/settings.json b/.vscode/settings.json @@ -5,6 +5,7 @@ ".clangd": "yaml", ".yabairc": "shellscript", "v.mod": "plaintext", - "limits": "cpp" + "limits": "cpp", + "typeinfo": "cpp" }, } \ No newline at end of file diff --git a/5g-client/5g-client.ino b/5g-client/5g-client.ino @@ -1,172 +0,0 @@ -#include "include/config.h" -#include "include/modem.h" -#include "include/remote.h" - -#include <Sodaq_LSM303AGR.h> -#include <Sodaq_UBlox_GPS.h> -#include <Wire.h> - -#define ADC_AREF 3.3f -#define BATVOLT_R1 4.7f -#define BATVOLT_R2 10.0f -#define BATVOLT_PIN BAT_VOLT - -#define statusDelay 0.5 // seconds - -#define batteryFactor (0.978 * (BATVOLT_R1 / BATVOLT_R2 + 1) / ADC_AREF) - -sara_modem modem; -Sodaq_LSM303AGR accel; -serial_remote remote; - -void setup() { - // -*- hardware initiation -*- - - usbSerial.begin(remoteBaud); - while (usbWait && !usbSerial) - ; - - pinMode(BATVOLT_PIN, INPUT); - pinMode(CHARGER_STATUS, INPUT); - - // modem.init(); - remote.begin(); - - Wire.begin(); - delay(1000); - sodaq_gps.init(GPS_ENABLE); - - accel.rebootAccelerometer(); - delay(1000); - - // Enable the Accelerometer - accel.enableAccelerometer(); - - remote.connect("127.0.0.1", 5000); - - // modem.send("ATE0"); // disable command-echo - - // if (modem.send("AT+CPIN=\"" SIM_PIN "\"") == sara_modem::COMMAND_ERROR) { - // // usbSerial.println("[EROR] sim can't be unlocked, wrong PIN"); - // return; - // } - // usbSerial.println(prefixInfo "sim successful unlocked"); - /* - - modem.send("AT+CPSMS=0"); // Disable Power Saving Mode - modem.send("AT+CEDRXS=0"); // Disable eDRX - // usbSerial.println(prefixInfo "disabled power safe"); - - - // -*- internet initialization -*- - char info[100]; - - modem.send("AT+CFUN=15", sara_modem::COMMAND_BLOCK); // Reset the module - modem.send("AT+UMNOPROF=1", sara_modem::COMMAND_BLOCK); // Set MNO profile (1=automatic,100=standard europe) - modem.send("AT+URAT?", info); - // usbSerial.print(prefixInfo "urat: "); - // usbSerial.println(info); - modem.send("AT+URAT=8", sara_modem::COMMAND_IGNORE); // Set URAT to LTE-M/NB-IOT - modem.send("AT+CEREG=3", sara_modem::COMMAND_IGNORE); // Enable URCs - modem.send("AT+CGDCONT=1,\"IP\",\"" simAPN "\"", sara_modem::COMMAND_BLOCK); // Set the APN - modem.send("AT+CFUN=1"); // enable radio - modem.send("AT+COPS=0,2", sara_modem::COMMAND_BLOCK); // Autoselect the operator - - // usbSerial.print(prefixInfo "waiting for connection"); - - char response[100]; - - // Check Siganl strenght, repeat till you have a valid CSQ (99,99 means no signal) - while (modem.send("AT+CSQ", response, sara_modem::COMMAND_SILENT) == sara_modem::COMMAND_OK && !strcmp(response, "+CSQ: 99,99")) { - delay(1000); - // usbSerial.print("."); - } - - // Wait for attach, 1 = attached - while (modem.send("AT+CGATT?", response, sara_modem::COMMAND_SILENT) == sara_modem::COMMAND_OK && strcmp(response, "+CGATT: 1")) { - delay(1000); - // usbSerial.print("."); - } - // usbSerial.println(); - - // usbSerial.println(prefixInfo "connected!"); - - // -*- server connection -*- - - - //AT+UHTTP=0,0,"86.92.67.21" - //AT+UHTTP=0,5,80 - //AT+UHTTPC=0,5,"/api/search_connect","","TEST!",1 - - modem.send("AT+UHTTP=0,0,\"86.92.67.21\""); - modem.send("AT+UHTTP=0,5,80"); - modem.send("AT+UHTTPC=0,5,\"/api/search_connect\",\"\",\"TEST!\",1");*/ - - - // usbSerial.println(prefixInfo "initiation completed, starting remote:"); -} - - -void loop() { - /* // -*- remote for custom commands -*- - while (usbSerial.available()) - modemSerial.write(usbSerial.read()); - - while (modemSerial.available()) - usbSerial.write(modemSerial.read()); - - char buffer[512]; - gps.read(buffer); - - if (buffer[0] != '\0') { - // usbSerial.print("gps | "); - // usbSerial.println(buffer); - }*/ - - static int last = 0; - int now = millis(); - - static double lat = 0, lon = 0, accuracy = 0; - - if (now - last > statusDelay * 1000) { - if (sodaq_gps.scan(true, 10000)) { - lat = sodaq_gps.getLat(); - lon = sodaq_gps.getLon(); - accuracy = 1.0 / sodaq_gps.getHDOP() * 100; - // -> 100% the best, 0% the worst - // usbSerial.print(sodaq_gps.getLat(), 13); - // usbSerial.print(" - "); - // usbSerial.print(sodaq_gps.getLon(), 13); - // usbSerial.print(" ~ accuracy "); - // usbSerial.print(1.0 / sodaq_gps.getHDOP() * 100, 1); - // usbSerial.println("%"); - } - - serial_remote::http_packet req, res; - req.method = "POST"; - req.endpoint = "/api/update"; - req.body["charging"] = (bool) digitalRead(CHARGER_STATUS); - req.body["latitude"] = lat; - req.body["longitude"] = lon; - req.body["accuracy"] = accuracy; - req.body["battery"] = batteryVoltage(); - req.body["temperature"] = temperature(); - - remote.send(req); - last = now; - } - - // usbSerial.print(batteryVoltage()); - // usbSerial.println("V battery"); - - // usbSerial.print(temperature()); - // usbSerial.println(" deg celsius"); -} - -int temperature() { - return accel.getTemperature(); -} - -int batteryVoltage() { - return batteryFactor * (float) analogRead(BATVOLT_PIN); -} diff --git a/5g-client/include/config.h b/5g-client/include/config.h @@ -1,36 +0,0 @@ -#pragma once - -// -*- hardware stuff -*- -#define usbSerial SerialUSB // usb serial port -#define modemSerial Serial1 // modem serial port -#define modemPowerPin SARA_ENABLE // modem power pin -#define modemEnablePin SARA_TX_ENABLE // modem enable pin -#define modemVoltagePin SARA_R4XX_TOGGLE // modem voltage pin - -// -*- behaviour settings -*- -#define remoteBaud 115200 // baud-rate of usb-serial -#define modemBaud 115200 // baud-rate of modem-serial -#define remoteFirstTimeout 5.0 // seconds to wait for the first timeout -#define remoteTimeout 1.0 // seconds to wait for remote to timeout -#define lineBuffer 512 // buffer-size (bytes) to use to store lines -#define commandTimeout 10.0 // seconds to cancel a command -#define commandDelay 0.1 // delay after every command -#define ignoreDelay 2.0 // seconds to wait if command is run with COMMAND_IGNORE -#define commandDebug true // send debug information about command requests -#define eventDebug true // print '+'-events -#define lineDebug false // print each line to debug -#define blockDebug true // print if command is blocking -#define usbWait true // wait for a usb-connection - -// -*- sim settings -*- -#define macAddress "CAFEBABE01234567" // the boards mac-address -#define simPin "0000" // PIN of the sim -#define simAPN "lpwa.vodafone.iot" // APN-network of the sim - -// -*- prefixes -*- -#define prefixInfo "info | " -#define prefixDebug "debug | " -#define prefixError "error | " -#define prefixLine "line | " -#define prefixWarn "warn | " -#define prefixEvent "event | " diff --git a/client/client.ino b/client/client.ino @@ -0,0 +1,85 @@ +#include "include/config.h" +#include "include/interface.h" +#include "include/led.h" + +#include <Sodaq_LSM303AGR.h> +#include <Sodaq_UBlox_GPS.h> +#include <Wire.h> + +interface client; +Sodaq_LSM303AGR accel; + +void (*reset)() = 0; + +void setup() { + pinMode(LED_RED, OUTPUT); + pinMode(LED_GREEN, OUTPUT); + pinMode(LED_BLUE, OUTPUT); + pinMode(trapPin, INPUT_PULLUP); + pinMode(BATVOLT_PIN, INPUT); + pinMode(CHARGER_STATUS, INPUT); + + config_current = config_flash.read(); + + client.begin(); + + json req; + // req["mac"] = macAddress; + client.send(interface::METHOD_POST, "/api/hello", req); + + Wire.begin(); + delay(1000); + sodaq_gps.init(GPS_ENABLE); + + accel.rebootAccelerometer(); + delay(1000); + + // Enable the Accelerometer + accel.enableAccelerometer(); +} + +void loop() { + static int last = 0; + int now = millis(); + + if (now - last > statusInterval * 1000) { + json gps; + if (sodaq_gps.scan(true, gpsTimeout * 1000)) { + gps["signal"] = true; + gps["latitude"] = sodaq_gps.getLat(); + gps["longitude"] = sodaq_gps.getLon(); + gps["accuracy"] = getAccuracy(); // -> 100% the best, 0% the worst + } else { + gps["signal"] = false; + } + + gps["satellites"] = sodaq_gps.getNumberOfSatellites(); + + json req; + req["battery"] = batteryVoltage(); + req["temperature"] = accel.getTemperature(); + req["charging"] = getCharging(); + req["trap"] = getTrapStatus(); + req["gps"] = gps; + + client.send(interface::METHOD_POST, "/api/update", req); + last = now; + } +} + +double batteryVoltage() { + return batteryFactor * (double) analogRead(BATVOLT_PIN); +} + +double getAccuracy() { + double hdop = sodaq_gps.getHDOP(); + return hdop > 1 ? 1.0 / hdop * 100 : hdop * 100; +} + +bool getTrapStatus() { + return digitalRead(trapPin); +} + +bool getCharging() { + return digitalRead(CHARGER_STATUS); +} +\ No newline at end of file diff --git a/client/config.ino b/client/config.ino @@ -0,0 +1,14 @@ +#include "include/config.h" + +FlashStorage(config_flash, configuration); + +configuration default_config{ + /*.valid =*/true, + /*.simPIN =*/"", + /*.simPUK =*/"", + /*.simAPN =*/"", + /*.domain =*/"muizenval.tk", + /*.userToken =*/"" +}; + +configuration config_current; +\ No newline at end of file diff --git a/client/include/config.h b/client/include/config.h @@ -0,0 +1,65 @@ +#pragma once + +#include <FlashStorage.h> + +// -*- hardware stuff -*- +#define usbSerial SerialUSB // usb serial port +#define modemSerial Serial1 // modem serial port +#define modemPowerPin SARA_ENABLE // modem power pin +#define modemEnablePin SARA_TX_ENABLE // modem enable pin +#define modemVoltagePin SARA_R4XX_TOGGLE // modem voltage pin +#define trapPin 10 // pin of magnet-sensor + +// -*- behaviour settings -*- +#define remoteBaud 115200 // baud-rate of usb-serial +#define modemBaud 115200 // baud-rate of modem-serial +#define remoteForce true // do not try connect to modem +#define remoteFirstTimeout 5.0 // seconds to wait for the first timeout +#define remoteTimeout 1.0 // seconds to wait for remote to timeout +#define lineBuffer 512 // buffer-size (bytes) to use to store lines +#define commandTimeout 10.0 // seconds to cancel a command +#define commandDelay 0.1 // delay after every command +#define ignoreDelay 2.0 // seconds to wait if command is run with COMMAND_IGNORE +#define commandDebug true // send debug information about command requests +#define eventDebug true // print '+'-events +#define lineDebug false // print each line to debug +#define blockDebug true // print if command is blocking +#define blinkInterval 0.25 // seconds to wait for blink +#define gpsTimeout 5 // seconds to gps-timeout +#define statusInterval 5 // send status every n seconds + +#define ADC_AREF 3.3f +#define BATVOLT_R1 4.7f +#define BATVOLT_R2 10.0f +#define BATVOLT_PIN BAT_VOLT +#define batteryFactor (0.978 * (BATVOLT_R1 / BATVOLT_R2 + 1) / ADC_AREF) + +// -*- sim settings -*- +//#define simPin "0000" // PIN of the sim +//#define simAPN "lpwa.vodafone.iot" // APN-network of the sim +//#define apiHostname "muizenval.tk" + +// -*- prefixes -*- +#define prefixInfo "info | " +#define prefixDebug "debug | " +#define prefixError "error | " +#define prefixLine "line | " +#define prefixWarn "warn | " +#define prefixEvent "event | " + + +struct configuration { + bool valid; + + char simPIN[4]; + char simPUK[8]; + char simAPN[50]; + char domain[50]; + + char userToken[16]; +}; + +extern FlashStorageClass<configuration> config_flash; + +extern configuration config_default; +extern configuration config_current; diff --git a/client/include/interface.h b/client/include/interface.h @@ -0,0 +1,50 @@ +#pragma once + +#include <Arduino_JSON.h> + + +typedef JSONVar json; + +extern json null_response; + +struct interface { + enum method { + METHOD_GET, + METHOD_POST + }; + + enum command_status { + STATUS_OK = 0, // command succeed + STATUS_ERROR = 1, // command returned an error + STATUS_TIMEOUT = 2, // command timed out + STATUS_NOT_READY = 3, // interface is not ready + }; + + enum command_flags { + COMMAND_NONE, // none of them underneath + COMMAND_FORCE = 1 << 0, // ignore modemReady/remoteReady + COMMAND_BLOCK = 1 << 1, // no time-out (for waiting commands) + COMMAND_IGNORE = 1 << 2, // don't wait for response, just wait $ignoreDelay secounds + }; + + protected: + bool remoteReady = false; + bool modemReady = false; + + char debugToken[16]; + + void beginModem(); + + void beginRemote(); + void endRemote(); + + public: + void begin(); + + int send(method method, const char* endpoint, json body = nullptr, json& response = null_response); + + command_status remote(const char* command, json params = nullptr, json& response = null_response, command_flags flags = COMMAND_NONE); + + command_status modem(const char* request, char* response, command_flags flags = COMMAND_NONE); + command_status modem(const char* request, command_flags flags = COMMAND_NONE); +}; +\ No newline at end of file diff --git a/client/include/led.h b/client/include/led.h @@ -0,0 +1,14 @@ +#pragma once + +enum color { + COLOR_NONE, + COLOR_RED, + COLOR_GREEN, + COLOR_BLUE, + COLOR_YELLOW, + COLOR_CYAN, + COLOR_MAGENTA, + COLOR_WHITE, +}; + +void writeLED(color c); diff --git a/client/interface.ino b/client/interface.ino @@ -0,0 +1,152 @@ +#include "include/config.h" +#include "include/interface.h" +#include "include/led.h" + +json null_response; + +static const char* method_strings[] = { + [interface::METHOD_GET] = "GET", + [interface::METHOD_POST] = "POST", +}; + +void interface::begin() { + writeLED(COLOR_MAGENTA); + + usbSerial.begin(remoteBaud); + usbSerial.setTimeout(remoteTimeout); + + if (remoteForce) + return; + + pinMode(modemPowerPin, OUTPUT); // Put voltage on the nb-iot module + pinMode(modemVoltagePin, OUTPUT); // Switch module voltage + pinMode(modemEnablePin, OUTPUT); // Set state to active + + digitalWrite(modemPowerPin, HIGH); + digitalWrite(modemVoltagePin, LOW); + digitalWrite(modemEnablePin, HIGH); + modemSerial.begin(modemBaud); + + while (!modemSerial) + ; + + // -*- module initialization -*- + for (;;) { + modemSerial.write("AT\r\n"); + delay(1000); + if (modemSerial.available()) + break; + } + + while (modemSerial.available()) // clear cache + modemSerial.read(); + + // commands... + + modemReady = true; + + writeLED(COLOR_BLUE); +} + +void interface::beginRemote() { + if (remoteReady) // already initalizised + return; + + writeLED(COLOR_RED); + while (!usbSerial) + ; + writeLED(COLOR_YELLOW); + + remoteReady = true; +} + +void interface::endRemote() { + if (!remoteReady) + return; + + writeLED(COLOR_BLUE); + + json response; + remote("hello", nullptr, response, COMMAND_FORCE); + const char* debug = response["debugToken"]; + memcpy(debugToken, debug, sizeof(debugToken)); + + remoteReady = false; +} + +int interface::send(interface::method method, const char* endpoint, json body, json& response) { + int code; + + if (usbSerial || !modemReady) { + beginRemote(); + + json request; + + body["debugToken"] = debugToken; + + request["method"] = method_strings[method]; + request["endpoint"] = endpoint; + request["body"] = body; + + json cmd_response; + if (remote("send", request, cmd_response)) + return 0; + + response = cmd_response["body"]; + return cmd_response["code"]; + } else { + endRemote(); + // modem + } + return code; +} + +interface::command_status interface::remote(const char* command, json params, json& response, command_flags flags) { + bool force = flags & COMMAND_FORCE; + + if (!force && !remoteReady) + return interface::STATUS_NOT_READY; + + usbSerial.print(command); + usbSerial.print(" "); + params.printTo(usbSerial); + usbSerial.print("\n"); + + String status = usbSerial.readStringUntil(' '); + if (!status.length()) { + return interface::STATUS_TIMEOUT; + } else if (status == "ok") { + response = json::parse(usbSerial.readStringUntil('\n')); + return interface::STATUS_OK; + } else { + response = status; + return interface::STATUS_ERROR; + } +} + +interface::command_status interface::modem(const char* request, char* response, command_flags flags) { + char line[lineBuffer]; + size_t lineLen; + char buf; + + bool force = flags & COMMAND_FORCE, + block = flags & COMMAND_BLOCK, + ignore = flags & COMMAND_IGNORE; + + if (!force && !modemReady) + return interface::STATUS_NOT_READY; + + if (response) + response[0] = '\0'; + + modemSerial.write("AT"); + modemSerial.write(request); + modemSerial.write("\r\n"); + + + delay(commandDelay * 1000); +} + +interface::command_status interface::modem(const char* request, command_flags flags) { + return modem(request, NULL, flags); +} diff --git a/client/led.ino b/client/led.ino @@ -0,0 +1,18 @@ +#include "include/led.h" + +static const bool colors[][3] = { + [COLOR_NONE] = { 0, 0, 0 }, + [COLOR_RED] = { 1, 0, 0 }, + [COLOR_GREEN] = { 0, 1, 0 }, + [COLOR_BLUE] = { 0, 0, 1 }, + [COLOR_YELLOW] = { 1, 1, 0 }, + [COLOR_CYAN] = { 0, 1, 1 }, + [COLOR_MAGENTA] = { 1, 0, 1 }, + [COLOR_WHITE] = { 1, 1, 1 }, +}; + +void writeLED(color c) { + digitalWrite(LED_RED, !colors[c][0]); + digitalWrite(LED_GREEN, !colors[c][1]); + digitalWrite(LED_BLUE, !colors[c][2]); +} +\ No newline at end of file diff --git a/5g-client/include/modem.h b/dump/modem.h diff --git a/5g-client/modem.ino b/dump/modem.ino diff --git a/dump/old_client.ino b/dump/old_client.ino @@ -0,0 +1,175 @@ +#include "include/config.h" +#include "include/http.h" +#include "include/modem.h" +#include "include/remote.h" + +#include <Sodaq_LSM303AGR.h> +#include <Sodaq_UBlox_GPS.h> +#include <Wire.h> + +#define ADC_AREF 3.3f +#define BATVOLT_R1 4.7f +#define BATVOLT_R2 10.0f +#define BATVOLT_PIN BAT_VOLT + +#define statusDelay 0.5 // seconds + +#define batteryFactor (0.978 * (BATVOLT_R1 / BATVOLT_R2 + 1) / ADC_AREF) + +interface http; + +sara_modem modem; +Sodaq_LSM303AGR accel; +serial_remote remote; + +void setup() { + // -*- hardware initiation -*- + + usbSerial.begin(remoteBaud); + while (true && !usbSerial) + ; + + pinMode(BATVOLT_PIN, INPUT); + pinMode(CHARGER_STATUS, INPUT); + + // modem.init(); + remote.begin(); + + Wire.begin(); + delay(1000); + sodaq_gps.init(GPS_ENABLE); + + accel.rebootAccelerometer(); + delay(1000); + + // Enable the Accelerometer + accel.enableAccelerometer(); + + remote.connect("127.0.0.1", 5000); + + // modem.send("ATE0"); // disable command-echo + + // if (modem.send("AT+CPIN=\"" SIM_PIN "\"") == sara_modem::COMMAND_ERROR) { + // // usbSerial.println("[EROR] sim can't be unlocked, wrong PIN"); + // return; + // } + // usbSerial.println(prefixInfo "sim successful unlocked"); + /* + + modem.send("AT+CPSMS=0"); // Disable Power Saving Mode + modem.send("AT+CEDRXS=0"); // Disable eDRX + // usbSerial.println(prefixInfo "disabled power safe"); + + + // -*- internet initialization -*- + char info[100]; + + modem.send("AT+CFUN=15", sara_modem::COMMAND_BLOCK); // Reset the module + modem.send("AT+UMNOPROF=1", sara_modem::COMMAND_BLOCK); // Set MNO profile (1=automatic,100=standard europe) + modem.send("AT+URAT?", info); + // usbSerial.print(prefixInfo "urat: "); + // usbSerial.println(info); + modem.send("AT+URAT=8", sara_modem::COMMAND_IGNORE); // Set URAT to LTE-M/NB-IOT + modem.send("AT+CEREG=3", sara_modem::COMMAND_IGNORE); // Enable URCs + modem.send("AT+CGDCONT=1,\"IP\",\"" simAPN "\"", sara_modem::COMMAND_BLOCK); // Set the APN + modem.send("AT+CFUN=1"); // enable radio + modem.send("AT+COPS=0,2", sara_modem::COMMAND_BLOCK); // Autoselect the operator + + // usbSerial.print(prefixInfo "waiting for connection"); + + char response[100]; + + // Check Siganl strenght, repeat till you have a valid CSQ (99,99 means no signal) + while (modem.send("AT+CSQ", response, sara_modem::COMMAND_SILENT) == sara_modem::COMMAND_OK && !strcmp(response, "+CSQ: 99,99")) { + delay(1000); + // usbSerial.print("."); + } + + // Wait for attach, 1 = attached + while (modem.send("AT+CGATT?", response, sara_modem::COMMAND_SILENT) == sara_modem::COMMAND_OK && strcmp(response, "+CGATT: 1")) { + delay(1000); + // usbSerial.print("."); + } + // usbSerial.println(); + + // usbSerial.println(prefixInfo "connected!"); + + // -*- server connection -*- + + + //AT+UHTTP=0,0,"86.92.67.21" + //AT+UHTTP=0,5,80 + //AT+UHTTPC=0,5,"/api/search_connect","","TEST!",1 + + modem.send("AT+UHTTP=0,0,\"86.92.67.21\""); + modem.send("AT+UHTTP=0,5,80"); + modem.send("AT+UHTTPC=0,5,\"/api/search_connect\",\"\",\"TEST!\",1");*/ + + + // usbSerial.println(prefixInfo "initiation completed, starting remote:"); +} + + +void loop() { + /* // -*- remote for custom commands -*- + while (usbSerial.available()) + modemSerial.write(usbSerial.read()); + + while (modemSerial.available()) + usbSerial.write(modemSerial.read()); + + char buffer[512]; + gps.read(buffer); + + if (buffer[0] != '\0') { + // usbSerial.print("gps | "); + // usbSerial.println(buffer); + }*/ + + static int last = 0; + int now = millis(); + + static double lat = 0, lon = 0, accuracy = 0; + + if (now - last > statusDelay * 1000) { + if (sodaq_gps.scan(true, 10000)) { + lat = sodaq_gps.getLat(); + lon = sodaq_gps.getLon(); + accuracy = 1.0 / sodaq_gps.getHDOP() * 100; + // -> 100% the best, 0% the worst + // usbSerial.print(sodaq_gps.getLat(), 13); + // usbSerial.print(" - "); + // usbSerial.print(sodaq_gps.getLon(), 13); + // usbSerial.print(" ~ accuracy "); + // usbSerial.print(1.0 / sodaq_gps.getHDOP() * 100, 1); + // usbSerial.println("%"); + } + + serial_remote::http_packet req, res; + req.method = "POST"; + req.endpoint = "/api/update"; + req.body["charging"] = (bool) digitalRead(CHARGER_STATUS); + req.body["latitude"] = lat; + req.body["longitude"] = lon; + req.body["accuracy"] = accuracy; + req.body["battery"] = batteryVoltage(); + req.body["temperature"] = temperature(); + + remote.send(req); + last = now; + } + + // usbSerial.print(batteryVoltage()); + // usbSerial.println("V battery"); + + // usbSerial.print(temperature()); + // usbSerial.println(" deg celsius"); +} + +int temperature() { + return accel.getTemperature(); +} + +int batteryVoltage() { + return batteryFactor * (float) analogRead(BATVOLT_PIN); +} diff --git a/5g-client/include/remote.h b/dump/remote.h diff --git a/5g-client/remote.ino b/dump/remote.ino diff --git a/remote.md b/dump/remote.md diff --git a/led-codes.txt b/led-codes.txt @@ -0,0 +1,7 @@ +NONE = off +CYAN = starting up + +BLUE = idle modem +MAGENTA = idle remote +YELLOW = idle modem with warnings +RED = no interface +\ No newline at end of file diff --git a/remote.py b/remote.py @@ -1,55 +1,37 @@ from http.client import HTTPConnection import serial +import random import sys import json if len(sys.argv) < 2: print(f'{sys.argv[0]} <serial>') -server_address = '' -serial_port = serial.Serial(port=sys.argv[2], baudrate=115200) - -client = None - -def handle(req): - global client - - if 'command' not in req: - return 'command ommitted' - elif req['command'] == 'hello': - return None - elif req['command'] == 'connect': - client = HTTPConnection(req['host'], req['port']) - elif req['command'] == 'send': - headers = req['headers'] or {} - headers['Content-Type'] = 'application/json' - if client is None: - return 'not connected' - client.request(req['method'], req['endpoint'], json.dumps(req['body']), headers) - res = client.getresponse() - return { 'code': res.status, 'headers': dict(res.headers), 'body': json.load(res) } - else: - return 'unknown command' +server_address = 'localhost', 5000 +serial_port = serial.Serial(port=sys.argv[1], baudrate=115200) +debug_chars = '0123456789abcdefghijklmnopqrstuvwxyz' + +client = HTTPConnection(server_address[0], server_address[1]) + +debug_token = ''.join(random.choice(debug_chars) for _ in range(16)) while serial_port.is_open: try: - req = json.loads(serial_port.readline()) - res = handle(req) - except Exception as e: - req = '<error>' - res = { 'error': 'internal', 'description': str(e) } - print('-> ' + repr(req)) - - if type(res) == str: - res = { "error": res } - elif res is None: - res = { "error": None } - elif type(res) == dict: - if 'error' not in res: - res['error'] = None - else: - res = { "error": None, "value": res } - print('<- ' + repr(res)) - - serial_port.write((json.dumps(res) + '\n').encode()) + command, params_raw = serial_port.readline().decode().split(' ', 1) + params = json.loads(params_raw) + + if command == 'hello': + serial_port.write(f'ok {json.dumps(dict(debugToken=debug_token))}\n'.encode()) + elif command == 'send': + method, endpoint, body = params["method"], params["endpoint"], params["body"] + print(f'-> {method} {endpoint} {body}') + + client.request(method, endpoint, json.dumps(body)) + res = client.getresponse() + response = res.read().decode() + print(f'<- {res.status} {response}') + + serial_port.write(f'ok {json.dumps(dict(code=res.status, body=response))}\n'.encode()) + except: + serial_port.write(b'0 {}\n') diff --git a/remote.txt b/remote.txt @@ -0,0 +1,7 @@ +request: <command:string> <parameters:json> +response: <status:string> [parameters:json] + +status: +- ok +- not-ready +- not-connected diff --git a/test-server.py b/test-server.py @@ -1,29 +1,20 @@ -from typing import cast -from flask import Flask, request, jsonify +from flask import json +from werkzeug.exceptions import HTTPException +from flask import Flask, request app = Flask(__name__) -latitude = 0 -longitude = 0 -accuracy = 0 -battery = 0 -temperature = 0 - +status = {} @app.post("/api/update") def update(): - global latitude, longitude, accuracy, battery, temperature - - if request.is_json: - req = cast(dict, request.get_json()) - latitude = req['latitude'] - longitude = req['longitude'] - accuracy = req['accuracy'] - battery = req['battery'] - temperature = req['temperature'] - return {} + global status - return {"error": "request must be json"}, 415 + req = request.get_json(True) + if not req: + return + status = req + return {} @app.get("/") def index(): @@ -36,7 +27,7 @@ def index(): crossorigin=""></script> <h1>Status update</h1> - <p>latitude: <code>{latitude:.10f}</code></p> + <p>latitude: <code>{status['']:.10f}</code></p> <p>longitude: <code>{longitude:.10f}</code></p> <p>accuracy: <code>{accuracy:.2f}%</code></p> <p>battery: <code>{battery}V</code></p> @@ -54,4 +45,15 @@ def index(): </script> ''' [email protected](HTTPException) +def handle_exception(e): + response = e.get_response() + response.data = json.dumps({ + "code": e.code, + "name": e.name, + "description": e.description, + }) + response.content_type = "application/json" + return response + app.run('0.0.0.0', 5000) diff --git a/test.txt b/test.txt @@ -1 +0,0 @@ -hallo! -\ No newline at end of file