#include #include #include #include #include #include #include #include #include #include #include #define RESET_CONFIG_ON_FIRST_BOOT // ========================== // Firmware framework_4 // Mejoras: // - Publicación inicial retenida de estados (valores/) al conectar a MQTT // - Confirmación inmediata de comandos OUTPUT publicando valor actual (retained) // - Publicación de IP y state retained en cada conexión // - Descarga de puertos desde servidor con HTTPS (insecure), usando config si existe // - Ajustes menores de estabilidad y logs // ========================== ESP8266WebServer server(80); WiFiClient espClient; WiFiClientSecure secureClient; PubSubClient client(espClient); Ticker watchdogReset; Ticker verificadorWiFi; Ticker verificadorMQTT; Ticker tickerA0; std::map pines; std::map modosPines; std::map ultimoEstadoEntrada; std::map esAnalogico; String ssid, password, mqttUser, mqttPass, mqttServer, url_puertos_json; String wifiSSID = ""; unsigned long ultimoIntentoMQTT = 0; const unsigned long intervaloReintentoMQTT = 5000; int mqttPort = 1883; struct TopicsMQTT { String state; String comando; String ip; String valores; String reboot; }; TopicsMQTT topics; bool estadoInicialPublicado = false; int ultimoValorA0 = -1; const int UMBRAL_A0 = 10; // Flags para ejecutar tareas pesadas fuera del contexto de Ticker volatile bool flagCheckWiFi = false; volatile bool flagCheckMQTT = false; void onTickWiFi() { flagCheckWiFi = true; } void onTickMQTT() { flagCheckMQTT = true; } String getChipID() { String id = String(ESP.getChipId(), HEX); id.toUpperCase(); return id; } void inicializarTopicsMQTT() { topics.state = "dispositivo/" + getChipID() + "/state"; topics.comando = "dispositivo/" + getChipID() + "/comando"; topics.ip = "dispositivo/" + getChipID() + "/ip"; topics.valores = "dispositivo/" + getChipID() + "/valores"; // prefijo topics.reboot = topics.comando + "/reboot"; } void resetWatchdog() { ESP.wdtFeed(); } void setupOTA() { ArduinoOTA.setHostname("ESP-Dispositivo"); ArduinoOTA.onStart([]() { Serial.println("\n🔄 OTA iniciado..."); }); ArduinoOTA.onEnd([]() { Serial.println("\n✅ OTA completado"); }); ArduinoOTA.onError([](ota_error_t error) { Serial.printf("❌ OTA Error[%u]\n", error); }); ArduinoOTA.begin(); } void configurarHora() { configTime(-3 * 3600, 0, "pool.ntp.org", "time.nist.gov"); Serial.print("⏳ Sincronizando hora"); while (time(nullptr) < 100000) { Serial.print("."); delay(200); ESP.wdtFeed(); yield(); } Serial.println("\n✅ Hora sincronizada"); } void leerConfig() { if (!SPIFFS.exists("/config.json")) { Serial.println("Creando valores por defecto en config.json"); ssid = ""; password = ""; mqttServer = ""; mqttUser = ""; mqttPass = ""; url_puertos_json = ""; mqttPort = 1883; File f = SPIFFS.open("/config.json", "w"); if (f) { StaticJsonDocument<512> doc; doc["ssid"] = ssid; doc["password"] = password; doc["mqttServer"] = mqttServer; doc["mqttUser"] = mqttUser; doc["mqttPass"] = mqttPass; doc["mqttPort"] = mqttPort; doc["url_puertos_json"] = url_puertos_json; serializeJson(doc, f); f.close(); } return; } File file = SPIFFS.open("/config.json", "r"); if (!file) { Serial.println("No se pudo abrir config.json"); return; } StaticJsonDocument<512> doc; auto err = deserializeJson(doc, file); file.close(); if (err) { Serial.println("Error al parsear JSON config"); return; } ssid = doc["ssid" ] | ""; password = doc["password"] | ""; mqttServer = doc["mqttServer"] | ""; mqttUser = doc["mqttUser"] | ""; mqttPass = doc["mqttPass"] | ""; mqttPort = doc["mqttPort"] | 1883; url_puertos_json = doc["url_puertos_json"] | ""; } bool descargarYActualizarPuertos() { String urlBase = url_puertos_json.length() ? url_puertos_json : String("esp.penki.com.ar"); String urlCompleta = "https://" + urlBase + "/api/get_puertos.php?chipid=" + getChipID(); Serial.println("📡 Solicitando: " + urlCompleta); secureClient.setInsecure(); HTTPClient http; http.setAuthorization("api", "API#2025"); if (!http.begin(secureClient, urlCompleta)) { Serial.println("❌ Error al iniciar conexión HTTPS"); return false; } int httpCode = http.GET(); if (httpCode <= 0) { Serial.printf("❌ Error HTTP al intentar descargar: %d\n", httpCode); http.end(); return false; } Serial.printf("📥 HTTP %d recibido\n", httpCode); if (httpCode == HTTP_CODE_OK) { String body = http.getString(); http.end(); StaticJsonDocument<4096> docRemoto; DeserializationError err = deserializeJson(docRemoto, body); if (err) { Serial.println("❌ Error al parsear JSON remoto"); return false; } const char* fechaRemota = docRemoto["update"] | ""; if (SPIFFS.exists("/puertos.json")) { File fl = SPIFFS.open("/puertos.json", "r"); StaticJsonDocument<4096> docLocal; DeserializationError e2 = deserializeJson(docLocal, fl); fl.close(); if (!e2) { const char* fechaLocal = docLocal["update"] | ""; if (strcmp(fechaRemota, fechaLocal) <= 0) { Serial.println("✅ puertos.json local actualizado o más reciente"); return true; } } } File fn = SPIFFS.open("/puertos.json", "w"); if (!fn) { Serial.println("❌ No se pudo abrir puertos.json para guardar"); return false; } fn.print(body); fn.close(); Serial.println("✅ puertos.json actualizado desde servidor remoto"); return true; } return false; } void leerPuertos() { pines.clear(); modosPines.clear(); ultimoEstadoEntrada.clear(); esAnalogico.clear(); if (!SPIFFS.exists("/puertos.json")) { Serial.println("❌ puertos.json no existe"); return; } File file = SPIFFS.open("/puertos.json", "r"); if (!file) { Serial.println("❌ No se pudo abrir puertos.json"); return; } StaticJsonDocument<4096> doc; DeserializationError error = deserializeJson(doc, file); file.close(); if (error) { Serial.println("❌ Error al parsear puertos.json"); return; } for (JsonPair kv : doc.as()) { String key = kv.key().c_str(); if (key == "update") continue; JsonObject obj = kv.value().as(); if (!obj.containsKey("gpio") || !obj.containsKey("modo")) continue; String gpioStr = obj["gpio"].as(); uint8_t gpio = (gpioStr == "A0") ? A0 : (uint8_t) gpioStr.toInt(); String alias = key; String modoStr = obj["modo"].as(); uint8_t modoPin; if (modoStr == "INPUT") modoPin = INPUT; else if (modoStr == "OUTPUT") modoPin = OUTPUT; else if (modoStr == "INPUT_PULLUP") modoPin = INPUT_PULLUP; else { modoPin = INPUT_PULLUP; } pines[alias] = gpio; modosPines[alias] = modoPin; esAnalogico[alias] = (gpio == A0); Serial.printf("🔧 %s => GPIO %d, modo %s\n", alias.c_str(), gpio, modoStr.c_str()); } Serial.println("✅ Pines y modos cargados"); } void imprimirPuertosJson() { if (!SPIFFS.begin()) { Serial.println("Error al montar SPIFFS"); return; } if (!SPIFFS.exists("/puertos.json")) { Serial.println("Archivo puertos.json no existe"); return; } File file = SPIFFS.open("/puertos.json", "r"); if (!file) { Serial.println("No se pudo abrir puertos.json"); return; } Serial.println("Contenido de puertos.json:"); while (file.available()) { Serial.write(file.read()); } file.close(); Serial.println(); } void conectarWiFi() { WiFi.mode(WIFI_STA); WiFi.begin(ssid.c_str(), password.c_str()); Serial.print("Conectando a WiFi"); int intentos = 20; while (WiFi.status() != WL_CONNECTED && intentos-- > 0) { delay(500); Serial.print("."); ESP.wdtFeed(); yield(); } if (WiFi.status() == WL_CONNECTED) { Serial.println("\n✅ WiFi conectado!"); Serial.print("IP: "); Serial.println(WiFi.localIP()); wifiSSID = WiFi.SSID(); } else { Serial.println("\n❌ No se pudo conectar. Modo AP."); WiFi.softAP("ESP_Config", "12345678"); Serial.print("AP IP: "); Serial.println(WiFi.softAPIP()); server.begin(); } } void publicarEstadoAlias(const String& alias, int val, uint8_t modo) { if (esAnalogico[alias]) return; String estado = val == HIGH ? "ON" : "OFF"; String modoStr = (modo == INPUT) ? "INPUT" : (modo == OUTPUT ? "OUTPUT" : (modo == INPUT_PULLUP ? "INPUT_PULLUP" : "DESCONOCIDO")); StaticJsonDocument<128> doc; doc["valor"] = estado; doc["modo"] = modoStr; char buffer[128]; serializeJson(doc, buffer, sizeof(buffer)); String topic = topics.valores + "/" + alias; client.publish(topic.c_str(), buffer, true); } void publicarEstadoInicialEntradas() { for (auto const& it : pines) { const String alias = it.first; const uint8_t gpio = it.second; if (esAnalogico[alias]) continue; int val = digitalRead(gpio); ultimoEstadoEntrada[alias] = val; publicarEstadoAlias(alias, val, modosPines[alias]); Serial.printf("Publicado estado inicial %s => %s\n", alias.c_str(), (val == HIGH ? "ON" : "OFF")); } estadoInicialPublicado = true; } void publicarCambiosEntradas() { for (auto const& it : pines) { const String alias = it.first; const uint8_t gpio = it.second; if (esAnalogico[alias]) continue; int val = digitalRead(gpio); if (!ultimoEstadoEntrada.count(alias) || ultimoEstadoEntrada[alias] != val) { ultimoEstadoEntrada[alias] = val; publicarEstadoAlias(alias, val, modosPines[alias]); Serial.printf("📤 Cambio en %s: %s\n", alias.c_str(), (val == HIGH ? "ON" : "OFF")); } } } void conectarMQTT(); void callback(char* topic, uint8_t* payload, unsigned int length) { payload[length] = '\0'; String mensaje = String((char*)payload); Serial.printf("Mensaje recibido en topic: %s -> %s\n", topic, mensaje.c_str()); if (String(topic) == topics.reboot.c_str()) { if (mensaje == "1") { Serial.println("Reiniciando dispositivo..."); delay(100); ESP.restart(); return; } } if (String(topic) == topics.comando) { StaticJsonDocument<256> doc; DeserializationError err = deserializeJson(doc, mensaje); if (err) { Serial.println("Error al parsear comando JSON"); return; } String alias = doc["alias"] | ""; String valor = doc["valor"] | ""; if (pines.count(alias) && (modosPines[alias] == OUTPUT)) { uint8_t gpio = pines[alias]; if (valor == "ON") { digitalWrite(gpio, HIGH); } else if (valor == "OFF") { digitalWrite(gpio, LOW); } else { Serial.println("Valor desconocido, usar ON u OFF"); return; } // Confirmar estado publicado (retained) int valNow = digitalRead(gpio); ultimoEstadoEntrada[alias] = valNow; publicarEstadoAlias(alias, valNow, modosPines[alias]); Serial.printf("✔ Comando aplicado %s => %s\n", alias.c_str(), (valNow == HIGH ? "ON" : "OFF")); } else { Serial.println("Alias no encontrado o no es salida"); } } } void conectarMQTT() { if (client.connected()) return; if ((long)(millis() - ultimoIntentoMQTT) < intervaloReintentoMQTT) return; ultimoIntentoMQTT = millis(); client.setServer(mqttServer.c_str(), mqttPort); client.setCallback(callback); String clientId = "WemosClient-" + String(ESP.getChipId()); Serial.print("Conectando a MQTT: "); Serial.println(mqttServer); if (client.connect(clientId.c_str(), mqttUser.c_str(), mqttPass.c_str(), topics.state.c_str(), 1, true, "offline")) { Serial.println("✅ Conectado al broker MQTT"); String ip = WiFi.localIP().toString(); client.publish(topics.state.c_str(), "online", true); client.publish(topics.ip.c_str(), ip.c_str(), true); client.subscribe(topics.comando.c_str()); client.subscribe(topics.reboot.c_str()); client.publish(topics.reboot.c_str(), "0", true); // Publicar estados iniciales una vez tras conectar publicarEstadoInicialEntradas(); } else { Serial.print("❌ MQTT falló. Código: "); Serial.println(client.state()); } } void verificarConexionWiFi() { if (WiFi.status() != WL_CONNECTED) { Serial.println("⚠️ WiFi desconectado. Intentando reconectar..."); WiFi.disconnect(); WiFi.begin(ssid.c_str(), password.c_str()); unsigned long inicio = millis(); while (WiFi.status() != WL_CONNECTED && millis() - inicio < 10000) { delay(500); Serial.print("."); } if (WiFi.status() == WL_CONNECTED) { Serial.println("\n✅ Reconectado al WiFi!"); } else { Serial.println("\n❌ No se pudo reconectar. Reiniciando ESP..."); ESP.restart(); } } } int intentosMQTTFallidos = 0; const int MAX_INTENTOS_MQTT = 5; void verificarConexionMQTT() { if (WiFi.status() != WL_CONNECTED) return; if (!client.connected()) { Serial.println("⚠️ MQTT desconectado. Intentando reconectar..."); conectarMQTT(); if (client.connected()) { Serial.println("✅ Reconexion MQTT exitosa"); intentosMQTTFallidos = 0; } else { intentosMQTTFallidos++; Serial.printf("❌ Fallo reconexión MQTT (%d/%d)\n", intentosMQTTFallidos, MAX_INTENTOS_MQTT); if (intentosMQTTFallidos >= MAX_INTENTOS_MQTT) { Serial.println("🚨 Demasiados fallos MQTT. Reiniciando ESP..."); ESP.restart(); } } } else { intentosMQTTFallidos = 0; } } void publicarA0() { int valorActual = analogRead(A0); if (ultimoValorA0 == -1 || abs(valorActual - ultimoValorA0) > UMBRAL_A0) { ultimoValorA0 = valorActual; String topic = topics.valores + "/A0"; String payload = String(valorActual); client.publish(topic.c_str(), payload.c_str(), true); Serial.printf("Publicado cambio A0: %d\n", valorActual); } } void manejarRoot() { String pagina = "

Configuración WiFi

"; pagina += ""; pagina += ""; pagina += ""; pagina += ""; pagina += "
"; pagina += "
"; pagina += ""; pagina += "
"; server.send(200, "text/html", pagina); } void manejarGuardar() { ssid = server.arg("ssid"); password = server.arg("password"); mqttServer = server.arg("mqttServer"); mqttUser = server.arg("mqttUser"); mqttPass = server.arg("mqttPass"); mqttPort = server.arg("mqttPort").toInt(); mqttPort = constrain(mqttPort, 1, 65535); String nuevo_url = server.arg("url_puertos_json"); if (nuevo_url.length() > 0) { url_puertos_json = nuevo_url; } StaticJsonDocument<512> doc; doc["ssid"] = ssid; doc["password"] = password; doc["mqttServer"] = mqttServer; doc["mqttUser"] = mqttUser; doc["mqttPass"] = mqttPass; doc["mqttPort"] = mqttPort; doc["url_puertos_json"] = url_puertos_json; File file = SPIFFS.open("/config.json", "w"); serializeJson(doc, file); file.close(); server.send(200, "text/html", "

Guardado con éxito

Reinicia el ESP para aplicar cambios.

Volver"); } void manejarBootstrap() { if (SPIFFS.exists("/bootstrap.min.css")) { File file = SPIFFS.open("/bootstrap.min.css", "r"); server.streamFile(file, "text/css"); file.close(); } else { server.send(404, "text/plain", "No encontrado"); } } void setup() { Serial.begin(115200); ESP.wdtDisable(); ESP.wdtEnable(8000); // Alimentar WDT desde el inicio para evitar resets durante tareas largas watchdogReset.attach_ms(1000, resetWatchdog); if (!SPIFFS.begin()) { Serial.println("❌ Error al montar SPIFFS"); } else { Serial.println("✅ SPIFFS montado correctamente"); } #ifdef RESET_CONFIG_ON_FIRST_BOOT // Eliminar config.json solo en el primer arranque después de flashear if (!SPIFFS.exists("/firstboot.flag")) { Serial.println("🗑️ Primera ejecución detectada, eliminando config.json..."); SPIFFS.remove("/config.json"); File f = SPIFFS.open("/firstboot.flag", "w"); if (f) { f.print("ok"); f.close(); } } #endif leerConfig(); inicializarTopicsMQTT(); conectarWiFi(); if (WiFi.status() == WL_CONNECTED) { if (!descargarYActualizarPuertos()) { Serial.println("⚠️ Usando puertos.json local"); } leerPuertos(); } else { Serial.println("⚠️ No hay WiFi, usando puertos.json local"); } // Configurar pines y estados iniciales LOW para OUTPUT for (auto const& it : pines) { const String alias = it.first; const uint8_t gpio = it.second; uint8_t modo = modosPines.count(alias) ? modosPines[alias] : INPUT; pinMode(gpio, modo); if (modo == OUTPUT) { digitalWrite(gpio, LOW); } Serial.printf("Pin %d (%s) configurado modo %d\n", gpio, alias.c_str(), modo); } if (WiFi.status() == WL_CONNECTED) { conectarMQTT(); } setupOTA(); // Web server server.on("/", manejarRoot); server.on("/guardar", HTTP_POST, manejarGuardar); server.on("/bootstrap.min.css", HTTP_GET, manejarBootstrap); server.begin(); // Tickers verificadorWiFi.attach(60, onTickWiFi); verificadorMQTT.attach(15, onTickMQTT); tickerA0.attach_ms(50, publicarA0); // 50ms, con UMBRAL evita spam } void loop() { if (WiFi.status() == WL_CONNECTED) { client.loop(); conectarMQTT(); } server.handleClient(); ArduinoOTA.handle(); publicarCambiosEntradas(); // Ejecutar verificaciones disparadas por Tickers en el contexto del loop if (flagCheckWiFi) { flagCheckWiFi = false; verificarConexionWiFi(); } if (flagCheckMQTT) { flagCheckMQTT = false; verificarConexionMQTT(); } }