382 lines
13 KiB
C++
382 lines
13 KiB
C++
#pragma once
|
|
|
|
// system includes
|
|
#include <atomic>
|
|
#include <string_view>
|
|
|
|
// esp-idf includes
|
|
#ifdef FEATURE_WEBSERVER
|
|
#include <esp_http_server.h>
|
|
#endif
|
|
#include <esp_log.h>
|
|
|
|
// 3rdparty lib includes
|
|
#include <fmt/core.h>
|
|
#include <espcppmacros.h>
|
|
#include <espstrutils.h>
|
|
#include <strutils.h>
|
|
#include <numberparsing.h>
|
|
|
|
// local includes
|
|
#include "screens.h"
|
|
#include "textinterface.h"
|
|
#include "menudisplay.h"
|
|
#include "changevaluedisplay.h"
|
|
#include "displays/updatedisplay.h"
|
|
//#include "esputils.h"
|
|
#include "buttons.h"
|
|
#include "esphttpdutils.h"
|
|
|
|
namespace {
|
|
#ifdef FEATURE_WEBSERVER
|
|
httpd_handle_t httpdHandle;
|
|
|
|
std::atomic<bool> shouldReboot;
|
|
|
|
class HtmlTag {
|
|
public:
|
|
HtmlTag(std::string_view tagName, std::string &body) :
|
|
m_tagName{tagName},
|
|
m_body{body}
|
|
{
|
|
m_body += '<';
|
|
m_body += m_tagName;
|
|
m_body += '>';
|
|
}
|
|
|
|
HtmlTag(std::string_view tagName, std::string_view attributes, std::string &body) :
|
|
m_tagName{tagName},
|
|
m_body{body}
|
|
{
|
|
m_body += '<';
|
|
m_body += m_tagName;
|
|
if (!attributes.empty())
|
|
{
|
|
m_body += ' ';
|
|
m_body += attributes;
|
|
}
|
|
m_body += '>';
|
|
}
|
|
|
|
~HtmlTag()
|
|
{
|
|
m_body += "</";
|
|
m_body += m_tagName;
|
|
m_body += '>';
|
|
}
|
|
|
|
private:
|
|
const std::string_view m_tagName;
|
|
std::string &m_body;
|
|
};
|
|
|
|
template<typename T> T htmlentities(const T &val) { return val; } // TODO
|
|
template<typename T> T htmlentities(T &&val) { return val; } // TODO
|
|
|
|
esp_err_t webserver_root_handler(httpd_req_t *req);
|
|
esp_err_t webserver_up_handler(httpd_req_t *req);
|
|
esp_err_t webserver_down_handler(httpd_req_t *req);
|
|
esp_err_t webserver_confirm_handler(httpd_req_t *req);
|
|
esp_err_t webserver_back_handler(httpd_req_t *req);
|
|
esp_err_t webserver_triggerItem_handler(httpd_req_t *req);
|
|
esp_err_t webserver_setValue_handler(httpd_req_t *req);
|
|
esp_err_t webserver_reboot_handler(httpd_req_t *req);
|
|
|
|
void initWebserver()
|
|
{
|
|
shouldReboot = false;
|
|
|
|
{
|
|
httpd_config_t httpConfig HTTPD_DEFAULT_CONFIG();
|
|
httpConfig.core_id = 1;
|
|
|
|
const auto result = httpd_start(&httpdHandle, &httpConfig);
|
|
ESP_LOG_LEVEL_LOCAL((result == ESP_OK ? ESP_LOG_INFO : ESP_LOG_ERROR), TAG, "httpd_start(): %s", esp_err_to_name(result));
|
|
if (result != ESP_OK)
|
|
return;
|
|
}
|
|
|
|
for (const httpd_uri_t &uri : {
|
|
httpd_uri_t { .uri = "/", .method = HTTP_GET, .handler = webserver_root_handler, .user_ctx = NULL },
|
|
httpd_uri_t { .uri = "/up", .method = HTTP_GET, .handler = webserver_up_handler, .user_ctx = NULL },
|
|
httpd_uri_t { .uri = "/down", .method = HTTP_GET, .handler = webserver_down_handler, .user_ctx = NULL },
|
|
httpd_uri_t { .uri = "/confirm", .method = HTTP_GET, .handler = webserver_confirm_handler, .user_ctx = NULL },
|
|
httpd_uri_t { .uri = "/back", .method = HTTP_GET, .handler = webserver_back_handler, .user_ctx = NULL },
|
|
httpd_uri_t { .uri = "/triggerItem", .method = HTTP_GET, .handler = webserver_triggerItem_handler, .user_ctx = NULL },
|
|
httpd_uri_t { .uri = "/setValue", .method = HTTP_GET, .handler = webserver_setValue_handler, .user_ctx = NULL },
|
|
httpd_uri_t { .uri = "/reboot", .method = HTTP_GET, .handler = webserver_reboot_handler, .user_ctx = NULL }
|
|
})
|
|
{
|
|
const auto result = httpd_register_uri_handler(httpdHandle, &uri);
|
|
ESP_LOG_LEVEL_LOCAL((result == ESP_OK ? ESP_LOG_INFO : ESP_LOG_ERROR), TAG, "httpd_register_uri_handler() for %s: %s", uri.uri, esp_err_to_name(result));
|
|
//if (result != ESP_OK)
|
|
// return result;
|
|
}
|
|
}
|
|
|
|
void handleWebserver()
|
|
{
|
|
if (shouldReboot)
|
|
{
|
|
shouldReboot = false;
|
|
esp_restart();
|
|
}
|
|
}
|
|
|
|
esp_err_t webserver_root_handler(httpd_req_t *req)
|
|
{
|
|
std::string body;
|
|
|
|
{
|
|
HtmlTag htmlTag{"html", body};
|
|
|
|
{
|
|
HtmlTag headTag{"head", body};
|
|
|
|
{
|
|
HtmlTag titleTag{"title", body};
|
|
body += "Bobbycar remote";
|
|
}
|
|
|
|
body += "<meta name=\"viewport\" content=\"width=device-width, initial-scale=1, shrink-to-fit=no\" />";
|
|
}
|
|
|
|
{
|
|
HtmlTag bodyTag{"body", body};
|
|
|
|
{
|
|
HtmlTag h1Tag{"h1", body};
|
|
body += "Bobbycar remote";
|
|
}
|
|
|
|
{
|
|
HtmlTag pTag{"p", body};
|
|
body += "<a href=\"/up\">Up</a> "
|
|
"<a href=\"/down\">Down</a> "
|
|
"<a href=\"/confirm\">Confirm</a> "
|
|
"<a href=\"/back\">Back</a>";
|
|
}
|
|
|
|
if (auto constCurrentDisplay = static_cast<const Display *>(currentDisplay.get()))
|
|
{
|
|
if (const auto *textInterface = constCurrentDisplay->asTextInterface())
|
|
{
|
|
HtmlTag h2Tag{"h2", body};
|
|
body += htmlentities(textInterface->text());
|
|
}
|
|
|
|
if (const auto *menuDisplay = constCurrentDisplay->asMenuDisplay())
|
|
{
|
|
HtmlTag ulTag{"ul", body};
|
|
|
|
int i{0};
|
|
menuDisplay->runForEveryMenuItem([&,selectedIndex=menuDisplay->selectedIndex()](const MenuItem &menuItem){
|
|
HtmlTag liTag = i == selectedIndex ?
|
|
HtmlTag{"li", "style=\"border: 1px solid black;\"", body} :
|
|
HtmlTag{"li", body};
|
|
|
|
body += fmt::format("<a href=\"/triggerItem?index={}\">{}</a>", i, htmlentities(menuItem.text()));
|
|
i++;
|
|
});
|
|
}
|
|
else if (const auto *changeValueDisplay = constCurrentDisplay->asChangeValueDisplayInterface())
|
|
{
|
|
HtmlTag formTag{"form", "action=\"/setValue\" method=\"GET\"", body};
|
|
body += fmt::format("<input type=\"number\" name=\"value\" value=\"{}\" />", changeValueDisplay->shownValue());
|
|
body += "<button type=\"submit\">Update</button>";
|
|
}
|
|
else
|
|
{
|
|
body += "No web control implemented for current display.";
|
|
}
|
|
}
|
|
else
|
|
{
|
|
body += "Currently no screen instantiated.";
|
|
}
|
|
}
|
|
}
|
|
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_type, req, "text/html")
|
|
CALL_AND_EXIT(httpd_resp_send, req, body.data(), body.size())
|
|
}
|
|
|
|
esp_err_t webserver_up_handler(httpd_req_t *req)
|
|
{
|
|
InputDispatcher::rotate(-1);
|
|
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_type, req, "text/html")
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_status, req, "302 Moved Permanently")
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
|
constexpr const std::string_view body{"Ok, continue at <a href=\"/\">/</a>"};
|
|
CALL_AND_EXIT(httpd_resp_send, req, body.data(), body.size())
|
|
}
|
|
|
|
esp_err_t webserver_down_handler(httpd_req_t *req)
|
|
{
|
|
InputDispatcher::rotate(1);
|
|
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_type, req, "text/html")
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_status, req, "302 Moved Permanently")
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
|
constexpr const std::string_view body{"Ok, continue at <a href=\"/\">/</a>"};
|
|
CALL_AND_EXIT(httpd_resp_send, req, body.data(), body.size())
|
|
}
|
|
|
|
esp_err_t webserver_confirm_handler(httpd_req_t *req)
|
|
{
|
|
InputDispatcher::confirmButton(true);
|
|
InputDispatcher::confirmButton(false);
|
|
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_type, req, "text/html")
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_status, req, "302 Moved Permanently")
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
|
constexpr const std::string_view body{"Ok, continue at <a href=\"/\">/</a>"};
|
|
CALL_AND_EXIT(httpd_resp_send, req, body.data(), body.size())
|
|
}
|
|
|
|
esp_err_t webserver_back_handler(httpd_req_t *req)
|
|
{
|
|
InputDispatcher::backButton(true);
|
|
InputDispatcher::backButton(false);
|
|
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_type, req, "text/html")
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_status, req, "302 Moved Permanently")
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
|
constexpr const std::string_view body{"Ok, continue at <a href=\"/\">/</a>"};
|
|
CALL_AND_EXIT(httpd_resp_send, req, body.data(), body.size())
|
|
}
|
|
|
|
esp_err_t webserver_triggerItem_handler(httpd_req_t *req)
|
|
{
|
|
std::string query;
|
|
|
|
if (const size_t queryLength = httpd_req_get_url_query_len(req))
|
|
{
|
|
query.resize(queryLength);
|
|
CALL_AND_EXIT_ON_ERROR(httpd_req_get_url_query_str, req, query.data(), query.size() + 1)
|
|
}
|
|
|
|
std::size_t index;
|
|
|
|
constexpr const std::string_view indexParamName{"index"};
|
|
|
|
{
|
|
char valueBufEncoded[256];
|
|
if (const auto result = httpd_query_key_value(query.data(), indexParamName.data(), valueBufEncoded, 256); result == ESP_OK)
|
|
{
|
|
char valueBuf[257];
|
|
esphttpdutils::urldecode(valueBuf, valueBufEncoded);
|
|
|
|
std::string_view value{valueBuf};
|
|
|
|
if (auto parsed = cpputils::fromString<typeof(index)>(value))
|
|
{
|
|
index = *parsed;
|
|
}
|
|
else
|
|
{
|
|
CALL_AND_EXIT(httpd_resp_send_err, req, HTTPD_400_BAD_REQUEST, fmt::format("could not parse {} {}", indexParamName, value).c_str());
|
|
}
|
|
}
|
|
else if (result != ESP_ERR_NOT_FOUND)
|
|
{
|
|
ESP_LOGE(TAG, "httpd_query_key_value() %.*s failed with %s", indexParamName.size(), indexParamName.data(), esp_err_to_name(result));
|
|
return result;
|
|
}
|
|
else
|
|
{
|
|
CALL_AND_EXIT(httpd_resp_send_err, req, HTTPD_400_BAD_REQUEST, fmt::format("{} not set", indexParamName).c_str());
|
|
}
|
|
}
|
|
|
|
if (!currentDisplay)
|
|
CALL_AND_EXIT(httpd_resp_send_err, req, HTTPD_400_BAD_REQUEST, "currentDisplay is null");
|
|
|
|
auto *menuDisplay = currentDisplay->asMenuDisplay();
|
|
if (!menuDisplay)
|
|
CALL_AND_EXIT(httpd_resp_send_err, req, HTTPD_400_BAD_REQUEST, "currentDisplay is not a menu display");
|
|
|
|
if (index < 0 || index >= menuDisplay->menuItemCount())
|
|
CALL_AND_EXIT(httpd_resp_send_err, req, HTTPD_400_BAD_REQUEST, fmt::format("{} out of range", indexParamName).c_str());
|
|
|
|
menuDisplay->getMenuItem(index).triggered();
|
|
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_type, req, "text/html")
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_status, req, "302 Moved Permanently")
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
|
constexpr const std::string_view body{"Ok, continue at <a href=\"/\">/</a>"};
|
|
CALL_AND_EXIT(httpd_resp_send, req, body.data(), body.size())
|
|
}
|
|
|
|
esp_err_t webserver_setValue_handler(httpd_req_t *req)
|
|
{
|
|
std::string query;
|
|
|
|
if (const size_t queryLength = httpd_req_get_url_query_len(req))
|
|
{
|
|
query.resize(queryLength);
|
|
CALL_AND_EXIT_ON_ERROR(httpd_req_get_url_query_str, req, query.data(), query.size() + 1)
|
|
}
|
|
|
|
int newValue;
|
|
|
|
constexpr const std::string_view valueParamName{"value"};
|
|
|
|
{
|
|
char valueBufEncoded[256];
|
|
if (const auto result = httpd_query_key_value(query.data(), valueParamName.data(), valueBufEncoded, 256); result == ESP_OK)
|
|
{
|
|
char valueBuf[257];
|
|
esphttpdutils::urldecode(valueBuf, valueBufEncoded);
|
|
|
|
std::string_view value{valueBuf};
|
|
|
|
if (auto parsed = cpputils::fromString<typeof(newValue)>(value))
|
|
{
|
|
newValue = *parsed;
|
|
}
|
|
else
|
|
{
|
|
CALL_AND_EXIT(httpd_resp_send_err, req, HTTPD_400_BAD_REQUEST, fmt::format("could not parse {} {}", valueParamName, value).c_str());
|
|
}
|
|
}
|
|
else if (result != ESP_ERR_NOT_FOUND)
|
|
{
|
|
ESP_LOGE(TAG, "httpd_query_key_value() %.*s failed with %s", valueParamName.size(), valueParamName.data(), esp_err_to_name(result));
|
|
return result;
|
|
}
|
|
else
|
|
{
|
|
CALL_AND_EXIT(httpd_resp_send_err, req, HTTPD_400_BAD_REQUEST, fmt::format("{} not set", valueParamName).c_str());
|
|
}
|
|
}
|
|
|
|
if (!currentDisplay)
|
|
CALL_AND_EXIT(httpd_resp_send_err, req, HTTPD_400_BAD_REQUEST, "currentDisplay is null");
|
|
|
|
auto *changeValueDisplay = currentDisplay->asChangeValueDisplayInterface();
|
|
if (!changeValueDisplay)
|
|
CALL_AND_EXIT(httpd_resp_send_err, req, HTTPD_400_BAD_REQUEST, "currentDisplay is not a change value display");
|
|
|
|
changeValueDisplay->setShownValue(newValue);
|
|
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_type, req, "text/html")
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_status, req, "302 Moved Permanently")
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_hdr, req, "Location", "/")
|
|
constexpr const std::string_view body{"Ok, continue at <a href=\"/\">/</a>"};
|
|
CALL_AND_EXIT(httpd_resp_send, req, body.data(), body.size())
|
|
}
|
|
|
|
esp_err_t webserver_reboot_handler(httpd_req_t *req)
|
|
{
|
|
shouldReboot = true;
|
|
|
|
CALL_AND_EXIT_ON_ERROR(httpd_resp_set_type, req, "text/html")
|
|
std::string_view body{"REBOOT called..."};
|
|
CALL_AND_EXIT(httpd_resp_send, req, body.data(), body.size())
|
|
}
|
|
|
|
#endif
|
|
}
|