#include #include #include #include #define WIFI_SSID "yourSSID" #define WIFI_PASSWORD "yourpass" #define MQTT_HOST IPAddress(192, 168, 130, 10) #define MQTT_PORT 1883 #define UPDATE_TOPIC "device/firmware/set" WiFiEventHandler wifiConnectHandler; WiFiEventHandler wifiDisconnectHandler; espMqttClient mqttClient; Ticker reconnectTimer; bool disconnectFlag = false; bool restartFlag = false; void connectToWiFi() { Serial.println("Connecting to Wi-Fi..."); WiFi.begin(WIFI_SSID, WIFI_PASSWORD); } void connectToMqtt() { Serial.println("Connecting to MQTT..."); mqttClient.connect(); } void onWiFiConnect(const WiFiEventStationModeGotIP& event) { Serial.println("Connected to Wi-Fi."); connectToMqtt(); } void onWiFiDisconnect(const WiFiEventStationModeDisconnected& event) { Serial.println("Disconnected from Wi-Fi."); reconnectTimer.detach(); // ensure we don't reconnect to MQTT while reconnecting to Wi-Fi reconnectTimer.once(5, connectToWiFi); } void onMqttConnect(bool sessionPresent) { Serial.println("Connected to MQTT."); Serial.print("Session present: "); Serial.println(sessionPresent); uint16_t packetIdSub = mqttClient.subscribe(UPDATE_TOPIC, 2); Serial.print("Subscribing at QoS 2, packetId: "); Serial.println(packetIdSub); } void onMqttDisconnect(espMqttClientTypes::DisconnectReason reason) { Serial.printf("Disconnected from MQTT: %u.\n", static_cast(reason)); if (disconnectFlag) { restartFlag = true; return; } if (WiFi.isConnected()) { reconnectTimer.once(5, connectToMqtt); } } void onMqttSubscribe(uint16_t packetId, const espMqttClientTypes::SubscribeReturncode* codes, size_t len) { Serial.println("Subscribe acknowledged."); Serial.print(" packetId: "); Serial.println(packetId); for (size_t i = 0; i < len; ++i) { Serial.print(" qos: "); Serial.println(static_cast(codes[i])); } } void handleUpdate(const uint8_t* payload, size_t length, size_t index, size_t total) { // The Updater class takes a non-const pointer to write data although it doesn't change the data uint8_t* data = const_cast(payload); static size_t written = 0; Update.runAsync(true); if (index == 0) { if (Update.isRunning()) { Update.end(); Update.clearError(); } Update.begin(total); written = Update.write(data, length); Serial.printf("Updating %u/%u\n", written, Update.size()); } else { if (!Update.isRunning()) return; written += Update.write(data, length); Serial.printf("Updating %u/%u\n", written, Update.size()); } if (Update.isFinished()) { if (Update.end()) { Serial.println("Update succes"); disconnectFlag = true; } else { Serial.printf("Update error: %u\n", Update.getError()); Update.printError(Serial); Update.clearError(); } } } void onMqttMessage(const espMqttClientTypes::MessageProperties& properties, const char* topic, const uint8_t* payload, size_t len, size_t index, size_t total) { if (strcmp(UPDATE_TOPIC, topic) != 0) { Serial.println("Topic mismatch"); return; } handleUpdate(payload, len, index, total); } void setup() { Serial.begin(74880); Serial.println(); Serial.println(); wifiConnectHandler = WiFi.onStationModeGotIP(onWiFiConnect); wifiDisconnectHandler = WiFi.onStationModeDisconnected(onWiFiDisconnect); mqttClient.onConnect(onMqttConnect); mqttClient.onDisconnect(onMqttDisconnect); mqttClient.onSubscribe(onMqttSubscribe); mqttClient.onMessage(onMqttMessage); mqttClient.setServer(MQTT_HOST, MQTT_PORT); connectToWiFi(); } void loop() { mqttClient.loop(); if (disconnectFlag) { // it's safe to call this multiple times mqttClient.disconnect(); } if (restartFlag) { Serial.println("Rebooting... See you next time!"); Serial.flush(); ESP.reset(); } }