diff --git a/.clang-format b/.clang-format index 8ba7304c4686d7bf03150814f7a17554ba424175..7abb256cdd1d5fc9f2a79f19dfa70d6f55c6fe63 100644 --- a/.clang-format +++ b/.clang-format @@ -9,6 +9,7 @@ AllowShortFunctionsOnASingleLine: Inline AlwaysBreakTemplateDeclarations: Yes BreakBeforeBinaryOperators: NonAssignment BreakBeforeBraces: Custom +NamespaceIndentation: All BraceWrapping: AfterFunction: false ... diff --git a/host/host_central_mast/include/MessageType.h b/host/host_central_mast/include/MessageType.h new file mode 100644 index 0000000000000000000000000000000000000000..787afad32ef6fba58a5e7519ca06dff97b827988 --- /dev/null +++ b/host/host_central_mast/include/MessageType.h @@ -0,0 +1,16 @@ +// +// Created by zoe on 1/24/23. +// + +#ifndef HOST_CENTRAL_MAST_MESSAGETYPE_H +#define HOST_CENTRAL_MAST_MESSAGETYPE_H + +enum MessageType { dataAck, hostChange }; + +typedef struct response { + MessageType type; + uint8_t mac[6]; + long time; // Clang-Tidy: Narrowing conversion from 'unsigned long' to signed type 'long' is implementation-defined + // (see rtc in main.cpp) +} response; +#endif // HOST_CENTRAL_MAST_MESSAGETYPE_H diff --git a/host/host_central_mast/lib/TimeManager/NTPException.hpp b/host/host_central_mast/lib/NTPManager/NTPException.hpp similarity index 100% rename from host/host_central_mast/lib/TimeManager/NTPException.hpp rename to host/host_central_mast/lib/NTPManager/NTPException.hpp diff --git a/host/host_central_mast/lib/TimeManager/TimeManager.cpp b/host/host_central_mast/lib/NTPManager/NTPManager.cpp similarity index 65% rename from host/host_central_mast/lib/TimeManager/TimeManager.cpp rename to host/host_central_mast/lib/NTPManager/NTPManager.cpp index 4d5c5b5fdf75b8616b4dcb56a65c63f4689ffc59..f99ff9e7c17706d16cb9ab1a33debc07f283df1d 100644 --- a/host/host_central_mast/lib/TimeManager/TimeManager.cpp +++ b/host/host_central_mast/lib/NTPManager/NTPManager.cpp @@ -2,18 +2,18 @@ // Created by zoe on 12/19/22. // -#include "TimeManager.h" +#include "NTPManager.h" -TimeManager::TimeManager(TinyGsmSim7000 &modem) : modem(modem) {} +NTPManager::NTPManager(TinyGsmSim7000 &modem) : modem(modem) {} -void TimeManager::syncNTP(const std::string &ntpServer, const std::string &backupNtpServer) { +void NTPManager::syncNTP(const std::string &ntpServer, const std::string &backupNtpServer) { // create list of ntp servers std::vector<std::string> ntpServers; ntpServers.push_back(ntpServer); ntpServers.push_back(backupNtpServer); tryNtpServerSync(backupNtpServer, ntpServers); } -void TimeManager::tryNtpServerSync(const std::string &backupNtpServer, +void NTPManager::tryNtpServerSync(const std::string &backupNtpServer, std::vector<std::string> &ntpServers) { // try to sync ntp with each server for (const auto &server : ntpServers) { try { @@ -21,15 +21,16 @@ void TimeManager::tryNtpServerSync(const std::string &backupNtpServer, return; } catch (NTPException &e) { if (server == backupNtpServer) { - esp_log_write(ESP_LOG_ERROR, TAG, "Failed to sync NTP with %s (%s)\n", server.c_str(), e.what()); + esp_log_write(ESP_LOG_ERROR, TAG_TIMEMANAGER, "Failed to sync NTP with %s (%s)\n", server.c_str(), + e.what()); } - esp_log_write(ESP_LOG_ERROR, TAG, "Failed to sync NTP with %s (%s). Retrying with backup...\n", + esp_log_write(ESP_LOG_ERROR, TAG_TIMEMANAGER, "Failed to sync NTP with %s (%s). Retrying with backup...\n", server.c_str(), e.what()); } } } -void TimeManager::synchronizeNTPWithModem(const std::string &ntpServer) { - esp_log_write(ESP_LOG_DEBUG, TAG, "NTP Server Syncing...\n"); +void NTPManager::synchronizeNTPWithModem(const std::string &ntpServer) { + esp_log_write(ESP_LOG_DEBUG, TAG_TIMEMANAGER, "NTP Server Syncing...\n"); long error = modem.NTPServerSync(ntpServer.c_str(), timeZone); /** @@ -46,11 +47,11 @@ void TimeManager::synchronizeNTPWithModem(const std::string &ntpServer) { if (error != 1 && error != 255) { throw NTPException(modem.ShowNTPError(error).c_str()); } else { - esp_log_write(ESP_LOG_DEBUG, TAG, "NTP: Network time synchronization is successful\n"); + esp_log_write(ESP_LOG_DEBUG, TAG_TIMEMANAGER, "NTP: Network time synchronization is successful\n"); } } -time_t TimeManager::timeToUnixEpochSeconds(const std::string &time, const std::string &format) { +time_t NTPManager::timeToUnixEpochSeconds(const std::string &time, const std::string &format) { // 22/10/27,10:16:20+00 struct tm tm {}; time_t dateInEpoch = 0; @@ -77,11 +78,11 @@ time_t TimeManager::timeToUnixEpochSeconds(const std::string &time, const std::s return dateInEpoch; } -void TimeManager::writeModemTimeToRTC() { +void NTPManager::writeModemTimeToRTC() { auto gsmDateTimeString = modem.getGSMDateTime(DATE_FULL); - esp_log_write(ESP_LOG_DEBUG, TAG, "GSM DateTime: %s\n", gsmDateTimeString.c_str()); + esp_log_write(ESP_LOG_DEBUG, TAG_TIMEMANAGER, "GSM DateTime: %s\n", gsmDateTimeString.c_str()); time_t time = timeToUnixEpochSeconds(gsmDateTimeString.c_str()); rtc.setTime(time); - esp_log_write(ESP_LOG_INFO, TAG, "Time set to EPOCH: %s\n", String(rtc.getEpoch()).c_str()); + esp_log_write(ESP_LOG_INFO, TAG_TIMEMANAGER, "Time set to EPOCH: %s\n", String(rtc.getEpoch()).c_str()); } diff --git a/host/host_central_mast/lib/TimeManager/TimeManager.h b/host/host_central_mast/lib/NTPManager/NTPManager.h similarity index 78% rename from host/host_central_mast/lib/TimeManager/TimeManager.h rename to host/host_central_mast/lib/NTPManager/NTPManager.h index d970a292d388ae4b6cc0bce139d0c9196b3bd147..37a2a192bd13c63fec4e1627e962dca6d07067ce 100644 --- a/host/host_central_mast/lib/TimeManager/TimeManager.h +++ b/host/host_central_mast/lib/NTPManager/NTPManager.h @@ -2,11 +2,12 @@ // Created by zoe on 12/19/22. // -#ifndef HOST_CENTRAL_MAST_TIMEMANAGER_H -#define HOST_CENTRAL_MAST_TIMEMANAGER_H +#ifndef HOST_CENTRAL_MAST_NTPMANAGER_H +#define HOST_CENTRAL_MAST_NTPMANAGER_H #define TINY_GSM_MODEM_SIM7000 +#include "Definitions.h" #include "ESP32Time.h" #include "NTPException.hpp" #include "StringToTimeConversionException.hpp" @@ -14,13 +15,11 @@ #include <iomanip> #include <string> -class TimeManager { +class NTPManager { public: - explicit TimeManager(TinyGsmSim7000 &modem); + explicit NTPManager(TinyGsmSim7000 &modem); private: - constexpr static char *TAG = "TimeManager"; - ESP32Time rtc; int timeZone = 0; // I would like this to be a const reference but the functions used by the modem are not const @@ -40,4 +39,4 @@ class TimeManager { void writeModemTimeToRTC(); }; -#endif // HOST_CENTRAL_MAST_TIMEMANAGER_H \ No newline at end of file +#endif // HOST_CENTRAL_MAST_NTPMANAGER_H \ No newline at end of file diff --git a/host/host_central_mast/lib/TimeManager/StringToTimeConversionException.hpp b/host/host_central_mast/lib/NTPManager/StringToTimeConversionException.hpp similarity index 83% rename from host/host_central_mast/lib/TimeManager/StringToTimeConversionException.hpp rename to host/host_central_mast/lib/NTPManager/StringToTimeConversionException.hpp index 32e63f4de431138be07c1746bccce54c3f0a8757..f93acc337a2333fae8e3d8c493f7cc2b069d0a5c 100644 --- a/host/host_central_mast/lib/TimeManager/StringToTimeConversionException.hpp +++ b/host/host_central_mast/lib/NTPManager/StringToTimeConversionException.hpp @@ -5,16 +5,16 @@ #ifndef HOST_CENTRAL_MAST_STRINGTOTIMECONVERSIONEXCEPTION_HPP #define HOST_CENTRAL_MAST_STRINGTOTIMECONVERSIONEXCEPTION_HPP +#include "NTPManager.h" #include <exception> #include <string> -#include "TimeManager.h" class StringToTimeConversionException : public std::exception { public: explicit StringToTimeConversionException(const std::string &message) : message(message) {} const char *what() const noexcept override { - esp_log_write(ESP_LOG_ERROR, "TimeManager", "Error converting time to epoch: %s", + esp_log_write(ESP_LOG_ERROR, "NTPManager", "Error converting time to epoch: %s", message.c_str()); return message.c_str(); } diff --git a/host/host_central_mast/lib/NetworkConnectionManager/ConnectionManager.cpp b/host/host_central_mast/lib/NetworkConnectionManager/ConnectionManager.cpp index 29feb8c907b49bf79d48081e135be854dd8a2488..bdadaf9ee807376bef8b8450d30e54380bb3a49d 100644 --- a/host/host_central_mast/lib/NetworkConnectionManager/ConnectionManager.cpp +++ b/host/host_central_mast/lib/NetworkConnectionManager/ConnectionManager.cpp @@ -3,10 +3,6 @@ // #include "ConnectionManager.h" -#include "LTEConnectionException.hpp" -#include "ModemFunctionalityLevel.h" - -#include <utility> ConnectionManager::ConnectionManager(TinyGsm &modem, GPRSCredentials credentials, ModemPins modemPins) : modem(modem), credentials(std::move(credentials)), modemPins(modemPins) {} @@ -14,17 +10,16 @@ ConnectionManager::ConnectionManager(TinyGsm &modem, GPRSCredentials credentials void ConnectionManager::restartModem() { // Restart takes quite some time // To skip it, call init() instead of restart() - esp_log_write(ESP_LOG_DEBUG, TAG_GSM.c_str(), "Restarting modem...\n"); + esp_log_write(ESP_LOG_DEBUG, TAG_GSM, "Restarting modem...\n"); if (!modem.restart()) { - esp_log_write(ESP_LOG_WARN, TAG_GSM.c_str(), - "Failed to restart modem, attempting to continue without restarting\n"); + esp_log_write(ESP_LOG_WARN, TAG_GSM, "Failed to restart modem, attempting to continue without restarting\n"); } } void ConnectionManager::initModem() { - esp_log_write(ESP_LOG_DEBUG, TAG_GSM.c_str(), "Initializing modem...\n"); + esp_log_write(ESP_LOG_DEBUG, TAG_GSM, "Initializing modem...\n"); if (!modem.init()) { - esp_log_write(ESP_LOG_WARN, TAG_GSM.c_str(), "Failed to initialize modem, attempting to continue...\n"); + esp_log_write(ESP_LOG_WARN, TAG_GSM, "Failed to initialize modem, attempting to continue...\n"); } setModemFunctionalityLevel(MINIMAL); } @@ -100,9 +95,9 @@ bool ConnectionManager::gprsConnect() { bool ConnectionManager::isNetworkConnected() { return modem.isNetworkConnected(); } -std::string ConnectionManager::connect(int port, RequestInformation requestInformation) { +std::string ConnectionManager::connect(RequestInformation requestInformation) { TinyGsmClient client{modem, 0}; - if (!client.connect(requestInformation.host.c_str(), port)) { + if (!client.connect(requestInformation.host.c_str(), requestInformation.port)) { throw LTEConnectionException("Failed to connect to host"); } @@ -115,7 +110,7 @@ std::string ConnectionManager::connect(int port, RequestInformation requestInfor while (client.connected()) { line += client.readStringUntil('\n'); if (line == "\r") { - esp_log_write(ESP_LOG_DEBUG, TAG_GSM.c_str(), "headers received\n"); + esp_log_write(ESP_LOG_DEBUG, TAG_GSM, "headers received\n"); break; } } @@ -131,9 +126,11 @@ void ConnectionManager::modemPowerOn() { } void ConnectionManager::modemPowerOff() { + // Try to power-off (modem may decide to restart automatically) + // To turn off modem completely, please use Reset/Enable pins modem.sendAT("+CPOWD=1"); if (modem.waitResponse(10000L) != 1) { - esp_log_write(ESP_LOG_WARN, TAG_GSM.c_str(), "Failed to power off modem\n"); + esp_log_write(ESP_LOG_WARN, TAG_GSM, "Failed to power off modem\n"); } modem.poweroff(); pinMode(modemPins.pwr, OUTPUT); @@ -157,3 +154,28 @@ bool ConnectionManager::waitForNetwork() { bool ConnectionManager::gprsDisconnect() { return modem.gprsDisconnect(); } +void ConnectionManager::logModemInformation() { + bool res = modem.isGprsConnected(); + esp_log_write(ESP_LOG_DEBUG, TAG_GSM, "GPRS connected: %s\n", res ? "true" : "false"); + + String ccid = modem.getSimCCID(); + esp_log_write(ESP_LOG_DEBUG, TAG_GSM, "CCID: %s\n", ccid.c_str()); + + String imei = modem.getIMEI(); + esp_log_write(ESP_LOG_DEBUG, TAG_GSM, "IMEI: %s\n", imei.c_str()); + + String imsi = modem.getIMSI(); + esp_log_write(ESP_LOG_DEBUG, TAG_GSM, "IMSI: %s\n", imsi.c_str()); + + String cop = modem.getOperator(); + esp_log_write(ESP_LOG_DEBUG, TAG_GSM, "Operator: %s\n", cop.c_str()); + + IPAddress local = modem.localIP(); + esp_log_write(ESP_LOG_DEBUG, TAG_GSM, "Local IP: %s\n", local.toString().c_str()); + + int csq = modem.getSignalQuality(); + esp_log_write(ESP_LOG_DEBUG, TAG_GSM, "Signal quality: %d\n", csq); +} +bool ConnectionManager::isGprsConnected() { + return modem.isGprsConnected(); +} diff --git a/host/host_central_mast/lib/NetworkConnectionManager/ConnectionManager.h b/host/host_central_mast/lib/NetworkConnectionManager/ConnectionManager.h index 4f774d0dc2738115cf535dd8edb219ed18d86da0..70b4668a810fb0ceb0a858f4f4d52218201603c4 100644 --- a/host/host_central_mast/lib/NetworkConnectionManager/ConnectionManager.h +++ b/host/host_central_mast/lib/NetworkConnectionManager/ConnectionManager.h @@ -7,15 +7,16 @@ #define TINY_GSM_MODEM_SIM7000 +#include "Definitions.h" #include "GPRSCredentials.h" +#include "LTEConnectionException.hpp" #include "Mode.h" #include "ModemFunctionalityLevel.h" #include "NetworkMode.h" #include "RequestInformation.h" #include "StreamDebugger.h" #include <TinyGsmClient.h> - -static const std::string TAG_GSM = "GSM"; +#include <utility> class ConnectionManager { @@ -35,7 +36,7 @@ class ConnectionManager { public: ConnectionManager(TinyGsm &modem, GPRSCredentials credentials, ModemPins modemPins); - std::string connect(int port, RequestInformation requestInformation); + std::string connect(RequestInformation requestInformation); void enableGPS(); @@ -66,6 +67,9 @@ class ConnectionManager { void unlockSimCard(); bool waitForNetwork(); + + void logModemInformation(); + bool isGprsConnected(); }; #endif // HOST_CENTRAL_MAST_CONNECTIONMANAGER_H diff --git a/host/host_central_mast/lib/NetworkConnectionManager/RequestInformation.h b/host/host_central_mast/lib/NetworkConnectionManager/RequestInformation.h index 84843cfb43fc8feffa7381d381f0cde7976d40b4..5d416119edc8ec9ad36e153efadc564a8f347c60 100644 --- a/host/host_central_mast/lib/NetworkConnectionManager/RequestInformation.h +++ b/host/host_central_mast/lib/NetworkConnectionManager/RequestInformation.h @@ -6,31 +6,42 @@ #define HOST_CENTRAL_MAST_REQUESTINFORMATION_H #include <Arduino.h> +#include <Definitions.h> +#include <map> + +enum RequestMethod { + GET, + POST, +}; struct RequestInformation { - String method; + RequestMethod method; + std::map<String, String> headers; + int port; String host; String path; String body; - RequestInformation(String method, String host, String path, String body) { + RequestInformation(const RequestMethod &method, const String &host, const int &port, const String &path, + const String &body, const std::map<String, String> &headers) { this->method = method; + this->port = port; + this->headers = headers; this->host = host; this->path = path; this->body = body; } - // TODO: Move to configuration file - const String INFLUXDB_TOKEN = - "dUh2gbVLv7e3egqocxriDsJQNUacA9qZ5YXsYtdnVAglnHgy4nx-jDVO7nGlSF34BosfnuwnUDaviC7dQeC5RQ=="; - String buildRequest() { + String methodString = method == RequestMethod::GET ? "GET" : "POST"; String request = ""; - request += method + " " + path + " HTTP/1.1\r\n"; + request += methodString + " " + path + " HTTP/1.1\r\n"; request += "Host: " + host + "\r\n"; - request += "Authorization: Token " + INFLUXDB_TOKEN + "\r\n"; - request += "User-Agent: ESP32\r\n"; - request += "Content-Type: text/plain\r\n"; + request += "User-Agent: ESP32-Host\r\n"; + for (auto &header : headers) { + request += header.first + ": " + header.second + "\r\n"; + } + // request += "Authorization: Token " + INFLUXDB_TOKEN + "\r\n"; request += "Content-Length: " + String(body.length()) + "\r\n"; if (body.length() > 0) { request += "\r\n"; diff --git a/host/host_central_mast/lib/ResendManager/ResendManager.cpp b/host/host_central_mast/lib/ResendManager/ResendManager.cpp index 1c8eb125fc9721b8a816b221231682e2c1135711..64d6557ab417c4535f6ab7801f0ce12b566d0021 100644 --- a/host/host_central_mast/lib/ResendManager/ResendManager.cpp +++ b/host/host_central_mast/lib/ResendManager/ResendManager.cpp @@ -14,7 +14,7 @@ void ResendManager::init() { } uint ResendManager::getLastResendFileId() const { // get the next file id to be resend - auto filesInDirectory = getFilesInDirectory(resendDirectoryPath); + auto filesInDirectory = SDUtilities::getFilesInDirectory(resendDirectoryPath); // convert the file names to uint std::list<uint> fileUintIDs; @@ -38,24 +38,24 @@ uint ResendManager::getLastResendFileId() const { // get the next file id to be } void ResendManager::createResendDirectory() const { // create directory if it doesn't exist - createDirectory("/resend"); + SDUtilities::createDirectory("/resend"); } void ResendManager::storeForResend(const String &messageToBeSend) { // create file String filename = String(nextResendFileId); - writeFile(messageToBeSend, resendDirectoryPath + "/" + filename); + SDUtilities::writeFile(messageToBeSend, resendDirectoryPath + "/" + filename); ResendManager::incrementCount(); } std::optional<ResendPointType> ResendManager::loadNextToBeResendMessage() { // get first file in resend directory - auto filename = getFirstFileNameInDirectory(resendDirectoryPath.c_str()); + auto filename = SDUtilities::getFirstFileNameInDirectory(resendDirectoryPath.c_str()); if (filename.has_value()) { // read file - auto message = readFile(resendDirectoryPath + "/" + filename.value().c_str()); + auto message = SDUtilities::readFile(resendDirectoryPath + "/" + filename.value().c_str()); return ResendPointType{message, filename.value()}; } return std::nullopt; diff --git a/host/host_central_mast/lib/Utilities/Definitions.h b/host/host_central_mast/lib/Utilities/Definitions.h index ed28c034d527ae43c36652bfff858197a1e0594b..5f5cc45c375b37d12d8845de1fdf8f605cc936d2 100644 --- a/host/host_central_mast/lib/Utilities/Definitions.h +++ b/host/host_central_mast/lib/Utilities/Definitions.h @@ -6,6 +6,7 @@ #define HOST_CENTRAL_MAST_DEFINITIONS_H #include "Arduino.h" +#include "ESP32Time.h" #define uS_TO_S_FACTOR 1000000ULL // Conversion factor for micro seconds to seconds #define TIME_TO_SLEEP 5 // Time ESP32 will go to sleep (in seconds) @@ -22,6 +23,23 @@ #define SD_CS 13 #define LED_PIN 12 +// See all AT commands, if wanted +// #define DUMP_AT_COMMANDS + +#define TAG_ESPNOW "ESPNOW" +#define TAG_MAIN "MAIN" +#define TAG_SD "SD" +#define TAG_GSM "GSM" +#define TAG_TIMEMANAGER "TIMEMANAGER" +#define TAG_SDUTILITIES "SDUTILITIES" + +#define SD_CARD_FAIL_COUNTER 100 + +constexpr bool PRINT_TO_SERIAL = true; + +// global definition of RTC. Initialised in main +extern ESP32Time rtc; + const String INFLUXDB_TOKEN = "dUh2gbVLv7e3egqocxriDsJQNUacA9qZ5YXsYtdnVAglnHgy4nx-jDVO7nGlSF34BosfnuwnUDaviC7dQeC5RQ=="; diff --git a/host/host_central_mast/lib/Utilities/SDCardLogger.cpp b/host/host_central_mast/lib/Utilities/SDCardLogger.cpp new file mode 100644 index 0000000000000000000000000000000000000000..b7fe7c7dcd2ef8b075cd67faae9ae187f01922b0 --- /dev/null +++ b/host/host_central_mast/lib/Utilities/SDCardLogger.cpp @@ -0,0 +1,60 @@ +// +// Created by zoe on 1/24/23. +// + +#include "SDCardLogger.h" + +namespace SDCardLogger { + + char log_print_buffer[512]; + bool printToSerial = false; + + void printDebugToSerial(bool printToSerialAsWell) { + printToSerial = printToSerialAsWell; + } + int vprintf_into_sd(const char *szFormat, va_list args) { + String logstring = "[" + rtc.getDateTime() + "] "; + logstring += szFormat; + + if (!SDUtilities::isSDAvailable()) { + if (printToSerial) { + logstring += " (SD card not available)\n"; + vprintf(logstring.c_str(), args); + } + return 1; + } + + // write evaluated format string into buffer + int ret = vsnprintf(log_print_buffer, sizeof(log_print_buffer), logstring.c_str(), args); + + String date = rtc.getDate(); + String filename = "/log_" + date + ".txt"; + + // output is now in buffer. write to file. + if (ret >= 0) { + if (!SD.exists(filename)) { + File writeLog = SD.open(filename, FILE_WRITE); + if (!writeLog) { + Serial.println("Couldn't open " + filename + " for writing"); + SDUtilities::setSDAvailable(false); + return 1; + } + delay(50); + writeLog.close(); + } + + File logFile = SD.open(filename, FILE_APPEND); + + // debug output + if (printToSerial) { + vprintf(logstring.c_str(), args); + } + logFile.write((uint8_t *)log_print_buffer, (size_t)ret); + // to be safe in case of crashes: flush the output + logFile.flush(); + logFile.close(); + } + return ret; + } + +} // namespace SDCardLogger diff --git a/host/host_central_mast/lib/Utilities/SDCardLogger.h b/host/host_central_mast/lib/Utilities/SDCardLogger.h new file mode 100644 index 0000000000000000000000000000000000000000..c14a07dc0ae0e4579daeb1b41c775253ebd9ca25 --- /dev/null +++ b/host/host_central_mast/lib/Utilities/SDCardLogger.h @@ -0,0 +1,17 @@ +// +// Created by zoe on 1/24/23. +// + +#ifndef HOST_CENTRAL_MAST_SDCARDLOGGER_H +#define HOST_CENTRAL_MAST_SDCARDLOGGER_H + +#include "Definitions.h" +#include "SD.h" +#include "Utilities.h" + +namespace SDCardLogger { + void printDebugToSerial(bool printToSerialAsWell); + int vprintf_into_sd(const char *szFormat, va_list args); +}; // namespace SDCardLogger + +#endif // HOST_CENTRAL_MAST_SDCARDLOGGER_H diff --git a/host/host_central_mast/lib/Utilities/Utilities.cpp b/host/host_central_mast/lib/Utilities/Utilities.cpp index 21f3aca77fc6d9ceaeff1491862931f78cce6f39..316d33c95d27260f7b3c0ca299ed049db6305bce 100644 --- a/host/host_central_mast/lib/Utilities/Utilities.cpp +++ b/host/host_central_mast/lib/Utilities/Utilities.cpp @@ -4,148 +4,302 @@ #include "Utilities.h" -void setupSDCard(); -void saveStringToSDCard(const std::string &dataString); -std::list<String> getFilesInDirectory(const String &dirname) { - std::list<String> files; - File dir = openDirectory(dirname); - while (true) { - File nextFile = dir.openNextFile(); - if (!nextFile) { - break; - } - if (!nextFile.isDirectory()) { - files.emplace_back(nextFile.name()); - } - nextFile.close(); - } - return files; -} +namespace SDUtilities { + + bool SDAvailable = true; -std::optional<String> getLastFileInDirectory(const String &dirname) { + void setSDAvailable(bool sdAvailable) { + SDAvailable = sdAvailable; + } - File root = openDirectory(dirname); - root.rewindDirectory(); + // Used to try to remount SD card if it fails more than x times (100) + int sdFailCounter = 0; + + bool isSDAvailable() { + return SDAvailable; + } - File file = root.openNextFile(); - while (file) { - File nextFile = root.openNextFile(); - if (!nextFile) { - break; + void tryRemountingSD() { + try { + setupSDCard(SD_MISO, SD_MOSI, SD_SCLK, SD_CS); + setSDAvailable(true); + } catch (SDSetupException &e) { + esp_log_write(ESP_LOG_ERROR, TAG_SDUTILITIES, "Couldn't remount SD card: %s\n", e.what()); } - file = nextFile; + sdFailCounter = 0; } - // log - if (file) { - esp_log_write(ESP_LOG_INFO, "getLastFileInDirectory", "Last file name: %s\n", file.name()); - return file.name(); - } else { - esp_log_write(ESP_LOG_INFO, "getLastFileInDirectory", "No file found\n"); - return std::nullopt; + + bool checkSDAvailability(const String &errorMessage) { + if (!isSDAvailable()) { + sdFailCounter++; + if (sdFailCounter % SD_CARD_FAIL_COUNTER == 0) { + esp_log_write(ESP_LOG_ERROR, TAG_SDUTILITIES, "SD card not available. Trying to remount...\n"); + tryRemountingSD(); + } + String message = errorMessage + " (SD card not available, failed " + sdFailCounter + " times)\n"; + esp_log_write(ESP_LOG_ERROR, TAG_SDUTILITIES, "%s", message.c_str()); + // if re-mount was successful, this will be true + return isSDAvailable(); + } + return true; + }; + + std::list<String> getFilesInDirectory(const String &dirname) { + if (!checkSDAvailability("Couldn't get files in directory " + dirname)) { + return {}; + } + std::list<String> files; + File dir = openDirectory(dirname); + while (true) { + File nextFile = dir.openNextFile(); + if (!nextFile) { + break; + } + if (!nextFile.isDirectory()) { + files.emplace_back(nextFile.name()); + } + nextFile.close(); + } + return files; } -} -File openDirectory(const String &dirname) { - File root = SD.open(dirname); - if (!root) { - esp_log_write(ESP_LOG_ERROR, "openDirectory", "Failed to open directory\n"); - throw; + + std::optional<String> getLastFileInDirectory(const String &dirname) { + + if (!checkSDAvailability("Couldn't get last file in directory " + dirname)) { + return std::nullopt; + } + + File root = openDirectory(dirname); + root.rewindDirectory(); + + File file = root.openNextFile(); + while (file) { + File nextFile = root.openNextFile(); + if (!nextFile) { + break; + } + file = nextFile; + } + // log + if (file) { + esp_log_write(ESP_LOG_INFO, TAG_SDUTILITIES, "Last file name: %s\n", file.name()); + return file.name(); + } else { + esp_log_write(ESP_LOG_INFO, TAG_SDUTILITIES, "No file found\n"); + return std::nullopt; + } } - if (!root.isDirectory()) { - esp_log_write(ESP_LOG_ERROR, "openDirectory", "Not a directory\n"); - throw; + File openDirectory(const String &dirname) { + + if (!checkSDAvailability("Couldn't open directory " + dirname)) { + throw; + } + + File root = SD.open(dirname); + if (!root) { + esp_log_write(ESP_LOG_ERROR, TAG_SDUTILITIES, "Failed to open directory\n"); + throw; + } + if (!root.isDirectory()) { + esp_log_write(ESP_LOG_ERROR, TAG_SDUTILITIES, "Not a directory\n"); + throw; + } + return root; } - return root; -} -std::optional<String> getFirstFileNameInDirectory(const String &dirname) { - File root = openDirectory(dirname); - root.rewindDirectory(); + std::optional<String> getFirstFileNameInDirectory(const String &dirname) { - File file = root.openNextFile(); - if (file) { - esp_log_write(ESP_LOG_INFO, "getFirstFileNameInDirectory", "file found: %s\n", file.name()); - return file.name(); - } else { - esp_log_write(ESP_LOG_INFO, "getFirstFileNameInDirectory", "no file found\n"); - return std::nullopt; + if (!checkSDAvailability("Couldn't get first file in directory " + dirname)) { + return std::nullopt; + } + + File root = openDirectory(dirname); + root.rewindDirectory(); + + File file = root.openNextFile(); + if (file) { + esp_log_write(ESP_LOG_INFO, TAG_SDUTILITIES, "file found: %s\n", file.name()); + return file.name(); + } else { + esp_log_write(ESP_LOG_INFO, TAG_SDUTILITIES, "no file found\n"); + return std::nullopt; + } } -} -File openForWrite(const String &filePath) { - File file = SD.open(filePath, FILE_WRITE); - if (!file) { - esp_log_write(ESP_LOG_ERROR, "SD", "Failed to open file for writing\n"); - throw; + File openForWrite(const String &filePath) { + + if (!checkSDAvailability("Couldn't open file " + filePath + " for writing")) { + throw; + } + + File file = SD.open(filePath, FILE_WRITE); + if (!file) { + esp_log_write(ESP_LOG_ERROR, TAG_SDUTILITIES, "Failed to open file for writing\n"); + throw; + } + return file; } - return file; -} -File openForRead(const String &filePath) { - File file = SD.open(filePath, FILE_READ); - if (!file) { - esp_log_write(ESP_LOG_ERROR, "SD", "Failed to open file for reading\n"); - throw; + File openForRead(const String &filePath) { + + if (!checkSDAvailability("Couldn't open file " + filePath + " for reading")) { + throw; + } + + File file = SD.open(filePath, FILE_READ); + if (!file) { + esp_log_write(ESP_LOG_ERROR, TAG_SDUTILITIES, "Failed to open file for reading\n"); + throw; + } + return file; } - return file; -} -void writeFile(const String &messageToBeSend, const String &filePath) { - File file = openForWrite(filePath); - if (file.print(messageToBeSend)) { - esp_log_write(ESP_LOG_INFO, "SD", "File written %s\n", filePath.c_str()); - } else { - esp_log_write(ESP_LOG_ERROR, "SD", "Write failed %s\n", filePath.c_str()); - throw; + void writeFile(const String &messageToBeSend, const String &filePath) { + + if (!checkSDAvailability("Couldn't write to file " + filePath)) { + throw; + } + + File file = openForWrite(filePath); + if (file.print(messageToBeSend)) { + esp_log_write(ESP_LOG_INFO, TAG_SDUTILITIES, "File written %s\n", filePath.c_str()); + } else { + esp_log_write(ESP_LOG_ERROR, TAG_SDUTILITIES, "Write failed %s\n", filePath.c_str()); + throw; + } + file.close(); } - file.close(); -} -String readFile(const String &filePath) { - File file = openForRead(filePath); - String ret; + String readFile(const String &filePath) { + + if (!checkSDAvailability("Couldn't read file " + filePath)) { + throw; + } + + File file = openForRead(filePath); - while (file.available()) { - ret += (char)file.read(); + String ret; + + while (file.available()) { + ret += (char)file.read(); + } + + file.close(); + return ret; } - file.close(); - return ret; -} + void createDirectory(const String &dirname) { -void createDirectory(const String &dirname) { - if (!SD.exists(dirname)) { - SD.mkdir(dirname); - esp_log_write(ESP_LOG_INFO, "createDirectory", "Created directory: %s\n", dirname.c_str()); - } else { - esp_log_write(ESP_LOG_WARN, "createDirectory", "Directory already exists\n"); + if (!checkSDAvailability("Couldn't create directory " + dirname)) { + throw; + } + + if (!SD.exists(dirname)) { + SD.mkdir(dirname); + esp_log_write(ESP_LOG_INFO, TAG_SDUTILITIES, "Created directory: %s\n", dirname.c_str()); + } else { + esp_log_write(ESP_LOG_WARN, TAG_SDUTILITIES, "Directory already exists\n"); + } } -} -void setupSDCard(int MISO, int MOSI, int SCLK, int CS) { - SPI.begin(SCLK, MISO, MOSI, CS); - if (!SD.begin(CS)) { - esp_log_write(ESP_LOG_ERROR, "Utilities", "Card MOUNT FAIL\n"); - throw SDSetupException("Card MOUNT FAIL"); - } else { - uint32_t cardSize = SD.cardSize() / (1024 * 1024); - String sdcardSizeString = "SDCard Size: " + String(cardSize) + "MB"; - esp_log_write(ESP_LOG_DEBUG, "Utilities", "%s\n", sdcardSizeString.c_str()); + void setupSDCard(int MISO, int MOSI, int SCLK, int CS) { + SPI.begin(SCLK, MISO, MOSI, CS); + if (!SD.begin(CS)) { + esp_log_write(ESP_LOG_ERROR, TAG_SDUTILITIES, "Card MOUNT FAIL\n"); + throw SDSetupException("Card MOUNT FAIL"); + } else { + uint32_t cardSize = SD.cardSize() / (1024 * 1024); + String sdcardSizeString = "SDCard Size: " + String(cardSize) + "MB"; + esp_log_write(ESP_LOG_DEBUG, TAG_SDUTILITIES, "%s\n", sdcardSizeString.c_str()); + } } -} -void saveStringToSDCard(const std::string &dataString) { - File dataFile = SD.open("/datalog.txt", FILE_APPEND); + void saveStringToSDCard(const std::string &dataString) { + + if (!checkSDAvailability("Couldn't save string to SD card")) { + throw SDCardException("Couldn't save string to SD card"); + } - // if the file is available, write to it: - if (dataFile) { - if (dataString.length() > 0) { - dataFile.println(dataString.c_str()); + File dataFile = SD.open("/datalog.txt", FILE_APPEND); + + // if the file is available, write to it: + if (dataFile) { + if (dataString.length() > 0) { + dataFile.println(dataString.c_str()); + } + dataFile.close(); + } + // if the file isn't open, pop up an error: + else { + esp_log_write(ESP_LOG_ERROR, TAG_SDUTILITIES, "error opening datalog.txt\n"); + throw SDCardException("error opening datalog.txt"); } - dataFile.close(); } - // if the file isn't open, pop up an error: - else { - esp_log_write(ESP_LOG_ERROR, "Utilities", "error opening datalog.txt\n"); - throw SDCardException("error opening datalog.txt"); +} // namespace SDUtilities + +// I don't think this does anything. Copied from the example +void turnOffLEDs() { // Set LED OFF + pinMode(LED_PIN, OUTPUT); + digitalWrite(LED_PIN, HIGH); + + pinMode(PWR_PIN, OUTPUT); + digitalWrite(PWR_PIN, HIGH); + delay(300); + digitalWrite(PWR_PIN, LOW); +} + +String getMacAddressAsString(const uint8_t *mac) { + String macAddress; + for (int i = 0; i < 6; i++) { + macAddress += String(mac[i], HEX); + } + esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW, "MAC: %s\n", macAddress.c_str()); + return macAddress; +} + +String documentToLineProtocolString(const DynamicJsonDocument &doc) { + String measurementType = doc["measurementType"].as<String>(); + String sensorName = doc["sensorName"].as<String>(); + String timestamp = doc["timestamp"].as<String>(); + String protocol = doc["protocol"].as<String>(); + String value = doc["value"].as<String>(); + String channel = doc["channel"].as<String>(); + String clientMac = doc["clientMac"].as<String>(); + + String lineData = sensorName + ",clientMac=" + clientMac + ",protocol=" + protocol + ",channel=" + channel + " " + + measurementType + "=" + value + " " + timestamp; + return lineData; +} +DynamicJsonDocument parseReceivedJsonData(char *data) { + DynamicJsonDocument doc(250); + auto error = deserializeJson(doc, data); + if (error) { + esp_log_write(ESP_LOG_ERROR, TAG_ESPNOW, "Error while parsing json: %s\n", error.c_str()); + // TODO error handling } + return doc; +} + +String documentToServerReadableString(const DynamicJsonDocument &doc) { + StaticJsonDocument<1024> serverDoc; + String hostMacAddressString = WiFi.macAddress(); + hostMacAddressString.replace(":", ""); + hostMacAddressString.toLowerCase(); + + serverDoc["host"] = hostMacAddressString; + serverDoc["client"] = doc["clientMac"].as<String>(); + serverDoc["sensorProtocol"] = doc["protocol"].as<String>(); + serverDoc["protocolAddress"] = doc["channel"].as<String>(); + serverDoc["hardwareName"] = doc["sensorName"].as<String>(); + // each value is a element in the readigs array + JsonArray readings = serverDoc.createNestedArray("readings"); + JsonObject reading = readings.createNestedObject(); + reading["name"] = doc["measurementType"].as<String>(); + reading["value"] = doc["value"].as<float>(); + serverDoc["time"] = doc["timestamp"].as<uint32_t>(); + + String serverString; + serializeJson(serverDoc, serverString); + esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW, "Server readable data: %s\n", serverString.c_str()); + return serverString; } \ No newline at end of file diff --git a/host/host_central_mast/lib/Utilities/Utilities.h b/host/host_central_mast/lib/Utilities/Utilities.h index 1156c0289983da90d2735fea2786e1933ab91b54..dc18e3459e0fabe83c64f2e09a4a9ae3bf8c45f6 100644 --- a/host/host_central_mast/lib/Utilities/Utilities.h +++ b/host/host_central_mast/lib/Utilities/Utilities.h @@ -5,21 +5,34 @@ #ifndef HOST_CENTRAL_MAST_UTILITIES_H #define HOST_CENTRAL_MAST_UTILITIES_H +#include "ArduinoJson.h" #include "SD.h" #include "SDCardException.h" #include "SDSetupException.h" +#include "WiFi.h" #include <Arduino.h> #include <Definitions.h> #include <WString.h> #include <list> -File openDirectory(const String &dirname); -std::list<String> getFilesInDirectory(const String &dirname); -std::optional<String> getFirstFileNameInDirectory(const String &dirname); -std::optional<String> getLastFileInDirectory(const String &dirname); -void writeFile(const String &messageToBeSend, const String &filePath); -String readFile(const String &filePath); -void createDirectory(const String &dirname); -void setupSDCard(int MISO, int MOSI, int SCLK, int CS); -void saveStringToSDCard(const std::string &dataString); +namespace SDUtilities { + File openDirectory(const String &dirname); + std::list<String> getFilesInDirectory(const String &dirname); + std::optional<String> getFirstFileNameInDirectory(const String &dirname); + std::optional<String> getLastFileInDirectory(const String &dirname); + void writeFile(const String &messageToBeSend, const String &filePath); + String readFile(const String &filePath); + void createDirectory(const String &dirname); + void setupSDCard(int MISO, int MOSI, int SCLK, int CS); + void saveStringToSDCard(const std::string &dataString); + void setSDAvailable(bool sdAvailable); + bool isSDAvailable(); +} // namespace SDUtilities + +void turnOffLEDs(); +String getMacAddressAsString(const uint8_t *mac); +String documentToLineProtocolString(const DynamicJsonDocument &doc); +DynamicJsonDocument parseReceivedJsonData(char *data); +String documentToServerReadableString(const DynamicJsonDocument &doc); + #endif // HOST_CENTRAL_MAST_UTILITIES_H diff --git a/host/host_central_mast/src/main.cpp b/host/host_central_mast/src/main.cpp index 850beacb573f5cbca01def4d59d8400f2089349f..042d4e623cda2bc188513c93b6a1c60b2a0caeb2 100644 --- a/host/host_central_mast/src/main.cpp +++ b/host/host_central_mast/src/main.cpp @@ -1,45 +1,34 @@ -#include "FS.h" -#include "SD.h" +/*** + * code is modified from the original code found at + * https://github.com/Xinyuan-LilyGO/LilyGO-T-SIM7000G/blob/master/examples/Arduino_TinyGSM/AllFunctions/AllFunctions.ino + */ + +#define TINY_GSM_MODEM_SIM7000 + +#include "ConnectionManager.h" +#include "MessageType.h" +#include "SDCardLogger.h" #include "SPI.h" #include "Utilities.h" -#include "time.h" -#include <Arduino.h> #include <ArduinoJson.h> +#include <Definitions.h> #include <ESP32Time.h> +#include <NTPManager.h> +#include <ResendManager.h> +#include <TinyGsmClient.h> #include <WiFi.h> #include <esp_log.h> #include <esp_now.h> +#include <map> #include <queue> #include <sys/unistd.h> -static const std::string TAG = "MAIN"; -static const std::string TAG_ESPNOW = "ESPNOW"; -static const std::string TAG_GSM = "GSM"; - -/* - FILE: AllFunctions.ino - AUTHOR: Koby Hale - PURPOSE: Test functionality -*/ +ESP32Time rtc; -#define TINY_GSM_MODEM_SIM7000 #define TINY_GSM_RX_BUFFER 1024 // Set RX buffer to 1Kb #define SerialAT Serial1 -// See all AT commands, if wanted -#define DUMP_AT_COMMANDS - -// set GSM PIN, if any -#define GSM_PIN "" - -// Your GPRS credentials, if any -const char apn[] = "m2m.public.at"; // SET TO YOUR APN -const char gprsUser[] = ""; -const char gprsPass[] = ""; - -#include <TinyGsmClient.h> - -#ifdef DUMP_AT_COMMANDS // if enabled it requires the streamDebugger lib +#ifdef DUMP_AT_COMMANDS // if enabled it requires the streamDebugger library #include <StreamDebugger.h> StreamDebugger debugger(SerialAT, Serial); TinyGsm modem(debugger); @@ -47,75 +36,35 @@ TinyGsm modem(debugger); TinyGsm modem(SerialAT); #endif -#include "Definitions.h" +const GPRSCredentials gprsCredentials = GPRSCredentials("m2m.public.at", "", ""); + +ConnectionManager connectionManager{modem, gprsCredentials, + ConnectionManager::ModemPins{PIN_DTR, PIN_TX, PIN_RX, PWR_PIN}}; -enum MessageType { dataAck, hostChange }; -typedef struct response { - MessageType type; - uint8_t mac[6]; - long time; -} response; +NTPManager ntpManager{modem}; uint8_t BROADCAST_MAC[6] = {0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF}; esp_now_peer_info_t broadcast = {}; response announce = {}; -ESP32Time rtc; - SemaphoreHandle_t xMutex; TaskHandle_t ESPNOWTask; static std::queue<String> queue; +ResendManager resendManager; + void on_data_sent(const uint8_t *mac_addr, esp_now_send_status_t status) { // go to sleep } -static char log_print_buffer[512]; - -int vprintf_into_sd(const char *szFormat, va_list args) { - String logstring = "[" + rtc.getDateTime() + "] "; - logstring += szFormat; - // write evaluated format string into buffer - int ret = vsnprintf(log_print_buffer, sizeof(log_print_buffer), logstring.c_str(), args); - - String date = rtc.getDate(); - String filename = "/log_" + date + ".txt"; - - // output is now in buffer. write to file. - if (ret >= 0) { - if (!SD.exists(filename)) { - File writeLog = SD.open(filename, FILE_WRITE); - if (!writeLog) - Serial.println("Couldn't open " + filename + " for writing"); - delay(50); - writeLog.close(); - } - - File logFile = SD.open(filename, FILE_APPEND); - - // debug output - vprintf(logstring.c_str(), args); - logFile.write((uint8_t *)log_print_buffer, (size_t)ret); - // to be safe in case of crashes: flush the output - logFile.flush(); - logFile.close(); - } - return ret; -} - -String getMacAddressAsString(const uint8_t *mac); -DynamicJsonDocument parseReceivedJsonData(char *data); -String documentToLineProtocolString(const DynamicJsonDocument &doc); - -void syncUTCTimeToRTC(); void on_data_recv(const uint8_t *mac, const uint8_t *incomingData, int len) { - esp_log_write(ESP_LOG_INFO, TAG_ESPNOW.c_str(), "Message recieved\n"); + esp_log_write(ESP_LOG_INFO, TAG_ESPNOW, "Message received\n"); // copy received data to a char array char data[len]; memcpy(data, incomingData, len); - esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW.c_str(), "Raw received Data: %s\n", data); + esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW, "Raw received Data: %s\n", data); if (!esp_now_is_peer_exist(mac)) { esp_now_peer_info_t client = {}; @@ -125,10 +74,18 @@ void on_data_recv(const uint8_t *mac, const uint8_t *incomingData, int len) { esp_err_t status = esp_now_add_peer(&client); if (status != ESP_OK) { - esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW.c_str(), "Failed to add new Peer: %d", status); + esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW, "Failed to add new Peer: %d", status); } } + // TODO: Think if response to client may contain error messages + response response = {}; + response.type = dataAck; + esp_read_mac(response.mac, ESP_MAC_WIFI_STA); + response.time = rtc.getEpoch(); + esp_err_t success = esp_now_send(mac, (uint8_t *)&response, sizeof(response)); + esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW, (success == ESP_OK) ? "Response sent\n" : "Failed to respond\n"); + DynamicJsonDocument doc = parseReceivedJsonData(data); String macAddress = getMacAddressAsString(mac); @@ -141,110 +98,82 @@ void on_data_recv(const uint8_t *mac, const uint8_t *incomingData, int len) { std::string dataString{}; serializeJson(doc, dataString); - saveStringToSDCard(dataString); + try { + SDUtilities::saveStringToSDCard(dataString); + } catch (const std::exception &e) { + esp_log_write(ESP_LOG_ERROR, TAG_ESPNOW, "Failed to save data to SD card: %s", e.what()); + } - String lineData = documentToLineProtocolString(doc); + String serverData = documentToServerReadableString(doc); - esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW.c_str(), "Line protocol data: %s\n", lineData.c_str()); + esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW, "Data to be sent: %s\n", serverData.c_str()); xSemaphoreTake(xMutex, portMAX_DELAY); - queue.push(lineData); + queue.push(serverData); xSemaphoreGive(xMutex); - - response response = {}; - response.type = dataAck; - esp_read_mac(response.mac, ESP_MAC_WIFI_STA); - response.time = rtc.getEpoch(); - esp_err_t success = esp_now_send(mac, (uint8_t *)&response, sizeof(response)); - esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW.c_str(), (success == ESP_OK) ? "Response sent\n" : "Failed to respond\n"); } -String documentToLineProtocolString(const DynamicJsonDocument &doc) { - String measurementType = doc["measurementType"].as<String>(); - String sensorName = doc["sensorName"].as<String>(); - String timestamp = doc["timestamp"].as<String>(); - String protocol = doc["protocol"].as<String>(); - String value = doc["value"].as<String>(); - String channel = doc["channel"].as<String>(); - String clientMac = doc["clientMac"].as<String>(); - - String lineData = sensorName + ",clientMac=" + clientMac + ",protocol=" + protocol + ",channel=" + channel + " " - + measurementType + "=" + value + " " + timestamp; - return lineData; -} -DynamicJsonDocument parseReceivedJsonData(char *data) { - DynamicJsonDocument doc(250); - auto error = deserializeJson(doc, data); - if (error) { - esp_log_write(ESP_LOG_ERROR, TAG_ESPNOW.c_str(), "Error while parsing json: %s\n", error.f_str()); - // TODO error handling - } - return doc; -} -String getMacAddressAsString(const uint8_t *mac) { - String macAddress; - for (int i = 0; i < 6; i++) { - macAddress += String(mac[i], HEX); +[[noreturn]] void esp_loop() { + while (true) { } - esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW.c_str(), "MAC: %s\n", macAddress.c_str()); - return macAddress; } [[noreturn]] void ESPNOWReceiveTask(void *parameter) { - esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW.c_str(), "ESPNOWReceiveTask started on core %d\n", xPortGetCoreID()); + esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW, "ESPNOWReceiveTask started on core %d\n", xPortGetCoreID()); WiFi.mode(WIFI_STA); - esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW.c_str(), "Initialising ESPNow...\n"); + esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW, "Initialising ESPNow...\n"); if (esp_now_init() != ESP_OK) { // initialization failed - esp_log_write(ESP_LOG_ERROR, TAG_ESPNOW.c_str(), "Initialising ESPNow FAILED\n"); + esp_log_write(ESP_LOG_ERROR, TAG_ESPNOW, "Initialising ESPNow FAILED\n"); exit(ESP_FAIL); } - esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW.c_str(), "Initialising ESPNow SUCCESS\n"); + esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW, "Initialising ESPNow SUCCESS\n"); esp_now_register_recv_cb(on_data_recv); - while (true) { + broadcast.channel = 0; + broadcast.encrypt = false; + memcpy(&broadcast.peer_addr, &BROADCAST_MAC, sizeof(BROADCAST_MAC)); + if (esp_now_add_peer(&broadcast) != ESP_OK) { + esp_log_write(ESP_LOG_WARN, TAG_ESPNOW, "Failed to add Broadcast Host"); } -} - -time_t timeToUnixEpochSeconds(const std::string &time) { - // 22/10/27,10:16:20+00 - struct tm tm {}; - time_t dateInEpoch = 0; - - if (strptime(time.c_str(), "%y/%m/%d,%T+00", &tm)) { - time_t curTime; - struct tm *timeinfo; - - timeinfo = localtime(&curTime); - - timeinfo->tm_year = tm.tm_year; - timeinfo->tm_mon = tm.tm_mon; - timeinfo->tm_mday = tm.tm_mday; - timeinfo->tm_hour = tm.tm_hour; - timeinfo->tm_min = tm.tm_min; - timeinfo->tm_sec = tm.tm_sec; - timeinfo->tm_isdst = -1; - dateInEpoch = mktime(timeinfo); + announce.type = hostChange; + esp_read_mac(announce.mac, ESP_MAC_WIFI_STA); + announce.time = rtc.getEpoch(); + if (esp_now_send(BROADCAST_MAC, (uint8_t *)&announce, sizeof(announce)) != ESP_OK) { + esp_log_write(ESP_LOG_WARN, TAG_ESPNOW, "Failed to announce mac"); + } else { + esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW, "Mac announced!"); } - return dateInEpoch; + + esp_loop(); } void setup() { // Set console baud rate Serial.begin(115200); delay(10); - setupSDCard(SD_MISO, SD_MOSI, SD_SCLK, SD_CS); + try { + SDUtilities::setupSDCard(SD_MISO, SD_MOSI, SD_SCLK, SD_CS); + } catch (const SDSetupException &e) { + SDUtilities::setSDAvailable(false); + esp_log_write(ESP_LOG_ERROR, TAG_MAIN, "SD Card setup failed: %s\n", e.what()); + } + + SDCardLogger::printDebugToSerial(PRINT_TO_SERIAL); // https://stackoverflow.com/questions/60442350/arduinos-esp-log-set-vprintf-does-not-work-on-esp32 - esp_log_set_vprintf(&vprintf_into_sd); + esp_log_set_vprintf(&SDCardLogger::vprintf_into_sd); esp_log_level_set("*", ESP_LOG_VERBOSE); - esp_log_write(ESP_LOG_DEBUG, TAG.c_str(), "%s", WiFi.macAddress().c_str()); + esp_log_write(ESP_LOG_DEBUG, TAG_MAIN, "%s", WiFi.macAddress().c_str()); + + resendManager.init(); + turnOffLEDs(); xMutex = xSemaphoreCreateMutex(); @@ -261,207 +190,101 @@ void setup() { 0); /* Core where the task should run */ SerialAT.begin(UART_BAUD, SERIAL_8N1, PIN_RX, PIN_TX); +} - // Restart takes quite some time - // To skip it, call init() instead of restart() - esp_log_write(ESP_LOG_DEBUG, TAG_GSM.c_str(), "Initializing modem...\n"); - if (!modem.restart()) { - esp_log_write(ESP_LOG_WARN, TAG_GSM.c_str(), - "Failed to restart modem, attempting to continue without restarting\n"); - } +void loop() { - syncUTCTimeToRTC(); + connectionManager.modemPowerOn(); + delay(5000L); - broadcast.channel = 0; - broadcast.encrypt = false; - memcpy(&broadcast.peer_addr, &BROADCAST_MAC, sizeof(BROADCAST_MAC)); - if (esp_now_add_peer(&broadcast) != ESP_OK) { - esp_log_write(ESP_LOG_WARN, TAG_ESPNOW.c_str(), "Failed to add Broadcast Host"); - } + // Restart takes quite some time + // To skip it, call init() instead of restart() + connectionManager.restartModem(); - announce.type = hostChange; - esp_read_mac(announce.mac, ESP_MAC_WIFI_STA); announce.time = rtc.getEpoch(); if (esp_now_send(BROADCAST_MAC, (uint8_t *)&announce, sizeof(announce)) != ESP_OK) { - esp_log_write(ESP_LOG_WARN, TAG_ESPNOW.c_str(), "Failed to announce mac"); + esp_log_write(ESP_LOG_WARN, TAG_ESPNOW, "Failed to announce mac\n"); } else { - esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW.c_str(), "Mac announced!"); + esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW, "Mac announced!\n"); } -} -void syncUTCTimeToRTC() { - esp_log_write(ESP_LOG_DEBUG, TAG_GSM.c_str(), "NTP Server Syncing...\n"); - modem.NTPServerSync("pool.ntp.org", 0); - auto gsmDateTimeString = modem.getGSMDateTime(DATE_FULL); - esp_log_write(ESP_LOG_DEBUG, TAG_GSM.c_str(), "GSM DateTime: %s\n", gsmDateTimeString.c_str()); - time_t time = timeToUnixEpochSeconds(gsmDateTimeString.c_str()); - rtc.setTime(time); - esp_log_write(ESP_LOG_INFO, TAG_GSM.c_str(), "Time set to EPOCH: %s\n", String(rtc.getEpoch()).c_str()); -} -struct RequestInformation { - String method; - String host; - String path; - String body; -}; - -String buildRequest(const RequestInformation &requestInformation) { - String request = ""; - request += requestInformation.method + " " + requestInformation.path + " HTTP/1.1\r\n"; - request += "Host: " + requestInformation.host + "\r\n"; - request += "Authorization: Token " + INFLUXDB_TOKEN + "\r\n"; - request += "User-Agent: ESP32\r\n"; - request += "Content-Type: text/plain\r\n"; - request += "Content-Length: " + String(requestInformation.body.length()) + "\r\n"; - request += "\r\n"; - request += requestInformation.body; - return request; -} - -void loop() { + connectionManager.disableGPS(); - // Restart takes quite some time - // To skip it, call init() instead of restart() - esp_log_write(ESP_LOG_DEBUG, TAG.c_str(), "Initializing modem...\n"); - if (!modem.init()) { - esp_log_write(ESP_LOG_DEBUG, TAG.c_str(), - "Failed to restart modem, attempting to continue without restarting\n"); + try { + // may fail if already set + connectionManager.setNetworkMode(LTE_ONLY); + connectionManager.setPreferredMode(CAT_M); + } catch (const std::exception &e) { + esp_log_write(ESP_LOG_ERROR, TAG_GSM, "Error setting network mode: %s\n", e.what()); } - announce.time = rtc.getEpoch(); - if (esp_now_send(BROADCAST_MAC, (uint8_t *)&announce, sizeof(announce)) != ESP_OK) { - esp_log_write(ESP_LOG_WARN, TAG_ESPNOW.c_str(), "Failed to announce mac\n"); - } else { - esp_log_write(ESP_LOG_DEBUG, TAG_ESPNOW.c_str(), "Mac announced!\n"); - } + esp_log_write(ESP_LOG_DEBUG, TAG_GSM, "Modem Name: %s\n", modem.getModemName().c_str()); - String name = modem.getModemName(); - delay(500); - esp_log_write(ESP_LOG_DEBUG, TAG.c_str(), "Modem Name %s\n", name.c_str()); + esp_log_write(ESP_LOG_DEBUG, TAG_GSM, "Modem Info: %s\n", modem.getModemInfo().c_str()); - String modemInfo = modem.getModemInfo(); - delay(500); - esp_log_write(ESP_LOG_DEBUG, TAG.c_str(), "Modem Info: %s\n", modemInfo.c_str()); + connectionManager.unlockSimCard(); - // Set SIM7000G GPIO4 LOW ,turn off GPS power - // CMD:AT+SGPIO=0,4,1,0 - // Only in version 20200415 is there a function to control GPS power - modem.sendAT("+SGPIO=0,4,1,0"); - if (modem.waitResponse(10000L) != 1) { - DBG(" SGPIO=0,4,1,0 false "); + if (!connectionManager.waitForNetwork()) { + esp_log_write(ESP_LOG_ERROR, TAG_GSM, "Network not available\n"); + return; } - modem.sendAT("+CFUN=0 "); - if (modem.waitResponse(10000L) != 1) { - DBG(" +CFUN=0 false "); + if (connectionManager.isNetworkConnected()) { + esp_log_write(ESP_LOG_DEBUG, TAG_GSM, "Network connected\n"); } + delay(200); - /* - 2 Automatic - 13 GSM only - 38 LTE only - 51 GSM and LTE only - * * * */ - String res; - res = modem.setNetworkMode(38); - if (res != "1") { - DBG("setNetworkMode false "); + esp_log_write(ESP_LOG_DEBUG, TAG_GSM, "Connecting to %s\n", gprsCredentials.apn.c_str()); + if (!connectionManager.gprsConnect()) { + esp_log_write(ESP_LOG_ERROR, TAG_GSM, "GPRS not connected\n"); return; } - delay(200); - /* - 1 CAT-M - 2 NB-Iot - 3 CAT-M and NB-IoT - * * */ - // res = modem.setPreferredMode(1); - // if (res != "1") { - // - // DBG("setPreferredMode false "); - // return; - // } - delay(200); + connectionManager.logModemInformation(); - /*AT+CBANDCFG=<mode>,<band>[,<band>…] - * <mode> "CAT-M" "NB-IOT" - * <band> The value of <band> must is in the band list of getting from AT+CBANDCFG=? - * For example, my SIM card carrier "NB-iot" supports B8. I will configure +CBANDCFG= "Nb-iot ",8 - */ - modem.sendAT("+CBANDCFG=\"CAT-M\",8 "); - if (modem.waitResponse(10000L) != 1) { - DBG(" +CBANDCFG=\"NB-IOT\" "); + ntpManager.syncNTP("pool.ntp.org", "at.pool.ntp.org"); + + try { + ntpManager.writeModemTimeToRTC(); + } catch (const std::exception &e) { + esp_log_write(ESP_LOG_ERROR, TAG_GSM, "Error writing time to rtc: %s\n", e.what()); } - delay(200); - modem.sendAT("+CFUN=1 "); - if (modem.waitResponse(10000L) != 1) { - DBG(" +CFUN=1 false "); + if (connectionManager.isGprsConnected()) { + esp_log_write(ESP_LOG_DEBUG, TAG_GSM, "GPRS connected\n"); + // make list of map of headers + std::map<String, String> headers; + headers["Content-Type"] = "application/json"; + headers["Accept"] = "application/json"; + + xSemaphoreTake(xMutex, portMAX_DELAY); + String data = queue.front(); + queue.pop(); + xSemaphoreGive(xMutex); + + RequestInformation requestInformation{POST, "influxdb.qe-forte.uibk.ac.at", + 80, "/api/v2/write?org=QE&bucket=esp32test&precision=s", + data, headers}; + + try { + connectionManager.connect(requestInformation); + } catch (const std::exception &e) { + esp_log_write(ESP_LOG_ERROR, TAG_GSM, "Error sending data: %s\n", e.what()); + } + esp_log_write(ESP_LOG_DEBUG, TAG_GSM, "Data sent: %s\n", data.c_str()); } - delay(200); - // modem.disableGPS(); - delay(200); - esp_log_write(ESP_LOG_DEBUG, TAG.c_str(), "%s\n", String(modem.getSignalQuality()).c_str()); - delay(200); - esp_log_write(ESP_LOG_DEBUG, TAG.c_str(), "Trying to connect to network\n"); - modem.gprsConnect(apn, gprsUser, gprsPass); - delay(200); - syncUTCTimeToRTC(); + connectionManager.gprsDisconnect(); + delay(5000L); - esp_log_write(ESP_LOG_DEBUG, TAG.c_str(), "Waiting for network...\n"); - if (!modem.isNetworkConnected()) { - esp_log_write(ESP_LOG_DEBUG, TAG.c_str(), "Network not connected\n"); - return; + if (!modem.isGprsConnected()) { + esp_log_write(ESP_LOG_DEBUG, TAG_GSM, "GPRS disconnected\n"); } else { - esp_log_write(ESP_LOG_DEBUG, TAG.c_str(), "Network connected\n"); - delay(200); - - // quality - esp_log_write(ESP_LOG_DEBUG, TAG.c_str(), "%s\n", String(modem.getSignalQuality()).c_str()); - // make a http post request - String url = "influxdb.qe-forte.uibk.ac.at"; - String path = "/api/v2/write?org=QE&bucket=esp32test&precision=s"; - Serial.print("Connecting to "); - esp_log_write(ESP_LOG_DEBUG, TAG.c_str(), "%s\n", url.c_str()); - // Use WiFiClient class to create TCP connections - - while (!queue.empty()) { - - xSemaphoreTake(xMutex, portMAX_DELAY); - String lineData = queue.front(); - queue.pop(); - xSemaphoreGive(xMutex); - - RequestInformation requestInformation{.method = "POST", .host = url, .path = path, .body = lineData}; - - //"sensorName":"DRS26","timestamp":1666872216,"protocol":"I2C","value":0,"channel":0,"measurementType":"CIRCUMFERENCE_INCREMENT" - - String request = buildRequest(requestInformation); - esp_log_write(ESP_LOG_VERBOSE, TAG.c_str(), "request: %s\n", request.c_str()); - - TinyGsmClient client{modem}; - const int httpPort = 80; - if (!client.connect(url.c_str(), httpPort)) { - esp_log_write(ESP_LOG_DEBUG, TAG.c_str(), "connection failed\n"); - return; - } - - client.print(request); - - // print response - while (client.connected()) { - String line = client.readStringUntil('\n'); - if (line == "\r") { - esp_log_write(ESP_LOG_DEBUG, TAG.c_str(), "headers received\n"); - break; - } - } - client.stop(); - delay(1000); - } - DBG("Network connected"); + esp_log_write(ESP_LOG_ERROR, TAG_GSM, "GPRS not disconnected\n"); } - delay(1000); + + connectionManager.modemPowerOff(); + + delay(10000); } diff --git a/host/host_central_mast/src/main2.cpp b/host/host_central_mast/src/main2.cpp deleted file mode 100644 index aaedda0570dfadd1926531602f0c527d39e6f870..0000000000000000000000000000000000000000 --- a/host/host_central_mast/src/main2.cpp +++ /dev/null @@ -1,304 +0,0 @@ -///************************************************************** -// * -// * This sketch connects to a website and downloads a page. -// * It can be used to perform HTTP/RESTful API calls. -// * -// * TinyGSM Getting Started guide: -// * https://tiny.cc/tinygsm-readme -// * -// **************************************************************/ -// -//// Select your modem: -//// #define TINY_GSM_MODEM_SIM800 -//// #define TINY_GSM_MODEM_SIM808 -//// #define TINY_GSM_MODEM_SIM868 -//// #define TINY_GSM_MODEM_SIM900 -//#define TINY_GSM_MODEM_SIM7000 -//// #define TINY_GSM_MODEM_SIM7000SSL -//// #define TINY_GSM_MODEM_SIM7080 -//// #define TINY_GSM_MODEM_SIM5360 -//// #define TINY_GSM_MODEM_SIM7600 -//// #define TINY_GSM_MODEM_UBLOX -//// #define TINY_GSM_MODEM_SARAR4 -//// #define TINY_GSM_MODEM_M95 -//// #define TINY_GSM_MODEM_BG96 -//// #define TINY_GSM_MODEM_A6 -//// #define TINY_GSM_MODEM_A7 -//// #define TINY_GSM_MODEM_M590 -//// #define TINY_GSM_MODEM_MC60 -//// #define TINY_GSM_MODEM_MC60E -//// #define TINY_GSM_MODEM_ESP8266 -//// #define TINY_GSM_MODEM_XBEE -//// #define TINY_GSM_MODEM_SEQUANS_MONARCH -// -//// Set serial for debug console (to the Serial Monitor, default speed 115200) -//#define SerialMon Serial -// -//// Set serial for AT commands (to the module) -//// Use Hardware Serial on Mega, Leonardo, Micro -//#ifndef __AVR_ATmega328P__ -//#define SerialAT Serial1 -// -//// or Software Serial on Uno, Nano -//#else -//#include <SoftwareSerial.h> -//SoftwareSerial SerialAT(2, 3); // RX, TX -//#endif -// -//// Increase RX buffer to capture the entire response -//// Chips without internal buffering (A6/A7, ESP8266, M590) -//// need enough space in the buffer for the entire response -//// else data will be lost (and the http library will fail). -//#if !defined(TINY_GSM_RX_BUFFER) -//#define TINY_GSM_RX_BUFFER 650 -//#endif -// -//// See all AT commands, if wanted -//#define DUMP_AT_COMMANDS -// -//// Define the serial console for debug prints, if needed -//#define TINY_GSM_DEBUG SerialMon -// -//// Range to attempt to autobaud -//// NOTE: DO NOT AUTOBAUD in production code. Once you've established -//// communication, set a fixed baud rate using modem.setBaud(#). -//#define GSM_AUTOBAUD_MIN 9600 -//#define GSM_AUTOBAUD_MAX 115200 -// -//// Add a reception delay, if needed. -//// This may be needed for a fast processor at a slow baud rate. -//// #define TINY_GSM_YIELD() { delay(2); } -// -//// Uncomment this if you want to use SSL -//// #define USE_SSL -// -//// Define how you're planning to connect to the internet. -//// This is only needed for this example, not in other code. -//#define TINY_GSM_USE_GPRS true -//#define TINY_GSM_USE_WIFI false -// -//// set GSM PIN, if any -//#define GSM_PIN "" -// -//// Your GPRS credentials, if any -//const char apn[] = "m2m.public.at"; -//const char gprsUser[] = ""; -//const char gprsPass[] = ""; -// -//// Your WiFi connection credentials, if applicable -//const char wifiSSID[] = "YourSSID"; -//const char wifiPass[] = "YourWiFiPass"; -// -//// Server details -//const char server[] = "vsh.pp.ua"; -//const char resource[] = "/TinyGSM/logo.txt"; -// -//#include <TinyGsmClient.h> -//#include <WiFi.h> -//#include <esp_now.h> -// -//// Just in case someone defined the wrong thing.. -//#if TINY_GSM_USE_GPRS && not defined TINY_GSM_MODEM_HAS_GPRS -//#undef TINY_GSM_USE_GPRS -//#undef TINY_GSM_USE_WIFI -//#define TINY_GSM_USE_GPRS false -//#define TINY_GSM_USE_WIFI true -//#endif -//#if TINY_GSM_USE_WIFI && not defined TINY_GSM_MODEM_HAS_WIFI -//#undef TINY_GSM_USE_GPRS -//#undef TINY_GSM_USE_WIFI -//#define TINY_GSM_USE_GPRS true -//#define TINY_GSM_USE_WIFI false -//#endif -// -//#ifdef DUMP_AT_COMMANDS -//#include <StreamDebugger.h> -//StreamDebugger debugger(SerialAT, SerialMon); -//TinyGsm modem(debugger); -//#else -//TinyGsm modem(SerialAT); -//#endif -// -//#ifdef USE_SSL -//TinyGsmClientSecure client_satellite(modem); -//const int port = 443; -//#else -//TinyGsmClient client_satellite(modem); -//const int port = 80; -//#endif -// -//TaskHandle_t Task1; -// -//void on_data_sent(const uint8_t *mac_addr, esp_now_send_status_t status) -//{ -// // go to sleep -//} -// -//void on_data_recv(const uint8_t *mac, const uint8_t *incomingData, int len) -//{ -// // print mac -// Serial.print("Message recieved: Core "); -// Serial.println(xPortGetCoreID()); -// for (int i = 0; i < 6; i++) { -// Serial.print(mac[i], HEX); -// Serial.print(":"); -// } -// Serial.println(); -// char data[len]; -// memcpy(data, incomingData, len); -// Serial.println(data); -//} -// -//[[noreturn]] void Task1code(void *parameter) -//{ -// Serial.println(xPortGetCoreID()); -// WiFi.mode(WIFI_STA); -// Serial.println("ESPNow init"); -// if (esp_now_init() != ESP_OK) { -// // initialization failed -// Serial.println("ESPNow init failed"); -// // not sure about this -// } -// Serial.println("ESPNow init success"); -// esp_now_register_recv_cb(on_data_recv); -// -// while (true) { -// } -//} -// -//void setup2() -//{ -// // Set console baud rate -// SerialMon.begin(115200); -// delay(10); -// -// // !!!!!!!!!!! -// // Set your reset, enable, power pins here -// // !!!!!!!!!!! -// -// SerialMon.println("Wait..."); -// -// xTaskCreatePinnedToCore(Task1code, /* Function to implement the task */ -// "Task1", /* Name of the task */ -// 10000, /* Stack size in words */ -// nullptr, /* Task input parameter */ -// 0, /* Priority of the task */ -// &Task1, /* Task handle. */ -// 0); /* Core where the task should run */ -// -// // Set GSM module baud rate -// // TinyGsmAutoBaud(SerialAT, GSM_AUTOBAUD_MIN, GSM_AUTOBAUD_MAX); -// modem.setBaud(9600); -// // SerialAT.begin(9600); -// delay(6000); -// -// // Restart takes quite some time -// // To skip it, call init() instead of restart() -// SerialMon.println("Initializing modem..."); -// modem.restart(); -// // modem.init(); -// -// String modemInfo = modem.getModemInfo(); -// SerialMon.print("Modem Info: "); -// SerialMon.println(modemInfo); -// -//#if TINY_GSM_USE_GPRS -// // Unlock your SIM card with a PIN if needed -// if (GSM_PIN && modem.getSimStatus() != 3) { -// modem.simUnlock(GSM_PIN); -// } -//#endif -//} -// -//void loop2() -//{ -//#if TINY_GSM_USE_WIFI -// // Wifi connection parameters must be set before waiting for the network -// SerialMon.print(F("Setting SSID/password...")); -// if (!modem.networkConnect(wifiSSID, wifiPass)) { -// SerialMon.println(" fail"); -// delay(10000); -// return; -// } -// SerialMon.println(" success"); -//#endif -// -//#if TINY_GSM_USE_GPRS && defined TINY_GSM_MODEM_XBEE -// // The XBee must run the gprsConnect function BEFORE waiting for network! -// modem.gprsConnect(apn, gprsUser, gprsPass); -//#endif -// -// SerialMon.print("Waiting for network..."); -// if (!modem.waitForNetwork()) { -// SerialMon.println(" fail"); -// delay(10000); -// return; -// } -// SerialMon.println(" success"); -// -// if (modem.isNetworkConnected()) { -// SerialMon.println("Network connected"); -// } -// -//#if TINY_GSM_USE_GPRS -// // GPRS connection parameters are usually set after network registration -// SerialMon.print(F("Connecting to ")); -// SerialMon.print(apn); -// if (!modem.gprsConnect(apn, gprsUser, gprsPass)) { -// SerialMon.println(" fail"); -// delay(10000); -// return; -// } -// SerialMon.println(" success"); -// -// if (modem.isGprsConnected()) { -// SerialMon.println("GPRS connected"); -// } -//#endif -// -// SerialMon.print("Connecting to "); -// SerialMon.println(server); -// if (!client_satellite.connect(server, port)) { -// SerialMon.println(" fail"); -// delay(10000); -// return; -// } -// SerialMon.println(" success"); -// -// // Make a HTTP GET request: -// SerialMon.println("Performing HTTP GET request..."); -// client_satellite.print(String("GET ") + resource + " HTTP/1.1\r\n"); -// client_satellite.print(String("Host: ") + server + "\r\n"); -// client_satellite.print("Connection: close\r\n\r\n"); -// client_satellite.println(); -// -// uint32_t timeout = millis(); -// while (client_satellite.connected() && millis() - timeout < 10000L) { -// // Print available data -// while (client_satellite.available()) { -// char c = client_satellite.read(); -// SerialMon.print(c); -// timeout = millis(); -// } -// } -// SerialMon.println(); -// -// // Shutdown -// -// client_satellite.stop(); -// SerialMon.println(F("Server disconnected")); -// -//#if TINY_GSM_USE_WIFI -// modem.networkDisconnect(); -// SerialMon.println(F("WiFi disconnected")); -//#endif -//#if TINY_GSM_USE_GPRS -// modem.gprsDisconnect(); -// SerialMon.println(F("GPRS disconnected")); -//#endif -// -// // Do nothing forevermore -// while (true) { -// delay(1000); -// } -//}