mirror of
https://github.com/esphome/esphome.git
synced 2025-09-06 21:32:21 +01:00
230 lines
8.0 KiB
C++
230 lines
8.0 KiB
C++
#include "ota_web_server.h"
|
|
#ifdef USE_WEBSERVER_OTA
|
|
|
|
#include "esphome/components/ota/ota_backend.h"
|
|
#include "esphome/core/application.h"
|
|
#include "esphome/core/log.h"
|
|
|
|
#ifdef USE_CAPTIVE_PORTAL
|
|
#include "esphome/components/captive_portal/captive_portal.h"
|
|
#endif
|
|
|
|
#ifdef USE_ARDUINO
|
|
#ifdef USE_ESP8266
|
|
#include <Updater.h>
|
|
#elif defined(USE_ESP32) || defined(USE_LIBRETINY)
|
|
#include <Update.h>
|
|
#endif
|
|
#endif // USE_ARDUINO
|
|
|
|
namespace esphome {
|
|
namespace web_server {
|
|
|
|
static const char *const TAG = "web_server.ota";
|
|
|
|
class OTARequestHandler : public AsyncWebHandler {
|
|
public:
|
|
OTARequestHandler(WebServerOTAComponent *parent) : parent_(parent) {}
|
|
void handleRequest(AsyncWebServerRequest *request) override;
|
|
void handleUpload(AsyncWebServerRequest *request, const String &filename, size_t index, uint8_t *data, size_t len,
|
|
bool final) override;
|
|
bool canHandle(AsyncWebServerRequest *request) const override {
|
|
// Check if this is an OTA update request
|
|
bool is_ota_request = request->url() == "/update" && request->method() == HTTP_POST;
|
|
|
|
#if defined(USE_WEBSERVER_OTA_DISABLED) && defined(USE_CAPTIVE_PORTAL)
|
|
// IMPORTANT: USE_WEBSERVER_OTA_DISABLED only disables OTA for the web_server component
|
|
// Captive portal can still perform OTA updates - check if request is from active captive portal
|
|
// Note: global_captive_portal is the standard way components communicate in ESPHome
|
|
return is_ota_request && captive_portal::global_captive_portal != nullptr &&
|
|
captive_portal::global_captive_portal->is_active();
|
|
#elif defined(USE_WEBSERVER_OTA_DISABLED)
|
|
// OTA disabled for web_server and no captive portal compiled in
|
|
return false;
|
|
#else
|
|
// OTA enabled for web_server
|
|
return is_ota_request;
|
|
#endif
|
|
}
|
|
|
|
// NOLINTNEXTLINE(readability-identifier-naming)
|
|
bool isRequestHandlerTrivial() const override { return false; }
|
|
|
|
protected:
|
|
void report_ota_progress_(AsyncWebServerRequest *request);
|
|
void schedule_ota_reboot_();
|
|
void ota_init_(const char *filename);
|
|
|
|
uint32_t last_ota_progress_{0};
|
|
uint32_t ota_read_length_{0};
|
|
WebServerOTAComponent *parent_;
|
|
bool ota_success_{false};
|
|
|
|
private:
|
|
std::unique_ptr<ota::OTABackend> ota_backend_{nullptr};
|
|
};
|
|
|
|
void OTARequestHandler::report_ota_progress_(AsyncWebServerRequest *request) {
|
|
const uint32_t now = millis();
|
|
if (now - this->last_ota_progress_ > 1000) {
|
|
float percentage = 0.0f;
|
|
if (request->contentLength() != 0) {
|
|
// Note: Using contentLength() for progress calculation is technically wrong as it includes
|
|
// multipart headers/boundaries, but it's only off by a small amount and we don't have
|
|
// access to the actual firmware size until the upload is complete. This is intentional
|
|
// as it still gives the user a reasonable progress indication.
|
|
percentage = (this->ota_read_length_ * 100.0f) / request->contentLength();
|
|
ESP_LOGD(TAG, "OTA in progress: %0.1f%%", percentage);
|
|
} else {
|
|
ESP_LOGD(TAG, "OTA in progress: %" PRIu32 " bytes read", this->ota_read_length_);
|
|
}
|
|
#ifdef USE_OTA_STATE_CALLBACK
|
|
// Report progress - use call_deferred since we're in web server task
|
|
this->parent_->state_callback_.call_deferred(ota::OTA_IN_PROGRESS, percentage, 0);
|
|
#endif
|
|
this->last_ota_progress_ = now;
|
|
}
|
|
}
|
|
|
|
void OTARequestHandler::schedule_ota_reboot_() {
|
|
ESP_LOGI(TAG, "OTA update successful!");
|
|
this->parent_->set_timeout(100, []() {
|
|
ESP_LOGI(TAG, "Performing OTA reboot now");
|
|
App.safe_reboot();
|
|
});
|
|
}
|
|
|
|
void OTARequestHandler::ota_init_(const char *filename) {
|
|
ESP_LOGI(TAG, "OTA Update Start: %s", filename);
|
|
this->ota_read_length_ = 0;
|
|
this->ota_success_ = false;
|
|
}
|
|
|
|
void OTARequestHandler::handleUpload(AsyncWebServerRequest *request, const String &filename, size_t index,
|
|
uint8_t *data, size_t len, bool final) {
|
|
ota::OTAResponseTypes error_code = ota::OTA_RESPONSE_OK;
|
|
|
|
if (index == 0 && !this->ota_backend_) {
|
|
// Initialize OTA on first call
|
|
this->ota_init_(filename.c_str());
|
|
|
|
#ifdef USE_OTA_STATE_CALLBACK
|
|
// Notify OTA started - use call_deferred since we're in web server task
|
|
this->parent_->state_callback_.call_deferred(ota::OTA_STARTED, 0.0f, 0);
|
|
#endif
|
|
|
|
// Platform-specific pre-initialization
|
|
#ifdef USE_ARDUINO
|
|
#ifdef USE_ESP8266
|
|
Update.runAsync(true);
|
|
#endif
|
|
#if defined(USE_ESP32) || defined(USE_LIBRETINY)
|
|
if (Update.isRunning()) {
|
|
Update.abort();
|
|
}
|
|
#endif
|
|
#endif // USE_ARDUINO
|
|
|
|
this->ota_backend_ = ota::make_ota_backend();
|
|
if (!this->ota_backend_) {
|
|
ESP_LOGE(TAG, "Failed to create OTA backend");
|
|
#ifdef USE_OTA_STATE_CALLBACK
|
|
this->parent_->state_callback_.call_deferred(ota::OTA_ERROR, 0.0f,
|
|
static_cast<uint8_t>(ota::OTA_RESPONSE_ERROR_UNKNOWN));
|
|
#endif
|
|
return;
|
|
}
|
|
|
|
// Web server OTA uses multipart uploads where the actual firmware size
|
|
// is unknown (contentLength includes multipart overhead)
|
|
// Pass 0 to indicate unknown size
|
|
error_code = this->ota_backend_->begin(0);
|
|
if (error_code != ota::OTA_RESPONSE_OK) {
|
|
ESP_LOGE(TAG, "OTA begin failed: %d", error_code);
|
|
this->ota_backend_.reset();
|
|
#ifdef USE_OTA_STATE_CALLBACK
|
|
this->parent_->state_callback_.call_deferred(ota::OTA_ERROR, 0.0f, static_cast<uint8_t>(error_code));
|
|
#endif
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (!this->ota_backend_) {
|
|
return;
|
|
}
|
|
|
|
// Process data
|
|
if (len > 0) {
|
|
error_code = this->ota_backend_->write(data, len);
|
|
if (error_code != ota::OTA_RESPONSE_OK) {
|
|
ESP_LOGE(TAG, "OTA write failed: %d", error_code);
|
|
this->ota_backend_->abort();
|
|
this->ota_backend_.reset();
|
|
#ifdef USE_OTA_STATE_CALLBACK
|
|
this->parent_->state_callback_.call_deferred(ota::OTA_ERROR, 0.0f, static_cast<uint8_t>(error_code));
|
|
#endif
|
|
return;
|
|
}
|
|
this->ota_read_length_ += len;
|
|
this->report_ota_progress_(request);
|
|
}
|
|
|
|
// Finalize
|
|
if (final) {
|
|
ESP_LOGD(TAG, "OTA final chunk: index=%zu, len=%zu, total_read=%" PRIu32 ", contentLength=%zu", index, len,
|
|
this->ota_read_length_, request->contentLength());
|
|
|
|
// For Arduino framework, the Update library tracks expected size from firmware header
|
|
// If we haven't received enough data, calling end() will fail
|
|
// This can happen if the upload is interrupted or the client disconnects
|
|
error_code = this->ota_backend_->end();
|
|
if (error_code == ota::OTA_RESPONSE_OK) {
|
|
this->ota_success_ = true;
|
|
#ifdef USE_OTA_STATE_CALLBACK
|
|
// Report completion before reboot - use call_deferred since we're in web server task
|
|
this->parent_->state_callback_.call_deferred(ota::OTA_COMPLETED, 100.0f, 0);
|
|
#endif
|
|
this->schedule_ota_reboot_();
|
|
} else {
|
|
ESP_LOGE(TAG, "OTA end failed: %d", error_code);
|
|
#ifdef USE_OTA_STATE_CALLBACK
|
|
this->parent_->state_callback_.call_deferred(ota::OTA_ERROR, 0.0f, static_cast<uint8_t>(error_code));
|
|
#endif
|
|
}
|
|
this->ota_backend_.reset();
|
|
}
|
|
}
|
|
|
|
void OTARequestHandler::handleRequest(AsyncWebServerRequest *request) {
|
|
AsyncWebServerResponse *response;
|
|
// Use the ota_success_ flag to determine the actual result
|
|
const char *msg = this->ota_success_ ? "Update Successful!" : "Update Failed!";
|
|
response = request->beginResponse(200, "text/plain", msg);
|
|
response->addHeader("Connection", "close");
|
|
request->send(response);
|
|
}
|
|
|
|
void WebServerOTAComponent::setup() {
|
|
// Get the global web server base instance and register our handler
|
|
auto *base = web_server_base::global_web_server_base;
|
|
if (base == nullptr) {
|
|
ESP_LOGE(TAG, "WebServerBase not found");
|
|
this->mark_failed();
|
|
return;
|
|
}
|
|
|
|
// AsyncWebServer takes ownership of the handler and will delete it when the server is destroyed
|
|
base->add_handler(new OTARequestHandler(this)); // NOLINT
|
|
#ifdef USE_OTA_STATE_CALLBACK
|
|
// Register with global OTA callback system
|
|
ota::register_ota_platform(this);
|
|
#endif
|
|
}
|
|
|
|
void WebServerOTAComponent::dump_config() { ESP_LOGCONFIG(TAG, "Web Server OTA"); }
|
|
|
|
} // namespace web_server
|
|
} // namespace esphome
|
|
|
|
#endif // USE_WEBSERVER_OTA
|