//----------------------------------------------------------------------------- // 2022 Ahoy, https://www.mikrocontroller.net/topic/525778 // Creative Commons - http://creativecommons.org/licenses/by-nc-sa/3.0/de/ //----------------------------------------------------------------------------- #include "main.h" #include "version.h" #include "html/h/style_css.h" #include "html/h/setup_html.h" #include "html/h/favicon.h" //----------------------------------------------------------------------------- Main::Main(void) { mDns = new DNSServer(); mWeb = new AsyncWebServer(80); //mUpdater = new ESP8266HTTPUpdateServer(); mUdp = new WiFiUDP(); mApActive = true; mWifiSettingsValid = false; mSettingsValid = false; mLimit = 10; mNextTryTs = 0; mApLastTick = 0; snprintf(mVersion, 12, "%d.%d.%d", VERSION_MAJOR, VERSION_MINOR, VERSION_PATCH); memset(&mDeviceName, 0, DEVNAME_LEN); mEep = new eep(); Serial.begin(115200); DPRINTLN(DBG_VERBOSE, F("Main::Main")); mUptimeSecs = 0; mUptimeTicker = 0xffffffff; mUptimeInterval = 1000; mShouldReboot = false; #ifdef AP_ONLY mTimestamp = 1; #else mTimestamp = 0; #endif mHeapStatCnt = 0; } //----------------------------------------------------------------------------- void Main::setup(uint32_t timeout) { DPRINTLN(DBG_VERBOSE, F("Main::setup")); bool startAp = mApActive; mLimit = timeout; mWeb->on("/setup", HTTP_ANY, std::bind(&Main::showSetup, this, std::placeholders::_1)); mWeb->on("/save", HTTP_ANY, std::bind(&Main::showSave, this, std::placeholders::_1)); mWeb->on("/uptime", HTTP_ANY, std::bind(&Main::showUptime, this, std::placeholders::_1)); mWeb->on("/time", HTTP_ANY, std::bind(&Main::showTime, this, std::placeholders::_1)); mWeb->on("/style.css", HTTP_ANY, std::bind(&Main::showCss, this, std::placeholders::_1)); mWeb->on("/favicon.ico", HTTP_ANY, std::bind(&Main::showFavicon, this, std::placeholders::_1)); mWeb->on("/reboot", HTTP_ANY, std::bind(&Main::showReboot, this, std::placeholders::_1)); mWeb->on("/factory", HTTP_ANY, std::bind(&Main::showFactoryRst, this, std::placeholders::_1)); mWeb->on("/update", HTTP_GET, std::bind(&Main::showUpdateForm, this, std::placeholders::_1)); mWeb->on("/update", HTTP_POST, std::bind(&Main::showUpdate, this, std::placeholders::_1), std::bind(&Main::showUpdate2, this, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3, std::placeholders::_4, std::placeholders::_5, std::placeholders::_6)); mWeb->onNotFound( std::bind(&Main::showNotFound, this, std::placeholders::_1)); startAp = getConfig(); #ifndef AP_ONLY if(false == startAp) startAp = setupStation(timeout); #endif mApActive = startAp; } //----------------------------------------------------------------------------- void Main::loop(void) { //DPRINTLN(DBG_VERBOSE, F("M")); if(mApActive) { mDns->processNextRequest(); #ifndef AP_ONLY if(checkTicker(&mNextTryTs, (WIFI_AP_ACTIVE_TIME * 1000))) { mApActive = setupStation(mLimit); if(mApActive) { if(strlen(WIFI_AP_PWD) < 8) DPRINTLN(DBG_ERROR, F("password must be at least 8 characters long")); mApLastTick = millis(); mNextTryTs = (millis() + (WIFI_AP_ACTIVE_TIME * 1000)); setupAp(WIFI_AP_SSID, WIFI_AP_PWD); } } else { if(millis() - mApLastTick > 10000) { uint8_t cnt = WiFi.softAPgetStationNum(); if(cnt > 0) { DPRINTLN(DBG_INFO, String(cnt) + F(" clients connected, resetting AP timeout")); mNextTryTs = (millis() + (WIFI_AP_ACTIVE_TIME * 1000)); } mApLastTick = millis(); DPRINTLN(DBG_INFO, F("AP will be closed in ") + String((mNextTryTs - mApLastTick) / 1000) + F(" seconds")); } } #endif } //mWeb->handleClient(); if(checkTicker(&mUptimeTicker, mUptimeInterval)) { mUptimeSecs++; if(0 != mTimestamp) mTimestamp++; else { if(!mApActive) { mTimestamp = getNtpTime(); DPRINTLN(DBG_INFO, "[NTP]: " + getDateTimeStr(mTimestamp)); } } /*if(++mHeapStatCnt >= 10) { mHeapStatCnt = 0; stats(); }*/ } if(mShouldReboot) { Serial.println("Rebooting..."); delay(100); ESP.restart(); } } //----------------------------------------------------------------------------- bool Main::getConfig(void) { DPRINTLN(DBG_VERBOSE, F("Main::getConfig")); bool mApActive = false; mWifiSettingsValid = checkEEpCrc(ADDR_START, ADDR_WIFI_CRC, ADDR_WIFI_CRC); mSettingsValid = checkEEpCrc(ADDR_START_SETTINGS, ((ADDR_NEXT)-(ADDR_START_SETTINGS)), ADDR_SETTINGS_CRC); if(mWifiSettingsValid) { mEep->read(ADDR_SSID, mStationSsid, SSID_LEN); mEep->read(ADDR_PWD, mStationPwd, PWD_LEN); mEep->read(ADDR_DEVNAME, mDeviceName, DEVNAME_LEN); } if((!mWifiSettingsValid) || (mStationSsid[0] == 0xff)) { snprintf(mStationSsid, SSID_LEN, "%s", FB_WIFI_SSID); snprintf(mStationPwd, PWD_LEN, "%s", FB_WIFI_PWD); snprintf(mDeviceName, DEVNAME_LEN, "%s", DEF_DEVICE_NAME); } return mApActive; } //----------------------------------------------------------------------------- void Main::setupAp(const char *ssid, const char *pwd) { DPRINTLN(DBG_VERBOSE, F("Main::setupAp")); IPAddress apIp(192, 168, 1, 1); DPRINTLN(DBG_INFO, F("\n---------\nAP MODE\nSSID: ") + String(ssid) + F("\nPWD: ") + String(pwd) + F("\nActive for: ") + String(WIFI_AP_ACTIVE_TIME) + F(" seconds") + F("\n---------\n")); DPRINTLN(DBG_DEBUG, String(mNextTryTs)); WiFi.mode(WIFI_AP); WiFi.softAPConfig(apIp, apIp, IPAddress(255, 255, 255, 0)); WiFi.softAP(ssid, pwd); mDns->start(mDnsPort, "*", apIp); /*mWeb->onNotFound([&]() { showSetup(); }); mWeb->on("/", std::bind(&Main::showSetup, this));*/ mWeb->begin(); } //----------------------------------------------------------------------------- bool Main::setupStation(uint32_t timeout) { DPRINTLN(DBG_VERBOSE, F("Main::setupStation")); int32_t cnt; bool startAp = false; if(timeout >= 3) cnt = (timeout - 3) / 2 * 10; else { timeout = 1; cnt = 1; } WiFi.mode(WIFI_STA); WiFi.begin(mStationSsid, mStationPwd); if(String(mDeviceName) != "") WiFi.hostname(mDeviceName); delay(2000); DPRINTLN(DBG_INFO, F("connect to network '") + String(mStationSsid) + F("' ...")); while (WiFi.status() != WL_CONNECTED) { delay(100); if(cnt % 100 == 0) Serial.println("."); else Serial.print("."); if(timeout > 0) { // limit == 0 -> no limit if(--cnt <= 0) { if(WiFi.status() != WL_CONNECTED) { startAp = true; WiFi.disconnect(); } delay(100); break; } } } Serial.println("."); if(false == startAp) { mWeb->begin(); } delay(1000); return startAp; } //----------------------------------------------------------------------------- void Main::showSetup(AsyncWebServerRequest *request) { DPRINTLN(DBG_VERBOSE, F("Main::showSetup")); String html = FPSTR(setup_html); html.replace(F("{SSID}"), mStationSsid); // PWD will be left at the default value (for protection) // -> the PWD will only be changed if it does not match the default "{PWD}" html.replace(F("{DEVICE}"), String(mDeviceName)); html.replace(F("{VERSION}"), String(mVersion)); if(mApActive) html.replace("{IP}", String(F("http://192.168.1.1"))); else html.replace("{IP}", (F("http://") + String(WiFi.localIP().toString()))); request->send(200, F("text/html"), html); } //----------------------------------------------------------------------------- void Main::showCss(AsyncWebServerRequest *request) { DPRINTLN(DBG_VERBOSE, F("Main::showCss")); request->send(200, "text/css", FPSTR(style_css)); } //----------------------------------------------------------------------------- void Main::showFavicon(AsyncWebServerRequest *request) { DPRINTLN(DBG_VERBOSE, F("app::showFavicon")); static const char favicon_type[] PROGMEM = "image/x-icon"; static const unsigned char favicon_content[] PROGMEM = FAVICON_PANEL_57; // mWeb->send_P(200, favicon_type, favicon_content, sizeof(favicon_content)); AsyncWebServerResponse *response = request->beginResponse_P(200, favicon_type, favicon_content, sizeof(favicon_content)); // response->addHeader(F("Content-Encoding"), "gzip"); request->send(response); } //----------------------------------------------------------------------------- void Main::showSave(AsyncWebServerRequest *request) { DPRINTLN(DBG_VERBOSE, F("Main::showSave")); saveValues(request, true); } //----------------------------------------------------------------------------- void Main::saveValues(AsyncWebServerRequest *request, bool webSend = true) { DPRINTLN(DBG_VERBOSE, F("Main::saveValues")); if(request->args() > 0) { if(request->arg("ssid") != "") { memset(mStationSsid, 0, SSID_LEN); request->arg("ssid").toCharArray(mStationSsid, SSID_LEN); mEep->write(ADDR_SSID, mStationSsid, SSID_LEN); if(request->arg("pwd") != "{PWD}") { memset(mStationPwd, 0, PWD_LEN); request->arg("pwd").toCharArray(mStationPwd, PWD_LEN); mEep->write(ADDR_PWD, mStationPwd, PWD_LEN); } } memset(mDeviceName, 0, DEVNAME_LEN); request->arg("device").toCharArray(mDeviceName, DEVNAME_LEN); mEep->write(ADDR_DEVNAME, mDeviceName, DEVNAME_LEN); mEep->commit(); updateCrc(); if(webSend) { if(request->arg("reboot") == "on") showReboot(request); else // TODO: add device name as redirect in AP-mode request->send(200, F("text/html"), F("
saved
")); } } } //----------------------------------------------------------------------------- void Main::updateCrc(void) { DPRINTLN(DBG_VERBOSE, F("Main::updateCrc")); uint16_t crc; crc = buildEEpCrc(ADDR_START, ADDR_WIFI_CRC); //Serial.println("new CRC: " + String(crc, HEX)); mEep->write(ADDR_WIFI_CRC, crc); mEep->commit(); } //----------------------------------------------------------------------------- void Main::showUptime(AsyncWebServerRequest *request) { //DPRINTLN(DBG_VERBOSE, F("Main::showUptime")); char time[20] = {0}; int upTimeSc = uint32_t((mUptimeSecs) % 60); int upTimeMn = uint32_t((mUptimeSecs / (60)) % 60); int upTimeHr = uint32_t((mUptimeSecs / (60 * 60)) % 24); int upTimeDy = uint32_t((mUptimeSecs / (60 * 60 * 24)) % 365); snprintf(time, 20, "%d Tage, %02d:%02d:%02d", upTimeDy, upTimeHr, upTimeMn, upTimeSc); request->send(200, "text/plain", String(time)); } //----------------------------------------------------------------------------- void Main::showTime(AsyncWebServerRequest *request) { //DPRINTLN(DBG_VERBOSE, F("Main::showTime")); request->send(200, "text/plain", getDateTimeStr(mTimestamp)); } //----------------------------------------------------------------------------- void Main::showNotFound(AsyncWebServerRequest *request) { DPRINTLN(DBG_VERBOSE, F("Main::showNotFound - ") + request->url()); String msg = F("File Not Found\n\nURL: "); msg += request->url(); msg += F("\nMethod: "); msg += ( request->method() == HTTP_GET ) ? "GET" : "POST"; msg += F("\nArguments: "); msg += request->args(); msg += "\n"; for(uint8_t i = 0; i < request->args(); i++ ) { msg += " " + request->argName(i) + ": " + request->arg(i) + "\n"; } request->send(404, F("text/plain"), msg); } //----------------------------------------------------------------------------- void Main::showReboot(AsyncWebServerRequest *request) { DPRINTLN(DBG_VERBOSE, F("Main::showReboot")); request->send(200, F("text/html"), F("