From bfae0e0defe389b1523fb006fa38d2d66b2248d6 Mon Sep 17 00:00:00 2001 From: lumapu Date: Thu, 6 Apr 2023 12:16:23 +0200 Subject: [PATCH 1/4] basic implementation of improv --- src/app.cpp | 3 +++ src/app.h | 16 ++-------------- src/hm/hmRadio.h | 2 +- 3 files changed, 6 insertions(+), 15 deletions(-) diff --git a/src/app.cpp b/src/app.cpp index 9ec6ff26..4d775473 100644 --- a/src/app.cpp +++ b/src/app.cpp @@ -82,6 +82,8 @@ void app::setup() { mPubSerial.setup(mConfig, &mSys, &mTimestamp); + mImprov.setup(mConfig->sys.deviceName, mVersion); + regularTickers(); @@ -169,6 +171,7 @@ void app::regularTickers(void) { if (mConfig->plugin.display.type != 0) everySec(std::bind(&DisplayType::tickerSecond, &mDisplay), "disp"); every(std::bind(&PubSerialType::tick, &mPubSerial), mConfig->serial.interval, "uart"); + everySec(std::bind(&Improv::tickSerial, &mImprov), "impro"); } //----------------------------------------------------------------------------- diff --git a/src/app.h b/src/app.h index 5ea40968..146e2210 100644 --- a/src/app.h +++ b/src/app.h @@ -22,6 +22,7 @@ #include "utils/crc.h" #include "utils/dbg.h" #include "utils/scheduler.h" +#include "utils/improv.h" #include "web/RestApi.h" #include "web/web.h" #include "wifi/ahoywifi.h" @@ -243,20 +244,6 @@ class app : public IApp, public ah::Scheduler { void tickMinute(void); void tickZeroValues(void); void tickMidnight(void); - /*void tickSerial(void) { - if(Serial.available() == 0) - return; - - uint8_t buf[80]; - uint8_t len = Serial.readBytes(buf, 80); - DPRINTLN(DBG_INFO, "got serial data, len: " + String(len)); - for(uint8_t i = 0; i < len; i++) { - if((0 != i) && (i % 8 == 0)) - DBGPRINTLN(""); - DBGPRINT(String(buf[i], HEX) + " "); - } - DBGPRINTLN(""); - }*/ innerLoopCb mInnerLoopCb; @@ -269,6 +256,7 @@ class app : public IApp, public ah::Scheduler { PayloadType mPayload; MiPayloadType mMiPayload; PubSerialType mPubSerial; + Improv mImprov; char mVersion[12]; settings mSettings; diff --git a/src/hm/hmRadio.h b/src/hm/hmRadio.h index c37ab7d0..602d76a1 100644 --- a/src/hm/hmRadio.h +++ b/src/hm/hmRadio.h @@ -1,6 +1,6 @@ //----------------------------------------------------------------------------- // 2023 Ahoy, https://github.com/lumpapu/ahoy -// Creative Commons - http://creativecommons.org/licenses/by-nc-sa/3.0/de/ +// Creative Commons - http://creativecommons.org/licenses/by-nc-sa/4.0/deed //----------------------------------------------------------------------------- #ifndef __RADIO_H__ From 62c9cb7529df766e7f4c906663d0959c828a9b0c Mon Sep 17 00:00:00 2001 From: lumapu Date: Thu, 6 Apr 2023 23:29:56 +0200 Subject: [PATCH 2/4] add missing file --- src/utils/improv.h | 173 +++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 173 insertions(+) create mode 100644 src/utils/improv.h diff --git a/src/utils/improv.h b/src/utils/improv.h new file mode 100644 index 00000000..2ed16599 --- /dev/null +++ b/src/utils/improv.h @@ -0,0 +1,173 @@ +//----------------------------------------------------------------------------- +// 2023 Ahoy, https://github.com/lumpapu/ahoy +// Creative Commons - http://creativecommons.org/licenses/by-nc-sa/4.0/deed +//----------------------------------------------------------------------------- + +#ifndef __IMPROV_H__ +#define __IMPROV_H__ + +#include +#include +#include "dbg.h" + +// https://www.improv-wifi.com/serial/ +// https://github.com/jnthas/improv-wifi-demo/blob/main/src/esp32-wifiimprov/esp32-wifiimprov.ino + +// configure ESP through Serial interface +class Improv { + public: + void setup(const char *devName, const char *version) { + mDevName = devName; + mVersion = version; + } + + void tickSerial(void) { + if(Serial.available() == 0) + return; + + uint8_t buf[80]; + uint8_t len = Serial.readBytes(buf, 80); + dumpBuf(buf, len); + + if(!checkPaket(&buf[0], 13, [this](uint8_t type, uint8_t buf[], uint8_t len) { + parsePayload(type, buf, len); + })) { + DBGPRINT(F("checkHeader failed")); + } + } + + private: + enum State : uint8_t { + STATE_STOPPED = 0x00, + STATE_AWAITING_AUTHORIZATION = 0x01, + STATE_AUTHORIZED = 0x02, + STATE_PROVISIONING = 0x03, + STATE_PROVISIONED = 0x04, + }; + + enum Command : uint8_t { + UNKNOWN = 0x00, + WIFI_SETTINGS = 0x01, + IDENTIFY = 0x02, + GET_CURRENT_STATE = 0x02, + GET_DEVICE_INFO = 0x03, + GET_WIFI_NETWORKS = 0x04, + BAD_CHECKSUM = 0xFF, + }; + + enum ImprovSerialType : uint8_t { + TYPE_CURRENT_STATE = 0x01, + TYPE_ERROR_STATE = 0x02, + TYPE_RPC = 0x03, + TYPE_RPC_RESPONSE = 0x04 + }; + + void dumpBuf(uint8_t buf[], uint8_t len) { + for(uint8_t i = 0; i < len; i++) { + DHEX(buf[i]); + DBGPRINT(" "); + } + DBGPRINTLN(""); + } + + inline uint8_t buildChecksum(uint8_t buf[], uint8_t len) { + uint8_t calc = 0; + for(uint8_t i = 0; i < (len - 1); i++) { + calc += buf[i]; + } + return calc; + } + + inline bool checkChecksum(uint8_t buf[], uint8_t len) { + return ((buildChecksum(buf, len)) == buf[len]); + } + + bool checkPaket(uint8_t buf[], uint8_t len, std::function cb) { + if(len < 11) + return false; + + if(0 != strncmp((char*)buf, "IMPROV", 6)) + return false; + + // verison check (only version 1 is supported!) + if(0x01 != buf[6]) + return false; + + if(!checkChecksum(buf, (6+3 + buf[9]))) + return false; + + cb(buf[7], &buf[9], buf[8]); + + return true; + } + + uint8_t char2Improv(const char *str, uint8_t buf[]) { + uint8_t len = strlen(str); + buf[0] = len; + for(uint8_t i = 1; i <= len; i++) { + buf[i] = (uint8_t)str[i-1]; + } + return len + 1; + } + + void sendDevInfo(void) { + uint8_t buf[100]; + buf[7] = TYPE_RPC_RESPONSE; + buf[9] = GET_DEVICE_INFO; // repsonse to cmd + uint8_t p = 11; + // firmware name + p += char2Improv("AhoyDTU", &buf[p]); + // firmware version + p += char2Improv(mVersion, &buf[p]); + // chip variant + #if defined(ESP32) + p += char2Improv("ESP32", &buf[p]); + #else + p += char2Improv("ESP8266", &buf[p]); + #endif + // device name + p += char2Improv(mDevName, &buf[p]); + + buf[10] = p - 11; // sub length + buf[8] = p - 9; // paket length + + sendPaket(buf, p); + } + + void setState(uint8_t state) { + uint8_t buf[20]; + buf[7] = TYPE_CURRENT_STATE; + buf[8] = 0x01; + buf[9] = state; + sendPaket(buf, 10); + } + + void sendPaket(uint8_t buf[], uint8_t len) { + buf[0] = 'I'; + buf[1] = 'M'; + buf[2] = 'P'; + buf[3] = 'R'; + buf[4] = 'O'; + buf[5] = 'V'; + buf[6] = 1; // protocol version + + buf[len] = buildChecksum(buf, (len+1)); + len++; + dumpBuf(buf, len); + Serial.write(buf, len); + } + + void parsePayload(uint8_t type, uint8_t buf[], uint8_t len) { + DBGPRINTLN(F("parsePayload")); + if(TYPE_RPC == type) { + if(GET_CURRENT_STATE == buf[0]) + setState(0x00); + else if(GET_DEVICE_INFO == buf[0]) + sendDevInfo(); + } + } + + const char *mDevName, *mVersion; +}; + +#endif /*__IMPROV_H__*/ From ba3f84d6845a012d0b019a7074fa74d8bc47125a Mon Sep 17 00:00:00 2001 From: lumapu Date: Fri, 7 Apr 2023 01:16:29 +0200 Subject: [PATCH 3/4] improv: first two commands are working (state and device info) --- src/app.cpp | 3 ++- src/utils/dbg.h | 18 ++++++++++-------- src/utils/improv.h | 30 ++++++++++++++++-------------- 3 files changed, 28 insertions(+), 23 deletions(-) diff --git a/src/app.cpp b/src/app.cpp index 571d933a..7f2fefd7 100644 --- a/src/app.cpp +++ b/src/app.cpp @@ -98,6 +98,7 @@ void app::setup() { //----------------------------------------------------------------------------- void app::loop(void) { mInnerLoopCb(); + mImprov.tickSerial(); } //----------------------------------------------------------------------------- @@ -173,7 +174,7 @@ void app::regularTickers(void) { if (mConfig->plugin.display.type != 0) everySec(std::bind(&DisplayType::tickerSecond, &mDisplay), "disp"); every(std::bind(&PubSerialType::tick, &mPubSerial), mConfig->serial.interval, "uart"); - everySec(std::bind(&Improv::tickSerial, &mImprov), "impro"); + //everySec(std::bind(&Improv::tickSerial, &mImprov), "impro"); } //----------------------------------------------------------------------------- diff --git a/src/utils/dbg.h b/src/utils/dbg.h index 4716d7ae..53c9f79e 100644 --- a/src/utils/dbg.h +++ b/src/utils/dbg.h @@ -49,21 +49,23 @@ #endif //template - inline void DBGPRINT(String str) { DSERIAL.print(str); if(NULL != mCb) mCb(str); } + inline void DBGPRINT(String str, bool ser = true) { if(ser) DSERIAL.print(str); if(NULL != mCb) mCb(str); } //template - inline void DBGPRINTLN(String str) { DBGPRINT(str); DBGPRINT(F("\r\n")); } - inline void DHEX(uint8_t b) { - if( b<0x10 ) DSERIAL.print(F("0")); - DSERIAL.print(b,HEX); + inline void DBGPRINTLN(String str, bool ser = true) { DBGPRINT(str); DBGPRINT(F("\r\n")); } + inline void DHEX(uint8_t b, bool ser = true) { + if(ser) { + if( b<0x10 ) DSERIAL.print(F("0")); + DSERIAL.print(b,HEX); + } if(NULL != mCb) { if( b<0x10 ) mCb(F("0")); mCb(String(b, HEX)); } } - inline void DBGHEXLN(uint8_t b) { - DHEX(b); - DBGPRINT(F("\r\n")); + inline void DBGHEXLN(uint8_t b, bool ser = true) { + DHEX(b, ser); + DBGPRINT(F("\r\n"), ser); } /*inline void DHEX(uint16_t b) { if( b<0x10 ) DSERIAL.print(F("000")); diff --git a/src/utils/improv.h b/src/utils/improv.h index 2ed16599..c3b8c87c 100644 --- a/src/utils/improv.h +++ b/src/utils/improv.h @@ -25,15 +25,15 @@ class Improv { if(Serial.available() == 0) return; - uint8_t buf[80]; - uint8_t len = Serial.readBytes(buf, 80); - dumpBuf(buf, len); + uint8_t buf[40]; + uint8_t len = Serial.readBytes(buf, 40); - if(!checkPaket(&buf[0], 13, [this](uint8_t type, uint8_t buf[], uint8_t len) { + if(!checkPaket(&buf[0], len, [this](uint8_t type, uint8_t buf[], uint8_t len) { parsePayload(type, buf, len); })) { - DBGPRINT(F("checkHeader failed")); + DBGPRINTLN(F("check paket failed")); } + dumpBuf(buf, len); } private: @@ -64,21 +64,24 @@ class Improv { void dumpBuf(uint8_t buf[], uint8_t len) { for(uint8_t i = 0; i < len; i++) { - DHEX(buf[i]); - DBGPRINT(" "); + DHEX(buf[i], false); + DBGPRINT(" ", false); } - DBGPRINTLN(""); + DBGPRINTLN("", false); } inline uint8_t buildChecksum(uint8_t buf[], uint8_t len) { uint8_t calc = 0; - for(uint8_t i = 0; i < (len - 1); i++) { + for(uint8_t i = 0; i < len; i++) { calc += buf[i]; } return calc; } inline bool checkChecksum(uint8_t buf[], uint8_t len) { + /*DHEX(buf[len], false); + DBGPRINT(F(" == "), false); + DBGHEXLN(buildChecksum(buf, len), false);*/ return ((buildChecksum(buf, len)) == buf[len]); } @@ -93,7 +96,7 @@ class Improv { if(0x01 != buf[6]) return false; - if(!checkChecksum(buf, (6+3 + buf[9]))) + if(!checkChecksum(buf, (9 + buf[8]))) return false; cb(buf[7], &buf[9], buf[8]); @@ -151,17 +154,16 @@ class Improv { buf[5] = 'V'; buf[6] = 1; // protocol version - buf[len] = buildChecksum(buf, (len+1)); + buf[len] = buildChecksum(buf, len); len++; - dumpBuf(buf, len); Serial.write(buf, len); + dumpBuf(buf, len); } void parsePayload(uint8_t type, uint8_t buf[], uint8_t len) { - DBGPRINTLN(F("parsePayload")); if(TYPE_RPC == type) { if(GET_CURRENT_STATE == buf[0]) - setState(0x00); + setState(STATE_AUTHORIZED); else if(GET_DEVICE_INFO == buf[0]) sendDevInfo(); } From efe3bc4d20326cb6efed51d98679ba897eec0dfa Mon Sep 17 00:00:00 2001 From: lumapu Date: Fri, 7 Apr 2023 02:00:19 +0200 Subject: [PATCH 4/4] improv does not open complete menu every time --- src/app.cpp | 2 +- src/app.h | 4 ++-- src/appInterface.h | 2 +- src/utils/dbg.cpp | 1 + src/utils/dbg.h | 9 ++++++-- src/utils/improv.h | 50 ++++++++++++++++++++++++++++++++++++++++--- src/wifi/ahoywifi.cpp | 6 ++++-- src/wifi/ahoywifi.h | 2 +- 8 files changed, 64 insertions(+), 12 deletions(-) diff --git a/src/app.cpp b/src/app.cpp index 7f2fefd7..001b18f6 100644 --- a/src/app.cpp +++ b/src/app.cpp @@ -84,7 +84,7 @@ void app::setup() { mPubSerial.setup(mConfig, &mSys, &mTimestamp); - mImprov.setup(mConfig->sys.deviceName, mVersion); + mImprov.setup(this, mConfig->sys.deviceName, mVersion); regularTickers(); diff --git a/src/app.h b/src/app.h index 010758c4..9a73ddcd 100644 --- a/src/app.h +++ b/src/app.h @@ -101,8 +101,8 @@ class app : public IApp, public ah::Scheduler { mWifi.scanAvailNetworks(); } - void getAvailNetworks(JsonObject obj) { - mWifi.getAvailNetworks(obj); + bool getAvailNetworks(JsonObject obj) { + return mWifi.getAvailNetworks(obj); } void setOnUpdate() { diff --git a/src/appInterface.h b/src/appInterface.h index 44491d91..a3240189 100644 --- a/src/appInterface.h +++ b/src/appInterface.h @@ -25,7 +25,7 @@ class IApp { virtual const char *getVersion() = 0; virtual statistics_t *getStatistics() = 0; virtual void scanAvailNetworks() = 0; - virtual void getAvailNetworks(JsonObject obj) = 0; + virtual bool getAvailNetworks(JsonObject obj) = 0; virtual uint32_t getUptime() = 0; virtual uint32_t getTimestamp() = 0; diff --git a/src/utils/dbg.cpp b/src/utils/dbg.cpp index 9f7c9fd3..e7ae4581 100644 --- a/src/utils/dbg.cpp +++ b/src/utils/dbg.cpp @@ -1,3 +1,4 @@ #include "dbg.h" DBG_CB mCb = NULL; +bool mDebugEn = true; diff --git a/src/utils/dbg.h b/src/utils/dbg.h index 53c9f79e..8ce23db9 100644 --- a/src/utils/dbg.h +++ b/src/utils/dbg.h @@ -39,6 +39,7 @@ #ifdef ARDUINO #define DBG_CB std::function extern DBG_CB mCb; + extern bool mDebugEn; inline void registerDebugCb(DBG_CB cb) { mCb = cb; @@ -48,12 +49,16 @@ #define DSERIAL Serial #endif + inline void setDebugEn(bool en) { + mDebugEn = en; + } + //template - inline void DBGPRINT(String str, bool ser = true) { if(ser) DSERIAL.print(str); if(NULL != mCb) mCb(str); } + inline void DBGPRINT(String str, bool ser = true) { if(ser && mDebugEn) DSERIAL.print(str); if(NULL != mCb) mCb(str); } //template inline void DBGPRINTLN(String str, bool ser = true) { DBGPRINT(str); DBGPRINT(F("\r\n")); } inline void DHEX(uint8_t b, bool ser = true) { - if(ser) { + if(ser && mDebugEn) { if( b<0x10 ) DSERIAL.print(F("0")); DSERIAL.print(b,HEX); } diff --git a/src/utils/improv.h b/src/utils/improv.h index c3b8c87c..ade1291e 100644 --- a/src/utils/improv.h +++ b/src/utils/improv.h @@ -9,6 +9,7 @@ #include #include #include "dbg.h" +#include "AsyncJson.h" // https://www.improv-wifi.com/serial/ // https://github.com/jnthas/improv-wifi-demo/blob/main/src/esp32-wifiimprov/esp32-wifiimprov.ino @@ -16,12 +17,18 @@ // configure ESP through Serial interface class Improv { public: - void setup(const char *devName, const char *version) { + void setup(IApp *app, const char *devName, const char *version) { + mApp = app; mDevName = devName; mVersion = version; + + mScanRunning = false; } void tickSerial(void) { + if(mScanRunning) + getNetworks(); + if(Serial.available() == 0) return; @@ -114,7 +121,7 @@ class Improv { } void sendDevInfo(void) { - uint8_t buf[100]; + uint8_t buf[50]; buf[7] = TYPE_RPC_RESPONSE; buf[9] = GET_DEVICE_INFO; // repsonse to cmd uint8_t p = 11; @@ -137,6 +144,37 @@ class Improv { sendPaket(buf, p); } + void getNetworks(void) { + if(!mScanRunning) + mApp->scanAvailNetworks(); + + JsonObject obj; + if(!mApp->getAvailNetworks(obj)) + return; + + mScanRunning = false; + + uint8_t buf[50]; + buf[7] = TYPE_RPC_RESPONSE; + buf[9] = GET_WIFI_NETWORKS; // repsonse to cmd + uint8_t p = 11; + + JsonArray arr = obj[F("networks")]; + for(uint8_t i = 0; i < arr.size(); i++) { + buf[p++] = strlen(arr[i][F("ssid")]); + // ssid + p += char2Improv(arr[i][F("ssid")], &buf[p]); + buf[p++] = String(arr[i][F("rssi")]).length(); + // rssi + p += char2Improv(String(arr[i][F("rssi")]).c_str(), &buf[p]); + + buf[10] = p - 11; // sub length + buf[8] = p - 9; // paket length + + sendPaket(buf, p); + } + } + void setState(uint8_t state) { uint8_t buf[20]; buf[7] = TYPE_CURRENT_STATE; @@ -162,14 +200,20 @@ class Improv { void parsePayload(uint8_t type, uint8_t buf[], uint8_t len) { if(TYPE_RPC == type) { - if(GET_CURRENT_STATE == buf[0]) + if(GET_CURRENT_STATE == buf[0]) { + setDebugEn(false); setState(STATE_AUTHORIZED); + } else if(GET_DEVICE_INFO == buf[0]) sendDevInfo(); + else if(GET_WIFI_NETWORKS == buf[0]) + getNetworks(); } } + IApp *mApp; const char *mDevName, *mVersion; + bool mScanRunning; }; #endif /*__IMPROV_H__*/ diff --git a/src/wifi/ahoywifi.cpp b/src/wifi/ahoywifi.cpp index 08c19070..f590f67e 100644 --- a/src/wifi/ahoywifi.cpp +++ b/src/wifi/ahoywifi.cpp @@ -274,12 +274,12 @@ void ahoywifi::scanAvailNetworks(void) { } //----------------------------------------------------------------------------- -void ahoywifi::getAvailNetworks(JsonObject obj) { +bool ahoywifi::getAvailNetworks(JsonObject obj) { JsonArray nets = obj.createNestedArray("networks"); int n = WiFi.scanComplete(); if (n < 0) - return; + return false; if(n > 0) { int sort[n]; sortRSSI(&sort[0], n); @@ -292,6 +292,8 @@ void ahoywifi::getAvailNetworks(JsonObject obj) { WiFi.scanDelete(); if(mStaConn == IN_AP_MODE) WiFi.mode(WIFI_AP); + + return true; } //----------------------------------------------------------------------------- diff --git a/src/wifi/ahoywifi.h b/src/wifi/ahoywifi.h index 3a3a35a8..3fde2ab3 100644 --- a/src/wifi/ahoywifi.h +++ b/src/wifi/ahoywifi.h @@ -27,7 +27,7 @@ class ahoywifi { void tickWifiLoop(void); bool getNtpTime(void); void scanAvailNetworks(void); - void getAvailNetworks(JsonObject obj); + bool getAvailNetworks(JsonObject obj); private: typedef enum WiFiStatus {