httpServer.ino 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824
  1. //extern ESP8266WebServer httpServer;
  2. static const char httpRoot[] PROGMEM =
  3. R"(<html><body>
  4. <h1><span id='devname'></span></h1>
  5. <h3>WiFi Thermostat</h3>
  6. <form id='minusBtnFrm'>
  7. <input type='hidden' name='minusBtn' value='1'>
  8. </form>
  9. <form id='plusBtnFrm'>
  10. <input type='hidden' name='plusBtn' value='1'>
  11. </form>
  12. <form id='pset0BtnFrm'>
  13. <input type='hidden' name='pset0Btn' value='1'>
  14. </form>
  15. <form id='pset1BtnFrm'>
  16. <input type='hidden' name='pset1Btn' value='1'>
  17. </form>
  18. <form id='pset2BtnFrm'>
  19. <input type='hidden' name='pset2Btn' value='1'>
  20. </form>
  21. <form id='onBtnFrm'>
  22. <input type='hidden' name='onBtn' value='1'>
  23. </form>
  24. <form id='offBtnFrm'>
  25. <input type='hidden' name='offBtn' value='1'>
  26. </form>
  27. <span style="font-size:xx-large" id='setTemp'></span> &#8451;<br>
  28. <input type='button' onclick='return sendMinusBtn()' value='-'/>
  29. <input type='button' onclick='return sendPlusBtn()' value='+'/><br>
  30. Curr.setTemp: <span id='currSetTemp'></span><br><br>
  31. Preset: <span id='pset'></span><br>
  32. <input id='btnPs0' type='button' onclick='return sendPset0Btn()' value=''/>
  33. <input id='btnPs1' type='button' onclick='return sendPset1Btn()' value=''/>
  34. <input id='btnPs2' type='button' onclick='return sendPset2Btn()' value=''/>
  35. <br>
  36. <br>
  37. Mode: <span id='mode'></span><br>
  38. <input id='btn_on' type='button' onclick='return sendOnBtn()' value='HEAT'/>
  39. <input id='btn_off' type='button' onclick='return sendOffBtn()' value='OFF'/><br>
  40. <br>
  41. Current Temp/Hum: <span id='temp'></span> &#8451;&nbsp;&nbsp;&nbsp;<span id='hum'></span> %<br><br>
  42. heating currently <span id='heating'></span><br>
  43. <br>
  44. WiFi connected to <i><span id='ssid'></span></i>.<br>
  45. <h6>Last update
  46. <span id='ut'></span> seconds ago.
  47. <span id='status'></span>
  48. </h6>
  49. <br>
  50. <a href='/wifi.htm'>WiFi settings</a><br>
  51. <a href='/conf'>Base configuration</a><br>
  52. <a href='/conf2'>Extended configuration</a><br>
  53. <a href='/update'>Firmware update</a><br>
  54. <a href='/restart'>Restart</a>
  55. <script>
  56. function g(i) { return document.getElementById(i) };
  57. var xhttp, updateTime, reqTime, reqFin;
  58. function sendMinusBtn() {
  59. var form = document.getElementById('minusBtnFrm');
  60. return transmit(form);
  61. }
  62. function sendPlusBtn() {
  63. var form = document.getElementById('plusBtnFrm');
  64. return transmit(form);
  65. }
  66. function sendPset0Btn() {
  67. var form = document.getElementById('pset0BtnFrm');
  68. return transmit(form);
  69. }
  70. function sendPset1Btn() {
  71. var form = document.getElementById('pset1BtnFrm');
  72. return transmit(form);
  73. }
  74. function sendPset2Btn() {
  75. var form = document.getElementById('pset2BtnFrm');
  76. return transmit(form);
  77. }
  78. function sendOnBtn() {
  79. var form = document.getElementById('onBtnFrm');
  80. return transmit(form);
  81. }
  82. function sendOffBtn() {
  83. var form = document.getElementById('offBtnFrm');
  84. return transmit(form);
  85. }
  86. function transmit(f) {
  87. if (!xhttp) {
  88. g('status').innerHTML = 'loading...';
  89. reqTime = 0;
  90. reqFin = false;
  91. xhttp = new XMLHttpRequest();
  92. xhttp.timeout = 2000;
  93. xhttp.open('POST', 'api');
  94. xhttp.send(f ? (new FormData(f)) : '');
  95. xhttp.onreadystatechange = function () {
  96. if (xhttp.readyState === XMLHttpRequest.DONE && xhttp.status === 200) {
  97. var data = JSON.parse(xhttp.responseText);
  98. if(data.devname != undefined) g('devname').innerHTML = data.devname;
  99. g('temp').innerHTML = data.temp.toFixed(1);
  100. g('hum').innerHTML = data.hum;
  101. g('setTemp').innerHTML = data.setTemp.toFixed(1);
  102. g('ssid').innerHTML = data.ssid;
  103. g('mode').innerHTML = data.modeName;
  104. g('pset').innerHTML = data.psetName;
  105. g('btnPs0').value = data.psetName0;
  106. g('btnPs1').value = data.psetName1;
  107. g('btnPs2').value = data.psetName2;
  108. g('currSetTemp').innerHTML = data.currSetTemp.toFixed(1);
  109. if(data.heating == '1') g('heating').innerHTML = 'ACTIVE';
  110. else if(data.heating == '0') g('heating').innerHTML = 'NOT ACTIVE';
  111. xhttp = null;
  112. g('status').innerHTML = '';
  113. updateTime = 0;
  114. reqFin = true;
  115. }
  116. else {
  117. if(!reqFin && reqTime > 10) {
  118. xhttp = null;
  119. reqFin = true;
  120. }
  121. }
  122. }
  123. }
  124. return false;
  125. }
  126. transmit();
  127. setInterval(function () { g('ut').innerHTML = ++updateTime; ++reqTime; }, 1000);
  128. setInterval(transmit, 5000);
  129. </script>
  130. </body></html>)";
  131. static const char httpConfPage[] PROGMEM =
  132. R"(<html><head><body>
  133. <h3>Base configuration</h3>
  134. <a href='/'>Home</a><br><br>
  135. <input type='button' value='reload' onclick='return transmit()'/><br><br>
  136. <form id='form1' onsubmit='return transmit(this)'>
  137. Device Name: <input type='text' name='devName' id='devName'/><br><br>
  138. HTTP User *: <input type='text' name='httpUser' id='httpUser'/><br>
  139. HTTP Password *: <input type='text' name='httpPass' id='httpPass'/><br><br>
  140. HTTP set token: <input type='text' name='httpToken' id='httpToken'/><br>
  141. MQTT Server *: <input type='text' name='mqttHost' id='mqttHost'/><br>
  142. MQTT Port *: <input type='number' name='mqttPort' id='mqttPort'/><br>
  143. MQTT User *: <input type='text' name='mqttUser' id='mqttUser'/><br>
  144. MQTT Password *: <input type='text' name='mqttPass' id='mqttPass'/><br><br>
  145. In Topic *: <input type='text' name='inTop' id='inTop'/><br>
  146. Out Topic: <input type='text' name='outTop' id='outTop'/><br>
  147. Out Retain *: <input type='checkbox' name='outRet' id='outRet'/><br><br>
  148. LastWill Topic *: <input type='text' name='willTop' id='willTop'/><br>
  149. LastWill Qos *: <select name='willQos' id='willQos'><option>0</option><option>1</option><option>2</option></select><br>
  150. LastWill Retain *: <input type='checkbox' name='willRet' id='willRet'/><br>
  151. LastWill Message *: <input type='text' name='willMsg' id='willMsg'/><br><br>
  152. Connect Message *: <input type='text' name='connMsg' id='connMsg'/><br><br>
  153. Domoticz Out Topic *: <input type='text' name='domOutTop' id='domOutTop'/><br>
  154. <br>
  155. <input type='submit' value='Save'/>
  156. </form>
  157. <form id='restartForm' onsubmit='return res()'>
  158. <input type='hidden' name='restart' value='1'>
  159. <input type='submit' value='Restart'/>
  160. </form>
  161. <script>
  162. function g(i) { return document.getElementById(i) };
  163. var xhttp, reqTime, reqFin, rxhttp;
  164. function res() {
  165. rxhttp = new XMLHttpRequest();
  166. rxhttp.timeout = 1000;
  167. rxhttp.open('POST', 'restart');
  168. rxhttp.send('');
  169. rxhttp = null;
  170. return false;
  171. }
  172. function setCheckbox(ele, dat) {
  173. if(dat == 1) {
  174. ele.checked = true;
  175. ele.style.visibility = 'visible';
  176. }
  177. else {
  178. ele.checked = false;
  179. ele.style.visibility = 'visible';
  180. }
  181. }
  182. function updateCheckboxValue(ele) {
  183. if (ele.checked) ele.value ='1';
  184. else {
  185. ele.value = '0';
  186. ele.checked = true;
  187. ele.style.visibility = 'hidden';
  188. }
  189. }
  190. function transmit(f) {
  191. if (!xhttp) {
  192. reqTime = 0;
  193. reqFin = false;
  194. updateCheckboxValue(g('outRet'));
  195. updateCheckboxValue(g('willRet'));
  196. xhttp = new XMLHttpRequest();
  197. xhttp.timeout = 2000;
  198. xhttp.open('POST', 'confdata');
  199. xhttp.send(f ? (new FormData(f)) : '');
  200. xhttp.onreadystatechange = function () {
  201. if (xhttp.readyState === XMLHttpRequest.DONE && xhttp.status === 200) {
  202. var data = JSON.parse(xhttp.responseText);
  203. g('httpUser').value = data.httpUser;
  204. g('httpPass').value = data.httpPass;
  205. g('httpToken').value = data.httpToken;
  206. g('devName').value = data.devName
  207. g('mqttHost').value = data.mqttHost;
  208. g('mqttPort').value = data.mqttPort;
  209. g('mqttUser').value = data.mqttUser;
  210. g('mqttPass').value = data.mqttPass;
  211. g('inTop').value = data.inTop;
  212. g('outTop').value = data.outTop;
  213. g('willTop').value = data.willTop;
  214. g('willQos').value = data.willQos;
  215. setCheckbox(g('outRet'), data.outRet);
  216. setCheckbox(g('willRet'), data.willRet);
  217. g('willMsg').value = data.willMsg;
  218. g('connMsg').value = data.connMsg;
  219. g('domOutTop').value = data.domOutTop;
  220. xhttp = null;
  221. reqFin = true;
  222. }
  223. else {
  224. if(!reqFin && reqTime > 10) {
  225. xhttp = null;
  226. reqFin = true;
  227. }
  228. }
  229. }
  230. }
  231. return false;
  232. }
  233. transmit();
  234. setInterval(function () { ++reqTime; }, 1000);
  235. </script>
  236. </body></html>)";
  237. static const char httpConf2Page[] PROGMEM =
  238. R"(<html><head><body>
  239. <h3>Extended configuration</h3>
  240. <a href='/'>Home</a><br><br>
  241. <input type='button' value='reload' onclick='return transmit()'/><br>
  242. <form id='form1' onsubmit='return transmit(this)'>
  243. <h4>Domoticz</h4>
  244. Idx setTemp: <input type='number' name='domIdxTherm' id='domIdxTherm'/><br>
  245. Idx mode: <input type='number' name='domIdxMode' id='domIdxMode'/><br>
  246. Idx TempHum Sensor: <input type='number' name='domIdxTempHum' id='domIdxTempHum'/><br>
  247. Idx Heating: <input type='number' name='domIdxHeating' id='domIdxHeating'/><br>
  248. Idx PIR: <input type='number' name='domIdxPIR' id='domIdxPIR'/><br>
  249. <h4>Outside Temp/Hum via MQTT</h4>
  250. Temp In-Topic: <input type='text' name='outTempTop' id='outTempTop'/><br>
  251. Hum In-Topic: <input type='text' name='outHumTop' id='outHumTop'/><br>
  252. <h4>Additional MQTT publish topics</h4>
  253. PIR sensor: <input type='text' name='PIRTop' id='PIRTop'/><br>
  254. <h4>Auto-Save</h4>
  255. setTemp: <input type='checkbox' name='autoSaveTemp' id='autoSaveTemp'/><br>
  256. mode/preset: <input type='checkbox' name='autoSaveMode' id='autoSaveMode'/><br>
  257. <h4>Thermostat / Heating</h4>
  258. Min. Off-Time: <input type='number' name='minOffTime' id='minOffTime'/><br>
  259. Min. Temp: <input type='text' name='tempMin' id='tempMin'/><br>
  260. Max. Temp: <input type='text' name='tempMax' id='tempMax'/><br>
  261. Reduction Mode Temp: <input type='text' name='tempLow' id='tempLow'/><br>
  262. Reduction Mode 2 Temp: <input type='text' name='tempLow2' id='tempLow2'/><br>
  263. set temp decrease value: <input type='text' name='tempDec' id='tempDec'/><br>
  264. Hysteresis: <input type='text' name='hyst' id='hyst'/><br>
  265. Temp Correction.: <input type='text' name='tempCorr' id='tempCorr'/><br>
  266. Hum Correction: <input type='number' name='humCorr' id='humCorr'/><br>
  267. <h4>Intervals / Timeouts</h4>
  268. Measure Interval: <input type='number' name='measInt' id='measInt'/><br>
  269. Display Interval: <input type='number' name='dispInt' id='dispInt'/><br>
  270. Display Timeout: <input type='number' name='dispTout' id='dispTout'/><br>
  271. <h4>Names</h4>
  272. used in MQTT status/commands and HTTP interface<br>
  273. max. 13 chars!<br>
  274. Off-Message: <input type='text' name='offMsg' id='offMsg'/><br>
  275. <h5>Mode</h5>
  276. Off: <input type='text' name='modename0' id='modename0'/><br>
  277. On: <input type='text' name='modename1' id='modename1'/><br>
  278. <br>
  279. <h5>Presets:</h5>
  280. Normal*: <input type='text' name='psetname0' id='psetname0'/><br>
  281. Reduction 1: <input type='text' name='psetname1' id='psetname1'/><br>
  282. Reduction 2: <input type='text' name='psetname2' id='psetname2'/><br>
  283. * will always send/receive "none" as hardcoded in Home Assistant.
  284. <h4>Misc</h4>
  285. PIR enables Display: <input type='checkbox' name='PIRenDisp' id='PIRenDisp'/><br>
  286. toggling I/A-Temp and Hum display*: <input type='checkbox' name='togTHdisp' id='togTHdisp'/><br>
  287. * otherwise fixed I/A-Temp<br>
  288. <br><br>
  289. <input type='submit' value='Save'/>
  290. </form>
  291. <form id='rebootForm' onsubmit='return res()'>
  292. <input type='submit' value='Restart'/>
  293. </form>
  294. <script>
  295. function g(i) { return document.getElementById(i) };
  296. var xhttp, reqTime, reqFin, rxhttp;
  297. function res() {
  298. rxhttp = new XMLHttpRequest();
  299. rxhttp.timeout = 1000;
  300. rxhttp.open('POST', 'restart');
  301. rxhttp.send('');
  302. rxhttp = null;
  303. return false;
  304. }
  305. function setCheckbox(ele, dat) {
  306. if(dat == 1) {
  307. ele.checked = true;
  308. ele.style.visibility = 'visible';
  309. }
  310. else {
  311. ele.checked = false;
  312. ele.style.visibility = 'visible';
  313. }
  314. }
  315. function updateCheckboxValue(ele) {
  316. if (ele.checked) ele.value ='1';
  317. else {
  318. ele.value = '0';
  319. ele.checked = true;
  320. ele.style.visibility = 'hidden';
  321. }
  322. }
  323. function transmit(f) {
  324. if (!xhttp) {
  325. updateCheckboxValue(g('autoSaveTemp'));
  326. updateCheckboxValue(g('autoSaveMode'));
  327. updateCheckboxValue(g('PIRenDisp'));
  328. xhttp = new XMLHttpRequest();
  329. xhttp.timeout = 2000;
  330. xhttp.open('POST', 'confdata2');
  331. xhttp.send(f ? (new FormData(f)) : '');
  332. xhttp.onreadystatechange = function () {
  333. if (xhttp.readyState === XMLHttpRequest.DONE && xhttp.status === 200) {
  334. var data = JSON.parse(xhttp.responseText);
  335. setCheckbox(g('autoSaveTemp'), data.autoSaveTemp);
  336. setCheckbox(g('autoSaveMode'), data.autoSaveMode);
  337. setCheckbox(g('PIRenDisp'), data.PIRenDisp);
  338. setCheckbox(g('togTHdisp'), data.togTHdisp);
  339. g('domIdxTherm').value = data.domIdxTherm;
  340. g('domIdxMode').value = data.domIdxMode;
  341. g('domIdxTempHum').value = data.domIdxTempHum;
  342. g('domIdxHeating').value = data.domIdxHeating;
  343. g('domIdxPIR').value = data.domIdxPIR;
  344. g('outTempTop').value = data.outTempTop;
  345. g('outHumTop').value = data.outHumTop;
  346. g('PIRTop').value = data.PIRTop;
  347. g('minOffTime').value = data.minOffTime;
  348. g('tempMin').value = data.tempMin;
  349. g('tempMax').value = data.tempMax;
  350. g('tempLow').value = data.tempLow;
  351. g('tempLow2').value = data.tempLow2;
  352. g('tempDec').value = data.tempDec;
  353. g('hyst').value = data.hyst;
  354. g('tempCorr').value = data.tempCorr;
  355. g('humCorr').value = data.humCorr;
  356. g('measInt').value = data.measInt;
  357. g('dispInt').value = data.dispInt;
  358. g('dispTout').value = data.dispTout;
  359. g('offMsg').value = data.offMsg;
  360. g('modename0').value = data.modename0;
  361. g('modename1').value = data.modename1;
  362. g('psetname0').value = data.psetname0;
  363. g('psetname1').value = data.psetname1;
  364. g('psetname2').value = data.psetname2;
  365. xhttp = null;
  366. reqFin = false;
  367. }
  368. else {
  369. if(!reqFin && reqTime > 10) {
  370. xhttp = null;
  371. reqFin = true;
  372. }
  373. }
  374. }
  375. }
  376. return false;
  377. }
  378. transmit();
  379. setInterval(function () { ++reqTime; }, 1000);
  380. </script>
  381. </body></html>)";
  382. void httpServerHandleRoot() {
  383. httpServer.send_P(200, "text/html", httpRoot);
  384. }
  385. void httpServerHandleConfPage() {
  386. httpServer.send_P(200, "text/html", httpConfPage);
  387. }
  388. void httpServerHandleConf2Page() {
  389. httpServer.send_P(200, "text/html", httpConf2Page);
  390. }
  391. //void httpServerHandleNotFound() {
  392. // String message = "File Not Found\n\n";
  393. // message += "URI: ";
  394. // message += httpServer.uri();
  395. // message += "\nMethod: ";
  396. // message += (httpServer.method() == HTTP_GET) ? "GET" : "POST";
  397. // message += "\nArguments: ";
  398. // message += httpServer.args();
  399. // message += "\n";
  400. // for (uint8_t i = 0; i < httpServer.args(); i++) {
  401. // message += " " + httpServer.argName(i) + ": " + httpServer.arg(i) + "\n";
  402. // }
  403. // httpServer.send(404, "text / plain", message);
  404. //}
  405. void httpServerHandleNotFound() {
  406. // if (strlen(http_user) > 0 && strlen(http_pass) > 0) {
  407. // if (!httpServer.authenticate(http_user, http_pass))
  408. // return httpServer.requestAuthentication();
  409. httpServer.send(404, "text/plain", "");
  410. //}
  411. }
  412. void httpServerInit() {
  413. httpServer.on("/delconf", []() {
  414. Serial.println("httpServer.on /delconf");
  415. if (httpServer.hasArg("token")) {
  416. char buf[20];
  417. httpServer.arg("token").toCharArray(buf, 20);
  418. if (strcmp(buf, CLEARCONF_TOKEN) == 0) {
  419. // httpServer.send(200, "text / plain", "Token OK - deleting config");
  420. deleteConfig();
  421. }
  422. } //if
  423. // else {
  424. // httpServer.send(200, "text / plain", "not allowed");
  425. // }
  426. });
  427. httpServer.on("/api", []() {
  428. boolean isAuthenticated = false;
  429. if (strlen(http_user) > 0 && strlen(http_pass) > 0) {
  430. if (!httpServer.authenticate(http_user, http_pass)) return httpServer.requestAuthentication();
  431. isAuthenticated = true;
  432. }
  433. else isAuthenticated = true;
  434. if (isAuthenticated) {
  435. DEBUG_PRINT("httpServer.on /api");
  436. if (httpServer.hasArg("plusBtn")) {
  437. setTempStepUp();
  438. DEBUG_PRINT(P("web plusBtn"));
  439. } //if
  440. if (httpServer.hasArg("minusBtn")) {
  441. setTempStepDown();
  442. DEBUG_PRINT(P("web minusBtn"));
  443. } //if
  444. if (httpServer.hasArg("pset0Btn")) {
  445. setPresetTo(0);
  446. DEBUG_PRINT(P("web pset0Btn"));
  447. } //if
  448. if (httpServer.hasArg("pset1Btn")) {
  449. setPresetTo(1);
  450. DEBUG_PRINT(P("web pset1Btn"));
  451. } //if
  452. if (httpServer.hasArg("pset2Btn")) {
  453. setPresetTo(2);
  454. DEBUG_PRINT(P("web pset2Btn"));
  455. } //if
  456. if (httpServer.hasArg("onBtn")) {
  457. setHeatingmodeTo(1);
  458. DEBUG_PRINT(P("web onBtn"));
  459. } //if
  460. if (httpServer.hasArg("offBtn")) {
  461. setHeatingmodeTo(0);
  462. DEBUG_PRINT(P("web offBtn"));
  463. } //if
  464. //char ch_currTemp[6];
  465. //char ch_currSetTemp[6];
  466. //dtostrf(currTemp, 1, 1, ch_currTemp );
  467. //dtostrf(setTemp, 1, 1, ch_currSetTemp );
  468. //build json object of program data
  469. StaticJsonBuffer<400> jsonBuffer;
  470. JsonObject &json = jsonBuffer.createObject();
  471. json["devname"] = deviceName;
  472. json["ssid"] = WiFi.SSID();
  473. json["setTemp"] = setTemp;
  474. json["currSetTemp"] = currSetTemp;
  475. json["temp"] = currTemp;
  476. json["hum"] = int(currHum);
  477. json["heating"] = turnHeatingOn;
  478. json["mode"] = heatingMode;
  479. json["modeName"] = currentModeName;
  480. json["pset"] = preset;
  481. json["psetName"] = currentPresetName;
  482. json["psetName0"] = psetname0;
  483. json["psetName1"] = psetname1;
  484. json["psetName2"] = psetname2;
  485. char jsonchar[400];
  486. json.printTo(jsonchar); //print to char array, takes more memory but sends in one piece
  487. httpServer.send(200, "application/json", jsonchar);
  488. }
  489. else {
  490. httpServer.send (401, "text/plain", "UNAUTHORIZED");
  491. }
  492. }); //httpServer.on /api
  493. httpServer.on("/restart", []() {
  494. if (httpServer.hasArg("token")) {
  495. Serial.println("web triggered restart");
  496. ESP.restart();
  497. }
  498. else {
  499. httpServer.send (403, "text/plain", "FORBIDDEN");
  500. }
  501. });
  502. httpServer.on("/mqttReconnect", []() {
  503. if (httpServer.hasArg("token")) {
  504. Serial.println("web triggered mqttReconnect");
  505. mqttReconnect();
  506. httpServer.sendHeader("Location", String("/"), true);
  507. httpServer.send (302, "text/plain", "OK");
  508. }
  509. else {
  510. httpServer.send (403, "text/plain", "FORBIDDEN");
  511. }
  512. });
  513. httpServer.on("/setPoint", []() {
  514. if (httpServer.hasArg("token")) {
  515. char buf[20];
  516. httpServer.arg("token").toCharArray(buf, 20);
  517. if (strcmp(buf, http_token) == 0) {
  518. Serial.println("web triggered setPoint");
  519. if (httpServer.hasArg("value")) {
  520. char bufVal[20];
  521. httpServer.arg("value").toCharArray(bufVal, 20);
  522. float valueFloat = round(atof(bufVal) * 2.0) / 2.0;
  523. setTempTo(valueFloat);
  524. httpServer.send (200, "text/plain", "OK");
  525. }
  526. }
  527. } //if
  528. else {
  529. httpServer.send (403, "text/plain", "FORBIDDEN");
  530. }
  531. });
  532. httpServer.on("/setMode", []() {
  533. if (httpServer.hasArg("token")) {
  534. char buf[20];
  535. httpServer.arg("token").toCharArray(buf, 20);
  536. if (strcmp(buf, http_token) == 0) {
  537. Serial.println("web triggered setMode");
  538. if (httpServer.hasArg("value")) {
  539. char bufVal[20];
  540. httpServer.arg("value").toCharArray(bufVal, 20);
  541. int valueInt = atoi(bufVal);
  542. if (valueInt >= 0 && valueInt <= 1) setHeatingmodeTo(valueInt);
  543. httpServer.send (200, "text/plain", "OK");
  544. }
  545. }
  546. } //if
  547. else {
  548. httpServer.send (403, "text/plain", "FORBIDDEN");
  549. }
  550. });
  551. httpServer.on("/setPreset", []() {
  552. if (httpServer.hasArg("token")) {
  553. char buf[20];
  554. httpServer.arg("token").toCharArray(buf, 20);
  555. if (strcmp(buf, http_token) == 0) {
  556. Serial.println("web triggered setPreset");
  557. if (httpServer.hasArg("value")) {
  558. char bufVal[20];
  559. httpServer.arg("value").toCharArray(bufVal, 20);
  560. int valueInt = atoi(bufVal);
  561. if (valueInt >= 0 && valueInt <= 2) setPresetTo(valueInt);
  562. httpServer.send (200, "text/plain", "OK");
  563. }
  564. }
  565. } //if
  566. else {
  567. httpServer.send (403, "text/plain", "FORBIDDEN");
  568. }
  569. });
  570. httpServer.on("/confdata", []() {
  571. boolean isAuthenticated = false;
  572. if (strlen(http_user) > 0 && strlen(http_pass) > 0) {
  573. if (!httpServer.authenticate(http_user, http_pass)) return httpServer.requestAuthentication();
  574. isAuthenticated = true;
  575. }
  576. else isAuthenticated = true;
  577. if (isAuthenticated) {
  578. Serial.println("httpServer.on /confdata");
  579. for (int i = 0; i < httpServer.args(); i++) {
  580. char bufName[20];
  581. char bufValue[101];
  582. httpServer.argName(i).toCharArray(bufName, 20);
  583. httpServer.arg(i).toCharArray(bufValue, 101);
  584. if (strlen(bufName) > 0) {
  585. Serial.print("web update ");
  586. Serial.print(bufName);
  587. Serial.print(" = ");
  588. Serial.println(bufValue);
  589. setConfig(bufName, bufValue);
  590. }
  591. saveConfigToFlash = true; // will be saved in next loop()
  592. Serial.println("web triggered saveConfigToFlash");
  593. }
  594. yield();
  595. //build json object of program data
  596. StaticJsonBuffer<1000> jsonBuffer;
  597. JsonObject &json = jsonBuffer.createObject();
  598. json["devName"] = deviceName;
  599. json["httpUser"] = http_user;
  600. json["httpPass"] = http_pass;
  601. json["httpToken"] = http_token;
  602. json["mqttHost"] = mqtt_server;
  603. json["mqttPort"] = mqtt_port;
  604. json["mqttUser"] = mqtt_user;
  605. json["mqttPass"] = mqtt_pass;
  606. json["inTop"] = mqtt_topic_in;
  607. json["outTop"] = mqtt_topic_out;
  608. json["outRet"] = mqtt_outRetain;
  609. json["willTop"] = mqtt_willTopic;
  610. json["willQos"] = mqtt_willQos;
  611. json["willRet"] = mqtt_willRetain;
  612. json["willMsg"] = mqtt_willMsg;
  613. json["connMsg"] = mqtt_connMsg;
  614. json["domOutTop"] = domoticz_out_topic;
  615. yield();
  616. char jsonchar[1000];
  617. json.printTo(jsonchar); //print to char array, takes more memory but sends in one piece
  618. httpServer.send(200, "application/json", jsonchar);
  619. }
  620. else {
  621. httpServer.send (401, "text/plain", "UNAUTHORIZED");
  622. }
  623. }); //httpServer.on /confdata
  624. httpServer.on("/confdata2", []() {
  625. boolean isAuthenticated = false;
  626. if (strlen(http_user) > 0 && strlen(http_pass) > 0) {
  627. if (!httpServer.authenticate(http_user, http_pass)) return httpServer.requestAuthentication();
  628. isAuthenticated = true;
  629. }
  630. else isAuthenticated = true;
  631. if (isAuthenticated) {
  632. Serial.println("httpServer.on /confdata2");
  633. for (int i = 0; i < httpServer.args(); i++) {
  634. char bufName[20];
  635. char bufValue[101];
  636. httpServer.argName(i).toCharArray(bufName, 20);
  637. httpServer.arg(i).toCharArray(bufValue, 101);
  638. if (strlen(bufName) > 0) {
  639. Serial.print("web update ");
  640. Serial.print(bufName);
  641. Serial.print(" = ");
  642. Serial.println(bufValue);
  643. setConfig(bufName, bufValue);
  644. }
  645. saveConfig2ToFlash = true;
  646. Serial.println("web triggered saveConfig2ToFlash");
  647. }
  648. yield();
  649. //build json object of program data
  650. StaticJsonBuffer<1000> jsonBuffer;
  651. JsonObject &json = jsonBuffer.createObject();
  652. json["domIdxTherm"] = domoticzIdx_Thermostat;
  653. json["domIdxMode"] = domoticzIdx_ThermostatMode;
  654. json["domIdxTempHum"] = domoticzIdx_TempHumSensor;
  655. json["domIdxHeating"] = domoticzIdx_Heating;
  656. json["domIdxPIR"] = domoticzIdx_PIR;
  657. json["outTempTop"] = outTemp_topic_in;
  658. json["outHumTop"] = outHum_topic_in;
  659. json["PIRTop"] = mqtt_topic_pir;
  660. json["autoSaveTemp"] = autoSaveSetTemp;
  661. json["autoSaveMode"] = autoSaveHeatingMode;
  662. json["minOffTime"] = heatingMinOffTime;
  663. json["tempMin"] = setTempMin;
  664. json["tempMax"] = setTempMax;
  665. json["tempLow"] = setTempLow;
  666. json["tempLow2"] = setTempLow2;
  667. json["tempDec"] = setTempDecreaseVal;
  668. json["hyst"] = hysteresis;
  669. json["tempCorr"] = tempCorrVal;
  670. json["humCorr"] = humCorrVal;
  671. json["measInt"] = measureInterval;
  672. json["dispInt"] = displayInterval;
  673. json["dispTout"] = displayTimeout;
  674. json["offMsg"] = offMessage;
  675. json["modename0"] = modename0;
  676. json["modename1"] = modename1;
  677. json["psetname0"] = psetname0;
  678. json["psetname1"] = psetname1;
  679. json["psetname2"] = psetname2;
  680. json["PIRenDisp"] = PIR_enablesDisplay;
  681. json["togTHdisp"] = togglingTempHumAIDisplay;
  682. yield();
  683. char jsonchar[1000];
  684. json.printTo(jsonchar); //print to char array, takes more memory but sends in one piece
  685. httpServer.send(200, "application/json", jsonchar);
  686. }
  687. else {
  688. httpServer.send (401, "text/plain", "UNAUTHORIZED");
  689. }
  690. }); //httpServer.on /confdata2
  691. //get heap status, analog input value and all GPIO statuses in one json call
  692. httpServer.on("/info", HTTP_GET, []() {
  693. boolean isAuthenticated = false;
  694. if (strlen(http_user) > 0 && strlen(http_pass) > 0) {
  695. if (!httpServer.authenticate(http_user, http_pass)) return httpServer.requestAuthentication();
  696. isAuthenticated = true;
  697. }
  698. else isAuthenticated = true;
  699. if (isAuthenticated) {
  700. String json = " {";
  701. json += "\"wifissid\":\"" + WiFi.SSID() + "\"";
  702. json += "\"heap\":" + String(ESP.getFreeHeap());
  703. json += "}";
  704. httpServer.send(200, "text/json", json);
  705. json = String();
  706. }
  707. else {
  708. httpServer.send (401, "text/plain", "UNAUTHORIZED");
  709. }
  710. }); //httpServer.on /info
  711. httpServer.on("/", []() {
  712. boolean isAuthenticated = false;
  713. if (strlen(http_user) > 0 && strlen(http_pass) > 0) {
  714. if (!httpServer.authenticate(http_user, http_pass)) return httpServer.requestAuthentication();
  715. isAuthenticated = true;
  716. }
  717. else isAuthenticated = true;
  718. if (isAuthenticated) {
  719. httpServerHandleRoot();
  720. }
  721. else {
  722. httpServer.send (401, "text/plain", "UNAUTHORIZED");
  723. }
  724. });
  725. httpServer.on("/conf", []() {
  726. boolean isAuthenticated = false;
  727. if (strlen(http_user) > 0 && strlen(http_pass) > 0) {
  728. if (!httpServer.authenticate(http_user, http_pass)) return httpServer.requestAuthentication();
  729. isAuthenticated = true;
  730. }
  731. else isAuthenticated = true;
  732. if (isAuthenticated) {
  733. httpServerHandleConfPage();
  734. }
  735. else {
  736. httpServer.send (401, "text/plain", "UNAUTHORIZED");
  737. }
  738. });
  739. httpServer.on("/conf2", []() {
  740. boolean isAuthenticated = false;
  741. if (strlen(http_user) > 0 && strlen(http_pass) > 0) {
  742. if (!httpServer.authenticate(http_user, http_pass)) return httpServer.requestAuthentication();
  743. isAuthenticated = true;
  744. }
  745. else isAuthenticated = true;
  746. if (isAuthenticated) {
  747. httpServerHandleConf2Page();
  748. }
  749. else {
  750. httpServer.send (401, "text/plain", "UNAUTHORIZED");
  751. }
  752. });
  753. httpServer.onNotFound([]() {
  754. httpServerHandleNotFound();
  755. }); //httpServer.onNotFound
  756. // HTTP Updater at /update
  757. httpUpdater.setup(&httpServer);
  758. httpServer.begin();
  759. }