Fork of the espurna firmware for `mhsw` switches
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1057 lines
30 KiB

8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
  1. /*
  2. ESPurna
  3. Copyright (C) 2016 by Xose Pérez <xose dot perez at gmail dot com>
  4. This program is free software: you can redistribute it and/or modify
  5. it under the terms of the GNU General Public License as published by
  6. the Free Software Foundation, either version 3 of the License, or
  7. (at your option) any later version.
  8. This program is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. GNU General Public License for more details.
  12. You should have received a copy of the GNU General Public License
  13. along with this program. If not, see <http://www.gnu.org/licenses/>.
  14. */
  15. #include <Arduino.h>
  16. #include <ESP8266WiFi.h>
  17. #include <ESP8266WebServer.h>
  18. #include <ESP8266mDNS.h>
  19. #include <PubSubClient.h>
  20. #include <DebounceEvent.h>
  21. #include <ArduinoOTA.h>
  22. #include <RemoteReceiver.h>
  23. #include <EEPROM.h>
  24. #include "FS.h"
  25. #include <stdio.h>
  26. #include <EmonLiteESP.h>
  27. // -----------------------------------------------------------------------------
  28. // Configuració
  29. // -----------------------------------------------------------------------------
  30. #define DEBUG
  31. #define ENABLE_RF 1
  32. #define ENABLE_OTA 1
  33. #define ENABLE_MQTT 1
  34. #define ENABLE_WEBSERVER 1
  35. #define ENABLE_ENERGYMONITOR 1
  36. #define APP_NAME "Espurna 0.9.1"
  37. #define APP_AUTHOR "xose.perez@gmail.com"
  38. #define APP_WEBSITE "http://tinkerman.cat"
  39. #define MODEL "SONOFF"
  40. #define BUTTON_PIN 0
  41. #define RELAY_PIN 12
  42. #define LED_PIN 13
  43. #define ADMIN_PASS "fibonacci"
  44. #define CONFIG_PATH "/.config"
  45. #define BUFFER_SIZE 1024
  46. #define STATUS_UPDATE_INTERVAL 10000
  47. #define RF_PIN 14
  48. #define RF_CHANNEL 31
  49. #define RF_DEVICE 1
  50. #define MQTT_RECONNECT_DELAY 10000
  51. #define MQTT_RETAIN true
  52. #define MQTT_TOPIC "/test/switch/{identifier}"
  53. #define MQTT_PORT 1883
  54. #define NETWORK_BUFFER 3
  55. #define WIFI_CONNECT_TIMEOUT 5000
  56. #define WIFI_RECONNECT_DELAY 5000
  57. #define WIFI_STATUS_CONNECTING 0
  58. #define WIFI_STATUS_CONNECTED 1
  59. #define WIFI_STATUS_AP 2
  60. #define CURRENT_PIN A0
  61. #define REFERENCE_VOLTAGE 1.0
  62. #define MAINS_VOLTAGE 230.0
  63. #define CURRENT_RATIO 156
  64. #define CURRENT_PRECISION 1
  65. #define SAMPLES_X_MEASUREMENT 1500
  66. #define MEASUREMENT_INTERVAL 10000
  67. #define MEASUREMENTS_X_MESSAGE 6
  68. // -----------------------------------------------------------------------------
  69. // Globals
  70. // -----------------------------------------------------------------------------
  71. char identifier[20] = {0};
  72. byte status = WIFI_STATUS_CONNECTING;
  73. String configSSID[NETWORK_BUFFER];
  74. String configPASS[NETWORK_BUFFER];
  75. DebounceEvent button1 = false;
  76. #if ENABLE_WEBSERVER
  77. ESP8266WebServer server(80);
  78. #endif
  79. #if ENABLE_MQTT
  80. WiFiClient client;
  81. PubSubClient mqtt(client);
  82. String mqttServer = "";
  83. String mqttTopic = MQTT_TOPIC;
  84. String mqttPort = String(MQTT_PORT);
  85. String mqttUser = "";
  86. String mqttPassword = "";
  87. char mqttStatusTopic[30];
  88. char mqttIPTopic[30];
  89. #if ENABLE_ENERGYMONITOR
  90. char mqttPowerTopic[30];
  91. #endif
  92. bool isMQTTMessage = false;
  93. #endif
  94. #if ENABLE_RF
  95. unsigned long rfCode = 0;
  96. unsigned long rfCodeON = 0;
  97. unsigned long rfCodeOFF = 0;
  98. String rfChannel = String(RF_CHANNEL);
  99. String rfDevice = String(RF_DEVICE);
  100. #endif
  101. #if ENABLE_ENERGYMONITOR
  102. EnergyMonitor monitor;
  103. #endif
  104. // -----------------------------------------------------------------------------
  105. // Utils
  106. // -----------------------------------------------------------------------------
  107. char * getCompileTime(char * buffer) {
  108. int day, month, year, hour, minute, second;
  109. // parse date
  110. String tmp = String(__DATE__);
  111. day = tmp.substring(4,6).toInt();
  112. year = tmp.substring(7).toInt();
  113. tmp = tmp.substring(0,3);
  114. if (tmp.equals("Jan")) month = 1;
  115. if (tmp.equals("Feb")) month = 2;
  116. if (tmp.equals("Mar")) month = 3;
  117. if (tmp.equals("Apr")) month = 4;
  118. if (tmp.equals("May")) month = 5;
  119. if (tmp.equals("Jun")) month = 6;
  120. if (tmp.equals("Jul")) month = 7;
  121. if (tmp.equals("Aug")) month = 8;
  122. if (tmp.equals("Sep")) month = 9;
  123. if (tmp.equals("Oct")) month = 10;
  124. if (tmp.equals("Nov")) month = 11;
  125. if (tmp.equals("Dec")) month = 12;
  126. // parse time
  127. tmp = String(__TIME__);
  128. hour = tmp.substring(0,2).toInt();
  129. minute = tmp.substring(3,5).toInt();
  130. second = tmp.substring(6,8).toInt();
  131. sprintf(buffer, "%d%02d%02d%02d%02d%02d", year, month, day, hour, minute, second);
  132. buffer[14] = 0;
  133. return buffer;
  134. }
  135. // -----------------------------------------------------------------------------
  136. // Relay
  137. // -----------------------------------------------------------------------------
  138. void switchRelayOn() {
  139. if (!digitalRead(RELAY_PIN)) {
  140. #ifdef DEBUG
  141. Serial.println("Turning the relay ON");
  142. #endif
  143. #if ENABLE_MQTT
  144. if (!isMQTTMessage && mqtt.connected()) {
  145. mqtt.publish(mqttStatusTopic, "1", MQTT_RETAIN);
  146. }
  147. #endif
  148. digitalWrite(RELAY_PIN, HIGH);
  149. if (EEPROM.read(0) == 0) {
  150. EEPROM.write(0, 1);
  151. EEPROM.commit();
  152. }
  153. }
  154. }
  155. void switchRelayOff() {
  156. if (digitalRead(RELAY_PIN)) {
  157. #ifdef DEBUG
  158. Serial.println("Turning the relay OFF");
  159. #endif
  160. #if ENABLE_MQTT
  161. if (!isMQTTMessage && mqtt.connected()) {
  162. mqtt.publish(mqttStatusTopic, "0", MQTT_RETAIN);
  163. }
  164. #endif
  165. digitalWrite(RELAY_PIN, LOW);
  166. if (EEPROM.read(0) == 1) {
  167. EEPROM.write(0, 0);
  168. EEPROM.commit();
  169. }
  170. }
  171. }
  172. void toggleRelay() {
  173. if (digitalRead(RELAY_PIN)) {
  174. switchRelayOff();
  175. } else {
  176. switchRelayOn();
  177. }
  178. }
  179. // -----------------------------------------------------------------------------
  180. // Wifi
  181. // -----------------------------------------------------------------------------
  182. char * getIdentifier() {
  183. if (identifier[0] == 0) {
  184. sprintf(identifier, "%s_%06X", MODEL, ESP.getChipId());
  185. }
  186. return identifier;
  187. }
  188. void wifiSetupAP() {
  189. // Set WIFI module AP mode
  190. WiFi.mode(WIFI_AP);
  191. #ifdef DEBUG
  192. WiFi.printDiag(Serial);
  193. #endif
  194. // SoftAP mode
  195. WiFi.softAP(getIdentifier(), ADMIN_PASS);
  196. status = WIFI_STATUS_AP;
  197. #ifdef DEBUG
  198. Serial.print("[AP Mode] SSID: ");
  199. Serial.print(getIdentifier());
  200. Serial.print(", Password: \"");
  201. Serial.print(ADMIN_PASS);
  202. Serial.print("\", IP address: ");
  203. Serial.println(WiFi.softAPIP());
  204. #endif
  205. }
  206. void wifiSetupSTA(bool force) {
  207. byte network = 0;
  208. if (force || (WiFi.status() != WL_CONNECTED)) {
  209. // Set WIFI module to STA mode
  210. WiFi.mode(WIFI_STA);
  211. #ifdef DEBUG
  212. WiFi.printDiag(Serial);
  213. #endif
  214. #if ENABLE_MQTT
  215. if (mqtt.connected()) mqtt.disconnect();
  216. #endif
  217. #if ENABLE_RF
  218. RemoteReceiver::disable();
  219. #endif
  220. while (network < 3) {
  221. if (configSSID[network].length() > 0) {
  222. char ssid[configSSID[network].length()+1];
  223. char pass[configPASS[network].length()+1];
  224. configSSID[network].toCharArray(ssid, configSSID[network].length()+1);
  225. configPASS[network].toCharArray(pass, configPASS[network].length()+1);
  226. WiFi.begin(ssid, pass);
  227. #ifdef DEBUG
  228. Serial.println("Connecting to WIFI " + configSSID[network]);
  229. #endif
  230. // Wait
  231. unsigned long timeout = millis() + WIFI_CONNECT_TIMEOUT;
  232. while (timeout > millis()) {
  233. showStatus();
  234. if (WiFi.status() == WL_CONNECTED) break;
  235. delay(100);
  236. }
  237. }
  238. if (WiFi.status() == WL_CONNECTED) break;
  239. network++;
  240. }
  241. #if ENABLE_RF
  242. RemoteReceiver::enable();
  243. #endif
  244. }
  245. if (WiFi.status() == WL_CONNECTED) {
  246. WiFi.setAutoConnect(true);
  247. status = WIFI_STATUS_CONNECTED;
  248. #ifdef DEBUG
  249. Serial.print("[STA Mode] SSID: ");
  250. Serial.print(WiFi.SSID());
  251. Serial.print(", IP address: ");
  252. Serial.println(WiFi.localIP());
  253. #endif
  254. } else {
  255. #ifdef DEBUG
  256. Serial.println("[STA Mode] NOT CONNECTED");
  257. #endif
  258. wifiSetupAP();
  259. }
  260. }
  261. void wifiLoop() {
  262. // Trying to reconnect in case of disconnection
  263. if ((status == WIFI_STATUS_CONNECTED) && (WiFi.status() != WL_CONNECTED)) {
  264. status = WIFI_STATUS_CONNECTING;
  265. wifiSetupSTA(false);
  266. }
  267. }
  268. // -----------------------------------------------------------------------------
  269. // WebServer
  270. // -----------------------------------------------------------------------------
  271. #if ENABLE_WEBSERVER
  272. String getContentType(String filename) {
  273. if (server.hasArg("download")) return "application/octet-stream";
  274. else if (filename.endsWith(".htm")) return "text/html";
  275. else if (filename.endsWith(".html")) return "text/html";
  276. else if (filename.endsWith(".css")) return "text/css";
  277. else if (filename.endsWith(".js")) return "application/javascript";
  278. else if (filename.endsWith(".png")) return "image/png";
  279. else if (filename.endsWith(".gif")) return "image/gif";
  280. else if (filename.endsWith(".jpg")) return "image/jpeg";
  281. else if (filename.endsWith(".ico")) return "image/x-icon";
  282. else if (filename.endsWith(".xml")) return "text/xml";
  283. else if (filename.endsWith(".pdf")) return "application/x-pdf";
  284. else if (filename.endsWith(".zip")) return "application/x-zip";
  285. else if (filename.endsWith(".gz")) return "application/x-gzip";
  286. return "text/plain";
  287. }
  288. void handleRelayOn() {
  289. #ifdef DEBUG
  290. Serial.println("Request: /relay/on");
  291. #endif
  292. switchRelayOn();
  293. server.send(200, "text/plain", "ON");
  294. }
  295. void handleRelayOff() {
  296. #ifdef DEBUG
  297. Serial.println("Request: /relay/off");
  298. #endif
  299. switchRelayOff();
  300. server.send(200, "text/plain", "OFF");
  301. }
  302. bool handleFileRead(String path) {
  303. #ifdef DEBUG
  304. Serial.println("Request: " + path);
  305. #endif
  306. if (path.endsWith("/")) path += "index.html";
  307. String contentType = getContentType(path);
  308. String pathWithGz = path + ".gz";
  309. if (SPIFFS.exists(pathWithGz)) path = pathWithGz;
  310. if (SPIFFS.exists(path)) {
  311. File file = SPIFFS.open(path, "r");
  312. size_t sent = server.streamFile(file, contentType);
  313. size_t contentLength = file.size();
  314. file.close();
  315. return true;
  316. }
  317. return false;
  318. }
  319. void handleHome() {
  320. #ifdef DEBUG
  321. Serial.println("Request: /index.html");
  322. #endif
  323. String filename = "/index.html";
  324. String content = "";
  325. char buffer[BUFFER_SIZE];
  326. // Read file in chunks
  327. File file = SPIFFS.open(filename, "r");
  328. int size = file.size();
  329. while (size > 0) {
  330. size_t len = std::min(BUFFER_SIZE-1, size);
  331. file.read((uint8_t *) buffer, len);
  332. buffer[len] = 0;
  333. content += buffer;
  334. size -= len;
  335. }
  336. file.close();
  337. // Replace placeholders
  338. getCompileTime(buffer);
  339. content.replace("{appname}", String(APP_NAME) + "." + String(buffer));
  340. content.replace("{status}", digitalRead(RELAY_PIN) ? "1" : "0");
  341. content.replace("{updateInterval}", String(STATUS_UPDATE_INTERVAL));
  342. content.replace("{ssid0}", configSSID[0]);
  343. content.replace("{pass0}", configPASS[0]);
  344. content.replace("{ssid1}", configSSID[1]);
  345. content.replace("{pass1}", configPASS[1]);
  346. content.replace("{ssid2}", configSSID[2]);
  347. content.replace("{pass2}", configPASS[2]);
  348. #if ENABLE_MQTT
  349. content.replace("{mqttServer}", mqttServer);
  350. content.replace("{mqttPort}", mqttPort);
  351. content.replace("{mqttUser}", mqttUser);
  352. content.replace("{mqttPassword}", mqttPassword);
  353. content.replace("{mqttTopic}", mqttTopic);
  354. #endif
  355. #if ENABLE_RF
  356. content.replace("{rfChannel}", rfChannel);
  357. content.replace("{rfDevice}", rfDevice);
  358. #endif
  359. // Serve content
  360. String contentType = getContentType(filename);
  361. server.send(200, contentType, content);
  362. }
  363. void handleSave() {
  364. #ifdef DEBUG
  365. Serial.println("Request: /save");
  366. #endif
  367. if (server.hasArg("status")) {
  368. if (server.arg("status") == "1") {
  369. switchRelayOn();
  370. } else {
  371. switchRelayOff();
  372. }
  373. }
  374. if (server.hasArg("ssid0")) configSSID[0] = server.arg("ssid0");
  375. if (server.hasArg("pass0")) configPASS[0] = server.arg("pass0");
  376. if (server.hasArg("ssid1")) configSSID[1] = server.arg("ssid1");
  377. if (server.hasArg("pass1")) configPASS[1] = server.arg("pass1");
  378. if (server.hasArg("ssid2")) configSSID[2] = server.arg("ssid2");
  379. if (server.hasArg("pass2")) configPASS[2] = server.arg("pass2");
  380. #if ENABLE_MQTT
  381. if (server.hasArg("mqttServer")) mqttServer = server.arg("mqttServer");
  382. if (server.hasArg("mqttPort")) mqttPort = server.arg("mqttPort");
  383. if (server.hasArg("mqttUser")) mqttUser = server.arg("mqttUser");
  384. if (server.hasArg("mqttPassword")) mqttPassword = server.arg("mqttPassword");
  385. if (server.hasArg("mqttTopic")) mqttTopic = server.arg("mqttTopic");
  386. #endif
  387. #if ENABLE_RF
  388. if (server.hasArg("rfChannel")) rfChannel = server.arg("rfChannel");
  389. if (server.hasArg("rfDevice")) rfDevice = server.arg("rfDevice");
  390. #endif
  391. server.send(202, "text/json", "{}");
  392. saveConfig();
  393. #if ENABLE_RF
  394. rfBuildCodes();
  395. #endif
  396. wifiSetupSTA(true);
  397. }
  398. void handleStatus() {
  399. #ifdef DEBUG
  400. //Serial.println("Request: /status");
  401. #endif
  402. String output = "{";
  403. output += "\"device\": \"";
  404. output += getIdentifier();
  405. output += "\", \"ip\": \"";
  406. output += (WiFi.status() == WL_CONNECTED) ? WiFi.localIP().toString() : "NOT CONNECTED";
  407. output += "\", \"network\": \"";
  408. output += (WiFi.status() == WL_CONNECTED) ? WiFi.SSID() : "";
  409. output += "\", \"relay\": ";
  410. output += (digitalRead(RELAY_PIN)) ? "1": "0";
  411. #if ENABLE_MQTT
  412. output += ", \"mqtt\": ";
  413. output += (mqtt.connected()) ? "1": "0";
  414. #endif
  415. output += "}";
  416. server.send(200, "text/json", output);
  417. }
  418. void webServerSetup() {
  419. // Relay control
  420. server.on("/relay/on", HTTP_GET, handleRelayOn);
  421. server.on("/relay/off", HTTP_GET, handleRelayOff);
  422. // Configuration page
  423. server.on("/save", HTTP_POST, handleSave);
  424. server.on("/status", HTTP_GET, handleStatus);
  425. server.on("/", HTTP_GET, handleHome);
  426. server.on("/index.html", HTTP_GET, handleHome);
  427. // Anything else
  428. server.onNotFound([]() {
  429. // Hidden files
  430. if (server.uri().startsWith("/.")) {
  431. server.send(403, "text/plain", "Forbidden");
  432. return;
  433. }
  434. // Existing files in SPIFFS
  435. if (!handleFileRead(server.uri())) {
  436. server.send(404, "text/plain", "NotFound");
  437. return;
  438. }
  439. });
  440. // Run server
  441. server.begin();
  442. }
  443. void webServerLoop() {
  444. server.handleClient();
  445. }
  446. #endif
  447. // -----------------------------------------------------------------------------
  448. // MQTT
  449. // -----------------------------------------------------------------------------
  450. #if ENABLE_MQTT
  451. void buildTopics() {
  452. String tmp;
  453. // Replace identifier
  454. String base = mqttTopic;
  455. base.replace("{identifier}", getIdentifier());
  456. // Get publish status topic
  457. base.toCharArray(mqttStatusTopic, base.length()+1);
  458. mqttStatusTopic[base.length()+1] = 0;
  459. // Get publish ip topic
  460. tmp = base + "/ip";
  461. tmp.toCharArray(mqttIPTopic, tmp.length()+1);
  462. mqttIPTopic[tmp.length()+1] = 0;
  463. // Get publish current topic
  464. #if ENABLE_ENERGYMONITOR
  465. tmp = base + "/power";
  466. tmp.toCharArray(mqttPowerTopic, tmp.length()+1);
  467. mqttPowerTopic[tmp.length()+1] = 0;
  468. #endif
  469. }
  470. void mqttCallback(char* topic, byte* payload, unsigned int length) {
  471. #ifdef DEBUG
  472. Serial.print("MQTT message ");
  473. Serial.print(topic);
  474. Serial.print(" => ");
  475. for (int i = 0; i < length; i++) {
  476. Serial.print((char)payload[i]);
  477. }
  478. Serial.println();
  479. #endif
  480. // Action to perform
  481. if ((char)payload[0] == '0') {
  482. isMQTTMessage = true;
  483. switchRelayOff();
  484. }
  485. if ((char)payload[0] == '1') {
  486. isMQTTMessage = true;
  487. switchRelayOn();
  488. }
  489. if ((char)payload[0] == '2') {
  490. toggleRelay();
  491. }
  492. isMQTTMessage = false;
  493. }
  494. void mqttConnect() {
  495. if (!mqtt.connected() && (mqttServer.length()>0)) {
  496. char buffer[mqttServer.length()+1];
  497. mqttServer.toCharArray(buffer, mqttServer.length()+1);
  498. mqtt.setServer(buffer, mqttPort.toInt());
  499. #ifdef DEBUG
  500. Serial.print("Connecting to MQTT broker at ");
  501. Serial.print(mqttServer);
  502. #endif
  503. if (mqttUser.length() > 0) {
  504. #ifdef DEBUG
  505. Serial.print(" as user ");
  506. Serial.print(mqttUser);
  507. Serial.print(": ");
  508. #endif
  509. char user[mqttUser.length() + 1];
  510. mqttUser.toCharArray(user, mqttUser.length() + 1);
  511. char password[mqttPassword.length() + 1];
  512. mqttPassword.toCharArray(password, mqttPassword.length() + 1);
  513. mqtt.connect(getIdentifier(), user, password);
  514. } else {
  515. #ifdef DEBUG
  516. Serial.print(" anonymously: ");
  517. #endif
  518. mqtt.connect(getIdentifier());
  519. }
  520. if (mqtt.connected()) {
  521. buildTopics();
  522. #ifdef DEBUG
  523. Serial.println("connected!");
  524. Serial.print("Subscribing to ");
  525. Serial.println(mqttStatusTopic);
  526. #endif
  527. // Say hello and report our IP
  528. String ipString = WiFi.localIP().toString();
  529. char ip[ipString.length()+1];
  530. ipString.toCharArray(ip, ipString.length()+1);
  531. mqtt.publish(mqttIPTopic, ip, MQTT_RETAIN);
  532. // Publish current relay status
  533. mqtt.publish(mqttStatusTopic, digitalRead(RELAY_PIN) ? "1" : "0", MQTT_RETAIN);
  534. // Subscribe to topic
  535. mqtt.subscribe(mqttStatusTopic);
  536. } else {
  537. #ifdef DEBUG
  538. Serial.print("failed, rc=");
  539. Serial.println(mqtt.state());
  540. #endif
  541. }
  542. }
  543. }
  544. void mqttSetup() {
  545. mqtt.setCallback(mqttCallback);
  546. }
  547. void mqttLoop() {
  548. static unsigned long timeout = millis();
  549. if (WiFi.status() == WL_CONNECTED) {
  550. if (!mqtt.connected()) {
  551. if (timeout < millis()) {
  552. mqttConnect();
  553. timeout = millis() + MQTT_RECONNECT_DELAY;
  554. }
  555. }
  556. if (mqtt.connected()) mqtt.loop();
  557. }
  558. }
  559. #endif
  560. // -----------------------------------------------------------------------------
  561. // RF
  562. // -----------------------------------------------------------------------------
  563. #if ENABLE_RF
  564. void rfLoop() {
  565. if (rfCode == 0) return;
  566. #ifdef DEBUG
  567. Serial.print("RF code: ");
  568. Serial.println(rfCode);
  569. #endif
  570. if (rfCode == rfCodeON) switchRelayOn();
  571. if (rfCode == rfCodeOFF) switchRelayOff();
  572. rfCode = 0;
  573. }
  574. void rfBuildCodes() {
  575. unsigned long code = 0;
  576. // channel
  577. unsigned int channel = rfChannel.toInt();
  578. for (byte i = 0; i < 5; i++) {
  579. code *= 3;
  580. if (channel & 1) code += 1;
  581. channel >>= 1;
  582. }
  583. // device
  584. unsigned int device = rfDevice.toInt();
  585. for (byte i = 0; i < 5; i++) {
  586. code *= 3;
  587. if (device != i) code += 2;
  588. }
  589. // status
  590. code *= 9;
  591. rfCodeOFF = code + 2;
  592. rfCodeON = code + 6;
  593. #ifdef DEBUG
  594. Serial.print("RF code ON: ");
  595. Serial.println(rfCodeON);
  596. Serial.print("RF code OFF: ");
  597. Serial.println(rfCodeOFF);
  598. #endif
  599. }
  600. void rfCallback(unsigned long code, unsigned int period) {
  601. rfCode = code;
  602. }
  603. void rfSetup() {
  604. rfBuildCodes();
  605. RemoteReceiver::init(RF_PIN, 3, rfCallback);
  606. RemoteReceiver::enable();
  607. }
  608. #endif
  609. // -----------------------------------------------------------------------------
  610. // Configuration
  611. // -----------------------------------------------------------------------------
  612. bool saveConfig() {
  613. File file = SPIFFS.open(CONFIG_PATH, "w");
  614. if (file) {
  615. file.println("ssid0=" + configSSID[0]);
  616. file.println("pass0=" + configPASS[0]);
  617. file.println("ssid1=" + configSSID[1]);
  618. file.println("pass1=" + configPASS[1]);
  619. file.println("ssid2=" + configSSID[2]);
  620. file.println("pass2=" + configPASS[2]);
  621. #if ENABLE_MQTT
  622. file.println("mqttServer=" + mqttServer);
  623. file.println("mqttPort=" + mqttPort);
  624. file.println("mqttTopic=" + mqttTopic);
  625. #endif
  626. #if ENABLE_RF
  627. file.println("rfChannel=" + rfChannel);
  628. file.println("rfDevice=" + rfDevice);
  629. #endif
  630. file.close();
  631. return true;
  632. }
  633. return false;
  634. }
  635. bool loadConfig() {
  636. if (SPIFFS.exists(CONFIG_PATH)) {
  637. #ifdef DEBUG
  638. Serial.println("Reading config file");
  639. #endif
  640. // Read contents
  641. File file = SPIFFS.open(CONFIG_PATH, "r");
  642. String content = file.readString();
  643. file.close();
  644. // Parse contents
  645. content.replace("\r\n", "\n");
  646. content.replace("\r", "\n");
  647. int start = 0;
  648. int end = content.indexOf("\n", start);
  649. while (end > 0) {
  650. String line = content.substring(start, end);
  651. #ifdef DEBUG
  652. Serial.println(line);
  653. #endif
  654. if (line.startsWith("ssid0=")) configSSID[0] = line.substring(6);
  655. else if (line.startsWith("pass0=")) configPASS[0] = line.substring(6);
  656. else if (line.startsWith("ssid1=")) configSSID[1] = line.substring(6);
  657. else if (line.startsWith("pass1=")) configPASS[1] = line.substring(6);
  658. else if (line.startsWith("ssid2=")) configSSID[2] = line.substring(6);
  659. else if (line.startsWith("pass2=")) configPASS[2] = line.substring(6);
  660. #if ENABLE_MQTT
  661. else if (line.startsWith("mqttServer=")) mqttServer = line.substring(11);
  662. else if (line.startsWith("mqttPort=")) mqttPort = line.substring(9);
  663. else if (line.startsWith("mqttTopic=")) mqttTopic = line.substring(10);
  664. #endif
  665. #if ENABLE_RF
  666. else if (line.startsWith("rfChannel=")) rfChannel = line.substring(10);
  667. else if (line.startsWith("rfDevice=")) rfDevice = line.substring(9);
  668. #endif
  669. if (end < 0) break;
  670. start = end + 1;
  671. end = content.indexOf("\n", start);
  672. }
  673. return true;
  674. }
  675. return false;
  676. }
  677. // -----------------------------------------------------------------------------
  678. // OTA
  679. // -----------------------------------------------------------------------------
  680. #if ENABLE_OTA
  681. void OTASetup() {
  682. // Port defaults to 8266
  683. ArduinoOTA.setPort(8266);
  684. // Hostname defaults to esp8266-[ChipID]
  685. ArduinoOTA.setHostname(getIdentifier());
  686. // No authentication by default
  687. ArduinoOTA.setPassword((const char *) ADMIN_PASS);
  688. ArduinoOTA.onStart([]() {
  689. #if ENABLE_RF
  690. RemoteReceiver::disable();
  691. #endif
  692. #ifdef DEBUG
  693. Serial.println("OTA - Start");
  694. #endif
  695. });
  696. ArduinoOTA.onEnd([]() {
  697. #ifdef DEBUG
  698. Serial.println("OTA - End");
  699. #endif
  700. #if ENABLE_RF
  701. RemoteReceiver::enable();
  702. #endif
  703. });
  704. ArduinoOTA.onProgress([](unsigned int progress, unsigned int total) {
  705. #ifdef DEBUG
  706. Serial.printf("Progress: %u%%\r", (progress / (total / 100)));
  707. #endif
  708. });
  709. ArduinoOTA.onError([](ota_error_t error) {
  710. #ifdef DEBUG
  711. Serial.printf("Error[%u]: ", error);
  712. if (error == OTA_AUTH_ERROR) Serial.println("Auth Failed");
  713. else if (error == OTA_BEGIN_ERROR) Serial.println("Begin Failed");
  714. else if (error == OTA_CONNECT_ERROR) Serial.println("Connect Failed");
  715. else if (error == OTA_RECEIVE_ERROR) Serial.println("Receive Failed");
  716. else if (error == OTA_END_ERROR) Serial.println("End Failed");
  717. #endif
  718. });
  719. ArduinoOTA.begin();
  720. }
  721. void OTALoop() {
  722. ArduinoOTA.handle();
  723. }
  724. #endif
  725. // -----------------------------------------------------------------------------
  726. // Energy Monitor
  727. // -----------------------------------------------------------------------------
  728. #if ENABLE_ENERGYMONITOR
  729. void energyMonitorSetup() {
  730. monitor.initCurrent(CURRENT_PIN, REFERENCE_VOLTAGE, CURRENT_RATIO);
  731. monitor.setPrecision(CURRENT_PRECISION);
  732. }
  733. void energyMonitorLoop() {
  734. static unsigned long next_measurement = millis();
  735. static byte measurements = 0;
  736. static double sum = 0;
  737. if (millis() > next_measurement) {
  738. double current = monitor.getCurrent(SAMPLES_X_MEASUREMENT);
  739. sum += current;
  740. ++measurements;
  741. #ifdef DEBUG
  742. Serial.print("Power reading: ");
  743. Serial.println(current * MAINS_VOLTAGE);
  744. #endif
  745. if (measurements == MEASUREMENTS_X_MESSAGE) {
  746. char buffer[8];
  747. sprintf(buffer, "%d", int(sum * MAINS_VOLTAGE / measurements));
  748. #ifdef DEBUG
  749. Serial.print("Power sending: ");
  750. Serial.println(buffer);
  751. #endif
  752. #if ENABLE_MQTT
  753. mqtt.publish(mqttPowerTopic, buffer, MQTT_RETAIN);
  754. #endif
  755. sum = 0;
  756. measurements = 0;
  757. }
  758. next_measurement += MEASUREMENT_INTERVAL;
  759. }
  760. }
  761. #endif
  762. // -----------------------------------------------------------------------------
  763. // Hardware (buttons, LEDs,...)
  764. // -----------------------------------------------------------------------------
  765. void hardwareSetup() {
  766. Serial.begin(115200);
  767. pinMode(RELAY_PIN, OUTPUT);
  768. pinMode(LED_PIN, OUTPUT);
  769. pinMode(RF_PIN, INPUT_PULLUP);
  770. button1 = DebounceEvent(BUTTON_PIN);
  771. SPIFFS.begin();
  772. EEPROM.begin(1);
  773. EEPROM.read(0) == 1 ? switchRelayOn() : switchRelayOff();
  774. }
  775. void blink(unsigned long delayOff, unsigned long delayOn) {
  776. static unsigned long next = millis();
  777. static bool status = HIGH;
  778. if (next < millis()) {
  779. status = !status;
  780. digitalWrite(LED_PIN, status);
  781. next += ((status) ? delayOff : delayOn);
  782. }
  783. }
  784. void showStatus() {
  785. if (WiFi.status() == WL_CONNECTED) {
  786. blink(5000, 500);
  787. } else {
  788. blink(500, 500);
  789. }
  790. }
  791. void hardwareLoop() {
  792. if (button1.loop()) {
  793. if (button1.getEvent() == EVENT_SINGLE_CLICK) toggleRelay();
  794. if (button1.getEvent() == EVENT_DOUBLE_CLICK) wifiSetupAP();
  795. if (button1.getEvent() == EVENT_LONG_CLICK) ESP.reset();
  796. }
  797. showStatus();
  798. }
  799. // -----------------------------------------------------------------------------
  800. // Booting
  801. // -----------------------------------------------------------------------------
  802. void welcome() {
  803. Serial.println();
  804. Serial.println(APP_NAME);
  805. Serial.println(APP_WEBSITE);
  806. Serial.println(APP_AUTHOR);
  807. Serial.println();
  808. Serial.print("Device: ");
  809. Serial.println(getIdentifier());
  810. Serial.print("Last reset reason: ");
  811. Serial.println(ESP.getResetReason());
  812. Serial.println();
  813. }
  814. void setup() {
  815. hardwareSetup();
  816. welcome();
  817. #if ENABLE_OTA
  818. OTASetup();
  819. #endif
  820. loadConfig();
  821. wifiSetupSTA(false);
  822. #if ENABLE_WEBSERVER
  823. webServerSetup();
  824. #endif
  825. #if ENABLE_MQTT
  826. mqttSetup();
  827. #endif
  828. #if ENABLE_RF
  829. rfSetup();
  830. #endif
  831. #if ENABLE_ENERGYMONITOR
  832. energyMonitorSetup();
  833. #endif
  834. }
  835. void loop() {
  836. #if ENABLE_OTA
  837. OTALoop();
  838. #endif
  839. wifiLoop();
  840. #if ENABLE_MQTT
  841. mqttLoop();
  842. #endif
  843. #if ENABLE_RF
  844. rfLoop();
  845. #endif
  846. #if ENABLE_WEBSERVER
  847. webServerLoop();
  848. #endif
  849. #if ENABLE_ENERGYMONITOR
  850. energyMonitorLoop();
  851. #endif
  852. hardwareLoop();
  853. delay(1);
  854. }