393 lines
15 KiB
C++
393 lines
15 KiB
C++
#include "webserver_ota.h"
|
|
|
|
// esp-idf includes
|
|
#include <esp_http_server.h>
|
|
#include <esp_log.h>
|
|
#include <esp_ota_ops.h>
|
|
|
|
// 3rdparty lib includes
|
|
#include <htmlbuilder.h>
|
|
#include <fmt/core.h>
|
|
#include <espcppmacros.h>
|
|
#include <esphttpdutils.h>
|
|
#include <lockhelper.h>
|
|
#include <tickchrono.h>
|
|
#include <strutils.h>
|
|
|
|
// local includes
|
|
#include "ota.h"
|
|
#include "webserver_lock.h"
|
|
#include "globals.h"
|
|
#include "newsettings.h"
|
|
|
|
using namespace std::chrono_literals;
|
|
using esphttpdutils::HtmlTag;
|
|
|
|
namespace {
|
|
constexpr const char * const TAG = "BOBBYWEB";
|
|
} // namespace
|
|
|
|
esp_err_t webserver_ota_percentage_handler(httpd_req_t *req)
|
|
{
|
|
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Access-Control-Allow-Origin", "http://web.bobbycar.cloud");
|
|
|
|
std::string body;
|
|
|
|
std::string wants_json_query;
|
|
if (auto result = esphttpdutils::webserver_get_query(req))
|
|
wants_json_query = *result;
|
|
else
|
|
{
|
|
ESP_LOGE(TAG, "%.*s", result.error().size(), result.error().data());
|
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", result.error());
|
|
}
|
|
|
|
char tmpBuf[256];
|
|
|
|
const auto key_result = httpd_query_key_value(wants_json_query.data(), "json", tmpBuf, 256);
|
|
if (key_result == ESP_OK && (configs.webserverPassword.value().empty() || configs.webserverPassword.value() == tmpBuf))
|
|
{
|
|
body += "{";
|
|
if (asyncOta)
|
|
{
|
|
const auto progress = asyncOta->progress();
|
|
const auto totalSize = asyncOta->totalSize();
|
|
|
|
body += fmt::format("\"cur_ota_percent\":\"{}\",", (totalSize && *totalSize > 0) ? fmt::format("{:.02f}", float(progress) / *totalSize * 100) : "?");
|
|
}
|
|
else
|
|
{
|
|
body += "\"info\":\"Updater is not constructed.\"";
|
|
}
|
|
|
|
body += "}";
|
|
}
|
|
else if (key_result != ESP_ERR_NOT_FOUND && configs.webserverPassword.value() == tmpBuf)
|
|
{
|
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::Unauthorized, "text/plain", "");
|
|
}
|
|
|
|
size_t lastGesch = body.rfind("},");
|
|
if (std::string::npos != lastGesch)
|
|
body = body.erase(lastGesch+1, 1);
|
|
|
|
size_t lastEckig = body.rfind("],");
|
|
if (std::string::npos != lastEckig)
|
|
body = body.erase(lastEckig+1, 1);
|
|
|
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::Ok, (key_result == ESP_OK) ? "application/json" : "text/html", body)
|
|
}
|
|
|
|
esp_err_t webserver_ota_handler(httpd_req_t *req)
|
|
{
|
|
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Access-Control-Allow-Origin", "http://web.bobbycar.cloud");
|
|
|
|
std::string body;
|
|
|
|
std::string wants_json_query;
|
|
if (auto result = esphttpdutils::webserver_get_query(req))
|
|
wants_json_query = *result;
|
|
else
|
|
{
|
|
ESP_LOGE(TAG, "%.*s", result.error().size(), result.error().data());
|
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", result.error());
|
|
}
|
|
|
|
char tmpBuf[256];
|
|
const auto key_result = httpd_query_key_value(wants_json_query.data(), "json", tmpBuf, 256);
|
|
if (key_result == ESP_OK && (configs.webserverPassword.value().empty() || configs.webserverPassword.value() == tmpBuf))
|
|
{
|
|
|
|
body += "{";
|
|
|
|
if (const esp_app_desc_t *app_desc = esp_ota_get_app_description())
|
|
{
|
|
body += fmt::format("\"cur_name\":\"{}\",", app_desc->project_name);
|
|
body += fmt::format("\"cur_ver\":\"{}\",", app_desc->version);
|
|
body += fmt::format("\"cur_secver\":\"{}\",", app_desc->secure_version);
|
|
body += fmt::format("\"cur_ts\":\"{}\",", app_desc->time);
|
|
body += fmt::format("\"cur_idf\":\"{}\",", app_desc->idf_ver);
|
|
body += fmt::format("\"cur_sha\":\"{}\",", cpputils::toHexString({app_desc->app_elf_sha256, 8}));
|
|
}
|
|
else
|
|
{
|
|
body += "\"err\":\"Could not access esp_ota_get_app_description()\",";
|
|
}
|
|
|
|
body += "\"updater\":{";
|
|
|
|
if (asyncOta)
|
|
{
|
|
body += fmt::format("\"status\":\"{}\"", toString(asyncOta->status()));
|
|
|
|
if (const auto &appDesc = asyncOta->appDesc())
|
|
{
|
|
const auto progress = asyncOta->progress();
|
|
const auto totalSize = asyncOta->totalSize();
|
|
|
|
body += fmt::format("\"cur_ota_percent\":\"{}\",", (totalSize && *totalSize > 0) ? fmt::format("{:.02f}", float(progress) / *totalSize * 100) : "?");
|
|
body += fmt::format("\"cur_ota_progress\":\"{}\",", progress);
|
|
body += fmt::format("\"cur_ota_total\":\"{}\",", totalSize ? std::to_string(*totalSize) : "?");
|
|
body += fmt::format("\"new_name\":\"{}\",", appDesc->project_name);
|
|
body += fmt::format("\"new_ver\":\"{}\",", appDesc->version);
|
|
body += fmt::format("\"new_secver\":\"{}\",", appDesc->secure_version);
|
|
body += fmt::format("\"new_ts\":\"{}\",", appDesc->time);
|
|
body += fmt::format("\"new_idf\":\"{}\",", appDesc->idf_ver);
|
|
body += fmt::format("\"new_sha\":\"{}\",", cpputils::toHexString({appDesc->app_elf_sha256, 8}));
|
|
|
|
}
|
|
else
|
|
{
|
|
body += "\"err\":\"Could not access asyncOta->appDesc()\",";
|
|
}
|
|
}
|
|
else
|
|
{
|
|
body += "\"info\":\"Updater is not constructed.\"";
|
|
}
|
|
|
|
body += "}}";
|
|
size_t lastGesch = body.rfind("},");
|
|
if (std::string::npos != lastGesch)
|
|
body = body.erase(lastGesch+1, 1);
|
|
|
|
size_t lastEckig = body.rfind("],");
|
|
if (std::string::npos != lastEckig)
|
|
body = body.erase(lastEckig+1, 1);
|
|
}
|
|
else if (key_result != ESP_ERR_NOT_FOUND && configs.webserverPassword.value() == tmpBuf)
|
|
{
|
|
|
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::Unauthorized, "text/plain", "");
|
|
}
|
|
else
|
|
{
|
|
HtmlTag htmlTag{"html", body};
|
|
|
|
{
|
|
HtmlTag headTag{"head", body};
|
|
|
|
{
|
|
HtmlTag titleTag{"title", body};
|
|
body += "Update";
|
|
}
|
|
|
|
body += "<meta name=\"viewport\" content=\"width=device-width, initial-scale=1, shrink-to-fit=no\" />";
|
|
}
|
|
|
|
{
|
|
HtmlTag bodyTag{"body", body};
|
|
|
|
{
|
|
HtmlTag h1Tag{"h1", body};
|
|
body += "Update";
|
|
}
|
|
|
|
{
|
|
HtmlTag pTag{"p", body};
|
|
body += "<a href=\"/\">Display control</a> - "
|
|
"<b>Update</b> - "
|
|
"<a href=\"/settings\">Settings</a> - "
|
|
"<a href=\"/newSettings\">New Settings</a> - "
|
|
"<a href=\"/dumpnvs\">Dump NVS</a>";
|
|
}
|
|
|
|
if (const esp_app_desc_t *app_desc = esp_ota_get_app_description())
|
|
{
|
|
HtmlTag tableTag{"table", "border=\"1\"", body};
|
|
|
|
{
|
|
HtmlTag trTag{"tr", body};
|
|
{ HtmlTag tdTag{"td", body}; body += "Current project_name"; }
|
|
{ HtmlTag tdTag{"td", body}; body += esphttpdutils::htmlentities(app_desc->project_name); }
|
|
}
|
|
|
|
{
|
|
HtmlTag trTag{"tr", body};
|
|
{ HtmlTag tdTag{"td", body}; body += "Current version"; }
|
|
{ HtmlTag tdTag{"td", body}; body += esphttpdutils::htmlentities(app_desc->version); }
|
|
}
|
|
|
|
{
|
|
HtmlTag trTag{"tr", body};
|
|
{ HtmlTag tdTag{"td", body}; body += "Current secure_version"; }
|
|
{ HtmlTag tdTag{"td", body}; body += std::to_string(app_desc->secure_version); }
|
|
}
|
|
|
|
{
|
|
HtmlTag trTag{"tr", body};
|
|
{ HtmlTag tdTag{"td", body}; body += "Current timestamp"; }
|
|
{ HtmlTag tdTag{"td", body}; body += esphttpdutils::htmlentities(fmt::format("{} {}", app_desc->date, app_desc->time)); }
|
|
}
|
|
|
|
{
|
|
HtmlTag trTag{"tr", body};
|
|
{ HtmlTag tdTag{"td", body}; body += "Current idf_ver"; }
|
|
{ HtmlTag tdTag{"td", body}; body += esphttpdutils::htmlentities(app_desc->idf_ver); }
|
|
}
|
|
|
|
{
|
|
HtmlTag trTag{"tr", body};
|
|
{ HtmlTag tdTag{"td", body}; body += "Current sha256"; }
|
|
{ HtmlTag tdTag{"td", body}; body += esphttpdutils::htmlentities(cpputils::toHexString({app_desc->app_elf_sha256, 8})); }
|
|
}
|
|
}
|
|
else
|
|
{
|
|
constexpr const std::string_view msg = "esp_ota_get_app_description() failed";
|
|
ESP_LOGE(TAG, "%.*s", msg.size(), msg.data());
|
|
HtmlTag pTag{"p", "style=\"color: red;\"", body};
|
|
body += esphttpdutils::htmlentities(msg);
|
|
}
|
|
|
|
if (asyncOta)
|
|
{
|
|
HtmlTag tableTag{"table", "border=\"1\"", body};
|
|
|
|
{
|
|
HtmlTag trTag{"tr", body};
|
|
{ HtmlTag tdTag{"td", body}; body += "Update status"; }
|
|
{ HtmlTag tdTag{"td", body}; body += esphttpdutils::htmlentities(toString(asyncOta->status())); }
|
|
}
|
|
|
|
{
|
|
HtmlTag trTag{"tr", body};
|
|
{ HtmlTag tdTag{"td", body}; body += "Update progress"; }
|
|
{
|
|
HtmlTag tdTag{"td", body};
|
|
const auto progress = asyncOta->progress();
|
|
const auto totalSize = asyncOta->totalSize();
|
|
body += fmt::format("{} / {}{}",
|
|
progress,
|
|
totalSize ? std::to_string(*totalSize) : "?",
|
|
(totalSize && *totalSize > 0) ? fmt::format(" ({:.02f}%)", float(progress) / *totalSize * 100) : "");
|
|
}
|
|
}
|
|
|
|
{
|
|
HtmlTag trTag{"tr", body};
|
|
{ HtmlTag tdTag{"td", body}; body += "Update message"; }
|
|
{ HtmlTag tdTag{"td", body}; body += esphttpdutils::htmlentities(asyncOta->message()); }
|
|
}
|
|
|
|
if (const auto &appDesc = asyncOta->appDesc())
|
|
{
|
|
{
|
|
HtmlTag trTag{"tr", body};
|
|
{ HtmlTag tdTag{"td", body}; body += "New project_name"; }
|
|
{ HtmlTag tdTag{"td", body}; body += esphttpdutils::htmlentities(appDesc->project_name); }
|
|
}
|
|
|
|
{
|
|
HtmlTag trTag{"tr", body};
|
|
{ HtmlTag tdTag{"td", body}; body += "New version"; }
|
|
{ HtmlTag tdTag{"td", body}; body += esphttpdutils::htmlentities(appDesc->version); }
|
|
}
|
|
|
|
{
|
|
HtmlTag trTag{"tr", body};
|
|
{ HtmlTag tdTag{"td", body}; body += "New secure_version"; }
|
|
{ HtmlTag tdTag{"td", body}; body += std::to_string(appDesc->secure_version); }
|
|
}
|
|
|
|
{
|
|
HtmlTag trTag{"tr", body};
|
|
{ HtmlTag tdTag{"td", body}; body += "New timestamp"; }
|
|
{ HtmlTag tdTag{"td", body}; body += esphttpdutils::htmlentities(fmt::format("{} {}", appDesc->date, appDesc->time)); }
|
|
}
|
|
|
|
{
|
|
HtmlTag trTag{"tr", body};
|
|
{ HtmlTag tdTag{"td", body}; body += "New idf_ver"; }
|
|
{ HtmlTag tdTag{"td", body}; body += esphttpdutils::htmlentities(appDesc->idf_ver); }
|
|
}
|
|
|
|
{
|
|
HtmlTag trTag{"tr", body};
|
|
{ HtmlTag tdTag{"td", body}; body += "New sha256"; }
|
|
{ HtmlTag tdTag{"td", body}; body += esphttpdutils::htmlentities(cpputils::toHexString({appDesc->app_elf_sha256, 8})); }
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
HtmlTag pTag{"p", body};
|
|
body += "Updater is sleeping (not constructed)";
|
|
}
|
|
|
|
{
|
|
HtmlTag formTag{"form", "action=\"/triggerOta\" method=\"GET\"", body};
|
|
HtmlTag fieldsetTag{"fieldset", body};
|
|
{
|
|
HtmlTag legendTag{"legend", body};
|
|
body += "Trigger Update";
|
|
}
|
|
|
|
body += fmt::format("<input type=\"text\" name=\"url\" value=\"{}\" />", esphttpdutils::htmlentities(configs.otaUrl.value()));
|
|
|
|
{
|
|
HtmlTag buttonTag{"button", "type=\"submit\"", body};
|
|
body += "Go";
|
|
}
|
|
|
|
body += "url is only used temporarely and not persisted in flash";
|
|
}
|
|
}
|
|
}
|
|
|
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::Ok, (key_result == ESP_OK) ? "application/json" : "text/html", body)
|
|
}
|
|
|
|
esp_err_t webserver_trigger_ota_handler(httpd_req_t *req)
|
|
{
|
|
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Access-Control-Allow-Origin", "http://web.bobbycar.cloud");
|
|
|
|
std::string query;
|
|
if (auto result = esphttpdutils::webserver_get_query(req))
|
|
query = *result;
|
|
else
|
|
{
|
|
ESP_LOGE(TAG, "%.*s", result.error().size(), result.error().data());
|
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", result.error());
|
|
}
|
|
|
|
std::string url;
|
|
constexpr const std::string_view urlParamName{"url"};
|
|
|
|
{
|
|
char valueBufEncoded[256];
|
|
if (const auto result = httpd_query_key_value(query.data(), urlParamName.data(), valueBufEncoded, 256); result != ESP_OK)
|
|
{
|
|
if (result == ESP_ERR_NOT_FOUND)
|
|
{
|
|
const auto msg = fmt::format("{} not set", urlParamName);
|
|
ESP_LOGW(TAG, "%.*s", msg.size(), msg.data());
|
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", msg);
|
|
}
|
|
else
|
|
{
|
|
const auto msg = fmt::format("httpd_query_key_value() {} failed with {}", urlParamName, esp_err_to_name(result));
|
|
ESP_LOGE(TAG, "%.*s", msg.size(), msg.data());
|
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", msg);
|
|
}
|
|
}
|
|
|
|
char valueBuf[257];
|
|
esphttpdutils::urldecode(valueBuf, valueBufEncoded);
|
|
|
|
url = valueBuf;
|
|
}
|
|
|
|
if (const auto result = triggerOta(url); !result)
|
|
{
|
|
ESP_LOGE(TAG, "%.*s", result.error().size(), result.error().data());
|
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::BadRequest, "text/plain", result.error());
|
|
}
|
|
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/ota")
|
|
CALL_AND_EXIT(esphttpdutils::webserver_resp_send, req, esphttpdutils::ResponseStatus::TemporaryRedirect, "text/html", "Ok, continue at <a href=\"/ota\">/</a>")
|
|
}
|