ESPAsyncWiFiManager.cpp 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915
  1. /**************************************************************
  2. AsyncWiFiManager is a library for the ESP8266/Arduino platform
  3. (https://github.com/esp8266/Arduino) to enable easy
  4. configuration and reconfiguration of WiFi credentials using a Captive Portal
  5. inspired by:
  6. http://www.esp8266.com/viewtopic.php?f=29&t=2520
  7. https://github.com/chriscook8/esp-arduino-apboot
  8. https://github.com/esp8266/Arduino/tree/esp8266/hardware/esp8266com/esp8266/libraries/DNSServer/examples/CaptivePortalAdvanced
  9. Built by AlexT https://github.com/tzapu
  10. Licensed under MIT license
  11. **************************************************************/
  12. #include "ESPAsyncWiFiManager.h"
  13. AsyncWiFiManagerParameter::AsyncWiFiManagerParameter(const char *custom) {
  14. _id = NULL;
  15. _placeholder = NULL;
  16. _length = 0;
  17. _value = NULL;
  18. _customHTML = custom;
  19. }
  20. AsyncWiFiManagerParameter::AsyncWiFiManagerParameter(const char *id, const char *placeholder, const char *defaultValue, int length) {
  21. init(id, placeholder, defaultValue, length, "");
  22. }
  23. AsyncWiFiManagerParameter::AsyncWiFiManagerParameter(const char *id, const char *placeholder, const char *defaultValue, int length, const char *custom) {
  24. init(id, placeholder, defaultValue, length, custom);
  25. }
  26. void AsyncWiFiManagerParameter::init(const char *id, const char *placeholder, const char *defaultValue, int length, const char *custom) {
  27. _id = id;
  28. _placeholder = placeholder;
  29. _length = length;
  30. _value = new char[length + 1];
  31. for (int i = 0; i < length; i++) {
  32. _value[i] = 0;
  33. }
  34. if (defaultValue != NULL) {
  35. strncpy(_value, defaultValue, length);
  36. }
  37. _customHTML = custom;
  38. }
  39. const char* AsyncWiFiManagerParameter::getValue() {
  40. return _value;
  41. }
  42. const char* AsyncWiFiManagerParameter::getID() {
  43. return _id;
  44. }
  45. const char* AsyncWiFiManagerParameter::getPlaceholder() {
  46. return _placeholder;
  47. }
  48. int AsyncWiFiManagerParameter::getValueLength() {
  49. return _length;
  50. }
  51. const char* AsyncWiFiManagerParameter::getCustomHTML() {
  52. return _customHTML;
  53. }
  54. AsyncWiFiManager::AsyncWiFiManager(AsyncWebServer *server, DNSServer *dns) :server(server), dnsServer(dns) {
  55. wifiSSIDs = NULL;
  56. wifiSSIDscan=true;
  57. _modeless=false;
  58. shouldscan=true;
  59. }
  60. void AsyncWiFiManager::addParameter(AsyncWiFiManagerParameter *p) {
  61. _params[_paramsCount] = p;
  62. _paramsCount++;
  63. DEBUG_WM("Adding parameter");
  64. DEBUG_WM(p->getID());
  65. }
  66. void AsyncWiFiManager::setupConfigPortal() {
  67. // dnsServer.reset(new DNSServer());
  68. // server.reset(new ESP8266WebServer(80));
  69. DEBUG_WM(F(""));
  70. _configPortalStart = millis();
  71. DEBUG_WM(F("Configuring access point... "));
  72. DEBUG_WM(_apName);
  73. if (_apPassword != NULL) {
  74. if (strlen(_apPassword) < 8 || strlen(_apPassword) > 63) {
  75. // fail passphrase to short or long!
  76. DEBUG_WM(F("Invalid AccessPoint password. Ignoring"));
  77. _apPassword = NULL;
  78. }
  79. DEBUG_WM(_apPassword);
  80. }
  81. //optional soft ip config
  82. if (_ap_static_ip) {
  83. DEBUG_WM(F("Custom AP IP/GW/Subnet"));
  84. WiFi.softAPConfig(_ap_static_ip, _ap_static_gw, _ap_static_sn);
  85. }
  86. if (_apPassword != NULL) {
  87. WiFi.softAP(_apName, _apPassword);//password option
  88. } else {
  89. WiFi.softAP(_apName);
  90. }
  91. delay(500); // Without delay I've seen the IP address blank
  92. DEBUG_WM(F("AP IP address: "));
  93. DEBUG_WM(WiFi.softAPIP());
  94. /* Setup the DNS server redirecting all the domains to the apIP */
  95. dnsServer->setErrorReplyCode(DNSReplyCode::NoError);
  96. dnsServer->start(DNS_PORT, "*", WiFi.softAPIP());
  97. /* Setup web pages: root, wifi config pages, SO captive portal detectors and not found. */
  98. server->on("/", std::bind(&AsyncWiFiManager::handleRoot, this,std::placeholders::_1)).setFilter(ON_AP_FILTER);
  99. server->on("/wifi", std::bind(&AsyncWiFiManager::handleWifi, this, std::placeholders::_1,true)).setFilter(ON_AP_FILTER);
  100. server->on("/0wifi", std::bind(&AsyncWiFiManager::handleWifi, this,std::placeholders::_1, false)).setFilter(ON_AP_FILTER);
  101. server->on("/wifisave", std::bind(&AsyncWiFiManager::handleWifiSave,this,std::placeholders::_1)).setFilter(ON_AP_FILTER);
  102. server->on("/i", std::bind(&AsyncWiFiManager::handleInfo,this, std::placeholders::_1)).setFilter(ON_AP_FILTER);
  103. server->on("/r", std::bind(&AsyncWiFiManager::handleReset, this,std::placeholders::_1)).setFilter(ON_AP_FILTER);
  104. //server->on("/generate_204", std::bind(&AsyncWiFiManager::handle204, this)); //Android/Chrome OS captive portal check.
  105. server->on("/fwlink", std::bind(&AsyncWiFiManager::handleRoot, this,std::placeholders::_1)).setFilter(ON_AP_FILTER); //Microsoft captive portal. Maybe not needed. Might be handled by notFound handler.
  106. server->onNotFound (std::bind(&AsyncWiFiManager::handleNotFound,this,std::placeholders::_1));
  107. server->begin(); // Web server start
  108. DEBUG_WM(F("HTTP server started"));
  109. }
  110. boolean AsyncWiFiManager::autoConnect() {
  111. String ssid = "ESP" + String(ESP.getChipId());
  112. return autoConnect(ssid.c_str(), NULL);
  113. }
  114. boolean AsyncWiFiManager::autoConnect(char const *apName, char const *apPassword) {
  115. DEBUG_WM(F(""));
  116. DEBUG_WM(F("AutoConnect"));
  117. // read eeprom for ssid and pass
  118. //String ssid = getSSID();
  119. //String pass = getPassword();
  120. // attempt to connect; should it fail, fall back to AP
  121. WiFi.mode(WIFI_STA);
  122. if (connectWifi("", "") == WL_CONNECTED) {
  123. DEBUG_WM(F("IP Address:"));
  124. DEBUG_WM(WiFi.localIP());
  125. //connected
  126. return true;
  127. }
  128. return startConfigPortal(apName, apPassword);
  129. }
  130. void AsyncWiFiManager::scan()
  131. {
  132. if (!shouldscan) return;
  133. DEBUG_WM(F("About to scan()"));
  134. if (wifiSSIDscan)
  135. {
  136. delay(100);
  137. }
  138. if (wifiSSIDscan)
  139. {
  140. int n = WiFi.scanNetworks();
  141. DEBUG_WM(F("Scan done"));
  142. if (n == 0) {
  143. DEBUG_WM(F("No networks found"));
  144. // page += F("No networks found. Refresh to scan again.");
  145. } else {
  146. if (wifiSSIDscan)
  147. {
  148. /* WE SHOULD MOVE THIS IN PLACE ATOMICALLY */
  149. if (wifiSSIDs) delete [] wifiSSIDs;
  150. wifiSSIDs = new WiFiResult[n];
  151. wifiSSIDCount = n;
  152. if (n>0)
  153. shouldscan=false;
  154. for (int i=0;i<n;i++)
  155. {
  156. wifiSSIDs[i].duplicate=false;
  157. bool res=WiFi.getNetworkInfo(i, wifiSSIDs[i].SSID, wifiSSIDs[i].encryptionType, wifiSSIDs[i].RSSI, wifiSSIDs[i].BSSID, wifiSSIDs[i].channel, wifiSSIDs[i].isHidden);
  158. }
  159. // RSSI SORT
  160. // old sort
  161. for (int i = 0; i < n; i++) {
  162. for (int j = i + 1; j < n; j++) {
  163. if (wifiSSIDs[j].RSSI > wifiSSIDs[i].RSSI) {
  164. std::swap(wifiSSIDs[i], wifiSSIDs[j]);
  165. }
  166. }
  167. }
  168. // remove duplicates ( must be RSSI sorted )
  169. if (_removeDuplicateAPs) {
  170. String cssid;
  171. for (int i = 0; i < n; i++) {
  172. if (wifiSSIDs[i].duplicate == true) continue;
  173. cssid = wifiSSIDs[i].SSID;
  174. for (int j = i + 1; j < n; j++) {
  175. if (cssid == wifiSSIDs[j].SSID) {
  176. DEBUG_WM("DUP AP: " +wifiSSIDs[j].SSID);
  177. wifiSSIDs[j].duplicate=true; // set dup aps to NULL
  178. }
  179. }
  180. }
  181. }
  182. }
  183. }
  184. }
  185. }
  186. void AsyncWiFiManager::startConfigPortalModeless(char const *apName, char const *apPassword) {
  187. _modeless =true;
  188. _apName = apName;
  189. _apPassword = apPassword;
  190. /*
  191. AJS - do we want this?
  192. */
  193. //setup AP
  194. WiFi.mode(WIFI_AP_STA);
  195. DEBUG_WM("SET AP STA");
  196. // try to connect
  197. if (connectWifi("", "") == WL_CONNECTED) {
  198. DEBUG_WM(F("IP Address:"));
  199. DEBUG_WM(WiFi.localIP());
  200. //connected
  201. // call the callback!
  202. _savecallback();
  203. }
  204. //notify we entered AP mode
  205. if ( _apcallback != NULL) {
  206. _apcallback(this);
  207. }
  208. connect = false;
  209. setupConfigPortal();
  210. scannow= -1 ;
  211. }
  212. void AsyncWiFiManager::loop(){
  213. dnsServer->processNextRequest();
  214. if (_modeless)
  215. {
  216. if ( scannow==-1 || millis() > scannow + 60000)
  217. {
  218. scan();
  219. scannow= millis() ;
  220. }
  221. if (connect) {
  222. connect = false;
  223. //delay(2000);
  224. DEBUG_WM(F("Connecting to new AP"));
  225. // using user-provided _ssid, _pass in place of system-stored ssid and pass
  226. if (connectWifi(_ssid, _pass) != WL_CONNECTED) {
  227. DEBUG_WM(F("Failed to connect."));
  228. } else {
  229. //connected
  230. // alanswx - should we have a config to decide if we should shut down AP?
  231. // WiFi.mode(WIFI_STA);
  232. //notify that configuration has changed and any optional parameters should be saved
  233. if ( _savecallback != NULL) {
  234. //todo: check if any custom parameters actually exist, and check if they really changed maybe
  235. _savecallback();
  236. }
  237. return;
  238. }
  239. if (_shouldBreakAfterConfig) {
  240. //flag set to exit after config after trying to connect
  241. //notify that configuration has changed and any optional parameters should be saved
  242. if ( _savecallback != NULL) {
  243. //todo: check if any custom parameters actually exist, and check if they really changed maybe
  244. _savecallback();
  245. }
  246. return;
  247. }
  248. }
  249. }
  250. }
  251. boolean AsyncWiFiManager::startConfigPortal(char const *apName, char const *apPassword) {
  252. //setup AP
  253. WiFi.mode(WIFI_AP_STA);
  254. DEBUG_WM("SET AP STA");
  255. _apName = apName;
  256. _apPassword = apPassword;
  257. //notify we entered AP mode
  258. if ( _apcallback != NULL) {
  259. _apcallback(this);
  260. }
  261. connect = false;
  262. setupConfigPortal();
  263. scannow= -1 ;
  264. while (_configPortalTimeout == 0 || millis() < _configPortalStart + _configPortalTimeout) {
  265. //DNS
  266. dnsServer->processNextRequest();
  267. //
  268. // we should do a scan every so often here
  269. //
  270. if ( millis() > scannow + 10000)
  271. {
  272. DEBUG_WM(F("About to scan()"));
  273. shouldscan=true; // since we are modal, we can scan every time
  274. scan();
  275. scannow= millis() ;
  276. }
  277. if (connect) {
  278. connect = false;
  279. delay(2000);
  280. DEBUG_WM(F("Connecting to new AP"));
  281. // using user-provided _ssid, _pass in place of system-stored ssid and pass
  282. if (connectWifi(_ssid, _pass) != WL_CONNECTED) {
  283. DEBUG_WM(F("Failed to connect."));
  284. } else {
  285. //connected
  286. WiFi.mode(WIFI_STA);
  287. //notify that configuration has changed and any optional parameters should be saved
  288. if ( _savecallback != NULL) {
  289. //todo: check if any custom parameters actually exist, and check if they really changed maybe
  290. _savecallback();
  291. }
  292. break;
  293. }
  294. if (_shouldBreakAfterConfig) {
  295. //flag set to exit after config after trying to connect
  296. //notify that configuration has changed and any optional parameters should be saved
  297. if ( _savecallback != NULL) {
  298. //todo: check if any custom parameters actually exist, and check if they really changed maybe
  299. _savecallback();
  300. }
  301. break;
  302. }
  303. }
  304. yield();
  305. }
  306. // server.reset();
  307. // dnsServer.reset();
  308. return WiFi.status() == WL_CONNECTED;
  309. }
  310. int AsyncWiFiManager::connectWifi(String ssid, String pass) {
  311. DEBUG_WM(F("Connecting as wifi client..."));
  312. // check if we've got static_ip settings, if we do, use those.
  313. if (_sta_static_ip) {
  314. DEBUG_WM(F("Custom STA IP/GW/Subnet"));
  315. WiFi.config(_sta_static_ip, _sta_static_gw, _sta_static_sn);
  316. DEBUG_WM(WiFi.localIP());
  317. }
  318. //fix for auto connect racing issue
  319. // if (WiFi.status() == WL_CONNECTED) {
  320. // DEBUG_WM("Already connected. Bailing out.");
  321. // return WL_CONNECTED;
  322. // }
  323. //check if we have ssid and pass and force those, if not, try with last saved values
  324. if (ssid != "") {
  325. WiFi.begin(ssid.c_str(), pass.c_str());
  326. } else {
  327. if (WiFi.SSID()) {
  328. DEBUG_WM("Using last saved values, should be faster");
  329. //trying to fix connection in progress hanging
  330. ETS_UART_INTR_DISABLE();
  331. wifi_station_disconnect();
  332. ETS_UART_INTR_ENABLE();
  333. WiFi.begin();
  334. } else {
  335. DEBUG_WM("No saved credentials");
  336. }
  337. }
  338. int connRes = waitForConnectResult();
  339. DEBUG_WM ("Connection result: ");
  340. DEBUG_WM ( connRes );
  341. //not connected, WPS enabled, no pass - first attempt
  342. if (_tryWPS && connRes != WL_CONNECTED && pass == "") {
  343. startWPS();
  344. //should be connected at the end of WPS
  345. connRes = waitForConnectResult();
  346. }
  347. return connRes;
  348. }
  349. uint8_t AsyncWiFiManager::waitForConnectResult() {
  350. if (_connectTimeout == 0) {
  351. return WiFi.waitForConnectResult();
  352. } else {
  353. DEBUG_WM (F("Waiting for connection result with time out"));
  354. unsigned long start = millis();
  355. boolean keepConnecting = true;
  356. uint8_t status;
  357. while (keepConnecting) {
  358. status = WiFi.status();
  359. if (millis() > start + _connectTimeout) {
  360. keepConnecting = false;
  361. DEBUG_WM (F("Connection timed out"));
  362. }
  363. if (status == WL_CONNECTED || status == WL_CONNECT_FAILED) {
  364. keepConnecting = false;
  365. }
  366. delay(100);
  367. }
  368. return status;
  369. }
  370. }
  371. void AsyncWiFiManager::startWPS() {
  372. DEBUG_WM("START WPS");
  373. WiFi.beginWPSConfig();
  374. DEBUG_WM("END WPS");
  375. }
  376. /*
  377. String AsyncWiFiManager::getSSID() {
  378. if (_ssid == "") {
  379. DEBUG_WM(F("Reading SSID"));
  380. _ssid = WiFi.SSID();
  381. DEBUG_WM(F("SSID: "));
  382. DEBUG_WM(_ssid);
  383. }
  384. return _ssid;
  385. }
  386. String AsyncWiFiManager::getPassword() {
  387. if (_pass == "") {
  388. DEBUG_WM(F("Reading Password"));
  389. _pass = WiFi.psk();
  390. DEBUG_WM("Password: " + _pass);
  391. //DEBUG_WM(_pass);
  392. }
  393. return _pass;
  394. }
  395. */
  396. String AsyncWiFiManager::getConfigPortalSSID() {
  397. return _apName;
  398. }
  399. void AsyncWiFiManager::resetSettings() {
  400. DEBUG_WM(F("settings invalidated"));
  401. DEBUG_WM(F("THIS MAY CAUSE AP NOT TO START UP PROPERLY. YOU NEED TO COMMENT IT OUT AFTER ERASING THE DATA."));
  402. WiFi.disconnect(true);
  403. //delay(200);
  404. }
  405. void AsyncWiFiManager::setTimeout(unsigned long seconds) {
  406. setConfigPortalTimeout(seconds);
  407. }
  408. void AsyncWiFiManager::setConfigPortalTimeout(unsigned long seconds) {
  409. _configPortalTimeout = seconds * 1000;
  410. }
  411. void AsyncWiFiManager::setConnectTimeout(unsigned long seconds) {
  412. _connectTimeout = seconds * 1000;
  413. }
  414. void AsyncWiFiManager::setDebugOutput(boolean debug) {
  415. _debug = debug;
  416. }
  417. void AsyncWiFiManager::setAPStaticIPConfig(IPAddress ip, IPAddress gw, IPAddress sn) {
  418. _ap_static_ip = ip;
  419. _ap_static_gw = gw;
  420. _ap_static_sn = sn;
  421. }
  422. void AsyncWiFiManager::setSTAStaticIPConfig(IPAddress ip, IPAddress gw, IPAddress sn) {
  423. _sta_static_ip = ip;
  424. _sta_static_gw = gw;
  425. _sta_static_sn = sn;
  426. }
  427. void AsyncWiFiManager::setMinimumSignalQuality(int quality) {
  428. _minimumQuality = quality;
  429. }
  430. void AsyncWiFiManager::setBreakAfterConfig(boolean shouldBreak) {
  431. _shouldBreakAfterConfig = shouldBreak;
  432. }
  433. /** Handle root or redirect to captive portal */
  434. void AsyncWiFiManager::handleRoot(AsyncWebServerRequest *request) {
  435. // AJS - maybe we should set a scan when we get to the root???
  436. // and only scan on demand? timer + on demand? plus a link to make it happen?
  437. shouldscan=true;
  438. scannow= -1 ;
  439. DEBUG_WM(F("Handle root"));
  440. if (captivePortal(request)) { // If captive portal redirect instead of displaying the page.
  441. return;
  442. }
  443. String page = FPSTR(WFM_HTTP_HEAD);
  444. page.replace("{v}", "Options");
  445. page += FPSTR(HTTP_SCRIPT);
  446. page += FPSTR(HTTP_STYLE);
  447. page += _customHeadElement;
  448. page += FPSTR(HTTP_HEAD_END);
  449. page += "<h1>";
  450. page += _apName;
  451. page += "</h1>";
  452. page += F("<h3>AsyncWiFiManager</h3>");
  453. page += FPSTR(HTTP_PORTAL_OPTIONS);
  454. page += FPSTR(HTTP_END);
  455. request->send(200, "text/html", page);
  456. }
  457. /** Wifi config page handler */
  458. void AsyncWiFiManager::handleWifi(AsyncWebServerRequest *request,boolean scan) {
  459. shouldscan=true;
  460. scannow= -1 ;
  461. String page = FPSTR(WFM_HTTP_HEAD);
  462. page.replace("{v}", "Config ESP");
  463. page += FPSTR(HTTP_SCRIPT);
  464. page += FPSTR(HTTP_STYLE);
  465. page += _customHeadElement;
  466. page += FPSTR(HTTP_HEAD_END);
  467. if (scan) {
  468. wifiSSIDscan=false;
  469. int n = wifiSSIDCount;
  470. DEBUG_WM(F("Scan done"));
  471. if (n == 0) {
  472. DEBUG_WM(F("No networks found"));
  473. page += F("No networks found. Refresh to scan again.");
  474. } else {
  475. //display networks in page
  476. for (int i = 0; i < n; i++) {
  477. if (wifiSSIDs[i].duplicate == true) continue; // skip dups
  478. DEBUG_WM(wifiSSIDs[i].SSID);
  479. DEBUG_WM(wifiSSIDs[i].RSSI);
  480. int quality = getRSSIasQuality(wifiSSIDs[i].RSSI);
  481. if (_minimumQuality == -1 || _minimumQuality < quality) {
  482. String item = FPSTR(HTTP_ITEM);
  483. String rssiQ;
  484. rssiQ += quality;
  485. item.replace("{v}", wifiSSIDs[i].SSID);
  486. item.replace("{r}", rssiQ);
  487. if (wifiSSIDs[i].encryptionType != ENC_TYPE_NONE) {
  488. item.replace("{i}", "l");
  489. } else {
  490. item.replace("{i}", "");
  491. }
  492. //DEBUG_WM(item);
  493. page += item;
  494. delay(0);
  495. } else {
  496. DEBUG_WM(F("Skipping due to quality"));
  497. }
  498. }
  499. page += "<br/>";
  500. }
  501. }
  502. wifiSSIDscan=true;
  503. page += FPSTR(HTTP_FORM_START);
  504. char parLength[2];
  505. // add the extra parameters to the form
  506. for (int i = 0; i < _paramsCount; i++) {
  507. if (_params[i] == NULL) {
  508. break;
  509. }
  510. String pitem = FPSTR(HTTP_FORM_PARAM);
  511. if (_params[i]->getID() != NULL) {
  512. pitem.replace("{i}", _params[i]->getID());
  513. pitem.replace("{n}", _params[i]->getID());
  514. pitem.replace("{p}", _params[i]->getPlaceholder());
  515. snprintf(parLength, 2, "%d", _params[i]->getValueLength());
  516. pitem.replace("{l}", parLength);
  517. pitem.replace("{v}", _params[i]->getValue());
  518. pitem.replace("{c}", _params[i]->getCustomHTML());
  519. } else {
  520. pitem = _params[i]->getCustomHTML();
  521. }
  522. page += pitem;
  523. }
  524. if (_params[0] != NULL) {
  525. page += "<br/>";
  526. }
  527. if (_sta_static_ip) {
  528. String item = FPSTR(HTTP_FORM_PARAM);
  529. item.replace("{i}", "ip");
  530. item.replace("{n}", "ip");
  531. item.replace("{p}", "Static IP");
  532. item.replace("{l}", "15");
  533. item.replace("{v}", _sta_static_ip.toString());
  534. page += item;
  535. item = FPSTR(HTTP_FORM_PARAM);
  536. item.replace("{i}", "gw");
  537. item.replace("{n}", "gw");
  538. item.replace("{p}", "Static Gateway");
  539. item.replace("{l}", "15");
  540. item.replace("{v}", _sta_static_gw.toString());
  541. page += item;
  542. item = FPSTR(HTTP_FORM_PARAM);
  543. item.replace("{i}", "sn");
  544. item.replace("{n}", "sn");
  545. item.replace("{p}", "Subnet");
  546. item.replace("{l}", "15");
  547. item.replace("{v}", _sta_static_sn.toString());
  548. page += item;
  549. page += "<br/>";
  550. }
  551. page += FPSTR(HTTP_FORM_END);
  552. page += FPSTR(HTTP_SCAN_LINK);
  553. page += FPSTR(HTTP_END);
  554. request->send(200, "text/html", page);
  555. DEBUG_WM(F("Sent config page"));
  556. }
  557. /** Handle the WLAN save form and redirect to WLAN config page again */
  558. void AsyncWiFiManager::handleWifiSave(AsyncWebServerRequest *request) {
  559. DEBUG_WM(F("WiFi save"));
  560. //SAVE/connect here
  561. _ssid = request->arg("s").c_str();
  562. _pass = request->arg("p").c_str();
  563. //parameters
  564. for (int i = 0; i < _paramsCount; i++) {
  565. if (_params[i] == NULL) {
  566. break;
  567. }
  568. //read parameter
  569. String value = request->arg(_params[i]->getID()).c_str();
  570. //store it in array
  571. value.toCharArray(_params[i]->_value, _params[i]->_length);
  572. DEBUG_WM(F("Parameter"));
  573. DEBUG_WM(_params[i]->getID());
  574. DEBUG_WM(value);
  575. }
  576. if (request->hasArg("ip")) {
  577. DEBUG_WM(F("static ip"));
  578. DEBUG_WM(request->arg("ip"));
  579. //_sta_static_ip.fromString(request->arg("ip"));
  580. String ip = request->arg("ip");
  581. optionalIPFromString(&_sta_static_ip, ip.c_str());
  582. }
  583. if (request->hasArg("gw")) {
  584. DEBUG_WM(F("static gateway"));
  585. DEBUG_WM(request->arg("gw"));
  586. String gw = request->arg("gw");
  587. optionalIPFromString(&_sta_static_gw, gw.c_str());
  588. }
  589. if (request->hasArg("sn")) {
  590. DEBUG_WM(F("static netmask"));
  591. DEBUG_WM(request->arg("sn"));
  592. String sn = request->arg("sn");
  593. optionalIPFromString(&_sta_static_sn, sn.c_str());
  594. }
  595. String page = FPSTR(WFM_HTTP_HEAD);
  596. page.replace("{v}", "Credentials Saved");
  597. page += FPSTR(HTTP_SCRIPT);
  598. page += FPSTR(HTTP_STYLE);
  599. page += _customHeadElement;
  600. page += F("<meta http-equiv=\"refresh\" content=\"5; url=/i\">");
  601. page += FPSTR(HTTP_HEAD_END);
  602. page += FPSTR(HTTP_SAVED);
  603. page += FPSTR(HTTP_END);
  604. request->send(200, "text/html", page);
  605. DEBUG_WM(F("Sent wifi save page"));
  606. connect = true; //signal ready to connect/reset
  607. }
  608. /** Handle the info page */
  609. void AsyncWiFiManager::handleInfo(AsyncWebServerRequest *request) {
  610. DEBUG_WM(F("Info"));
  611. String page = FPSTR(WFM_HTTP_HEAD);
  612. page.replace("{v}", "Info");
  613. page += FPSTR(HTTP_SCRIPT);
  614. page += FPSTR(HTTP_STYLE);
  615. page += _customHeadElement;
  616. if (connect==true)
  617. page += F("<meta http-equiv=\"refresh\" content=\"5; url=/i\">");
  618. page += FPSTR(HTTP_HEAD_END);
  619. page += F("<dl>");
  620. if (connect==true)
  621. {
  622. page += F("<dt>Trying to connect</dt><dd>");
  623. page += WiFi.status();
  624. page += F("</dd>");
  625. }
  626. page += F("<dt>Chip ID</dt><dd>");
  627. page += ESP.getChipId();
  628. page += F("</dd>");
  629. page += F("<dt>Flash Chip ID</dt><dd>");
  630. page += ESP.getFlashChipId();
  631. page += F("</dd>");
  632. page += F("<dt>IDE Flash Size</dt><dd>");
  633. page += ESP.getFlashChipSize();
  634. page += F(" bytes</dd>");
  635. page += F("<dt>Real Flash Size</dt><dd>");
  636. page += ESP.getFlashChipRealSize();
  637. page += F(" bytes</dd>");
  638. page += F("<dt>Soft AP IP</dt><dd>");
  639. page += WiFi.softAPIP().toString();
  640. page += F("</dd>");
  641. page += F("<dt>Soft AP MAC</dt><dd>");
  642. page += WiFi.softAPmacAddress();
  643. page += F("</dd>");
  644. page += F("<dt>Station SSID</dt><dd>");
  645. page += WiFi.SSID();
  646. page += F("</dd>");
  647. page += F("<dt>Station IP</dt><dd>");
  648. page += WiFi.localIP().toString();
  649. page += F("</dd>");
  650. page += F("<dt>Station MAC</dt><dd>");
  651. page += WiFi.macAddress();
  652. page += F("</dd>");
  653. page += F("</dl>");
  654. page += FPSTR(HTTP_END);
  655. request->send(200, "text/html", page);
  656. DEBUG_WM(F("Sent info page"));
  657. }
  658. /** Handle the reset page */
  659. void AsyncWiFiManager::handleReset(AsyncWebServerRequest *request) {
  660. DEBUG_WM(F("Reset"));
  661. String page = FPSTR(WFM_HTTP_HEAD);
  662. page.replace("{v}", "Info");
  663. page += FPSTR(HTTP_SCRIPT);
  664. page += FPSTR(HTTP_STYLE);
  665. page += _customHeadElement;
  666. page += FPSTR(HTTP_HEAD_END);
  667. page += F("Module will reset in a few seconds.");
  668. page += FPSTR(HTTP_END);
  669. request->send(200, "text/html", page);
  670. DEBUG_WM(F("Sent reset page"));
  671. delay(5000);
  672. ESP.reset();
  673. delay(2000);
  674. }
  675. //removed as mentioned here https://github.com/tzapu/AsyncWiFiManager/issues/114
  676. /*void AsyncWiFiManager::handle204(AsyncWebServerRequest *request) {
  677. DEBUG_WM(F("204 No Response"));
  678. request->sendHeader("Cache-Control", "no-cache, no-store, must-revalidate");
  679. request->sendHeader("Pragma", "no-cache");
  680. request->sendHeader("Expires", "-1");
  681. request->send ( 204, "text/plain", "");
  682. }*/
  683. void AsyncWiFiManager::handleNotFound(AsyncWebServerRequest *request) {
  684. if (captivePortal(request)) { // If captive portal redirect instead of displaying the error page.
  685. return;
  686. }
  687. String message = "File Not Found\n\n";
  688. message += "URI: ";
  689. message += request->url();
  690. message += "\nMethod: ";
  691. message += ( request->method() == HTTP_GET ) ? "GET" : "POST";
  692. message += "\nArguments: ";
  693. message += request->args();
  694. message += "\n";
  695. for ( uint8_t i = 0; i < request->args(); i++ ) {
  696. message += " " + request->argName ( i ) + ": " + request->arg ( i ) + "\n";
  697. }
  698. AsyncWebServerResponse *response = request->beginResponse(404,"text/plain",message);
  699. response->addHeader("Cache-Control", "no-cache, no-store, must-revalidate");
  700. response->addHeader("Pragma", "no-cache");
  701. response->addHeader("Expires", "-1");
  702. request->send (response );
  703. }
  704. /** Redirect to captive portal if we got a request for another domain. Return true in that case so the page handler do not try to handle the request again. */
  705. boolean AsyncWiFiManager::captivePortal(AsyncWebServerRequest *request) {
  706. if (!isIp(request->host()) ) {
  707. DEBUG_WM(F("Request redirected to captive portal"));
  708. AsyncWebServerResponse *response = request->beginResponse(302,"text/plain","");
  709. response->addHeader("Location", String("http://") + toStringIp(request->client()->localIP()));
  710. request->send ( response);
  711. return true;
  712. }
  713. return false;
  714. }
  715. //start up config portal callback
  716. void AsyncWiFiManager::setAPCallback( void (*func)(AsyncWiFiManager* myAsyncWiFiManager) ) {
  717. _apcallback = func;
  718. }
  719. //start up save config callback
  720. void AsyncWiFiManager::setSaveConfigCallback( void (*func)(void) ) {
  721. _savecallback = func;
  722. }
  723. //sets a custom element to add to head, like a new style tag
  724. void AsyncWiFiManager::setCustomHeadElement(const char* element) {
  725. _customHeadElement = element;
  726. }
  727. //if this is true, remove duplicated Access Points - defaut true
  728. void AsyncWiFiManager::setRemoveDuplicateAPs(boolean removeDuplicates) {
  729. _removeDuplicateAPs = removeDuplicates;
  730. }
  731. template <typename Generic>
  732. void AsyncWiFiManager::DEBUG_WM(Generic text) {
  733. if (_debug) {
  734. Serial.print("*WM: ");
  735. Serial.println(text);
  736. }
  737. }
  738. int AsyncWiFiManager::getRSSIasQuality(int RSSI) {
  739. int quality = 0;
  740. if (RSSI <= -100) {
  741. quality = 0;
  742. } else if (RSSI >= -50) {
  743. quality = 100;
  744. } else {
  745. quality = 2 * (RSSI + 100);
  746. }
  747. return quality;
  748. }
  749. /** Is this an IP? */
  750. boolean AsyncWiFiManager::isIp(String str) {
  751. for (int i = 0; i < str.length(); i++) {
  752. int c = str.charAt(i);
  753. if (c != '.' && (c < '0' || c > '9')) {
  754. return false;
  755. }
  756. }
  757. return true;
  758. }
  759. /** IP to String? */
  760. String AsyncWiFiManager::toStringIp(IPAddress ip) {
  761. String res = "";
  762. for (int i = 0; i < 3; i++) {
  763. res += String((ip >> (8 * i)) & 0xFF) + ".";
  764. }
  765. res += String(((ip >> 8 * 3)) & 0xFF);
  766. return res;
  767. }