In this tutorial I will show you how to create Web Server using ESP32 Module and ESP-IDF. You will create how to create a HTML Web Page for Login and Credentials check.
Prerequisite
Before proceeding further make sure you have installed VS Code with ESP-IDF extension. You can follow steps mentioned at this link to install and configure ESP-IDF in VS Code.
Hardware Used
In this tutorial we are using ESP32-C6 Dev Kit (You can use any ESP32 based board with this code).
Create Example Project ESP-IDF
Open VS Code and Press CTRL+SHIFT+P or Click on View > Command Palette to Open Command Palette option. Select ESP-IDF: Show Example Projects Option to view all available examples from ESP-IDF.
Select captive_portal under http_server section, then Select location at which you want to create example project.
After project is created you will see project folder open in VS Code.
ESP-IDF Web-Server Code
For this project you need to replace 3 files with provided code and add 2 new files from.
- main/main.c
/* Captive Portal Example This example code is in the Public Domain (or CC0 licensed, at your option.) Unless required by applicable law or agreed to in writing, this software is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. */ #include <sys/param.h> #include "esp_event.h" #include "esp_log.h" #include "esp_mac.h" #include "nvs_flash.h" #include "esp_wifi.h" #include "esp_netif.h" #include "lwip/inet.h" #include "driver/gpio.h" #include "esp_http_server.h" #include "dns_server.h" #define EXAMPLE_ESP_WIFI_SSID "EmbeTronics" #define EXAMPLE_ESP_WIFI_PASS "12345678" #define EXAMPLE_MAX_STA_CONN 5 #define USERNAME "EmbeTronics" #define PASSWORD "12345678" extern const char root_start[] asm("_binary_root_html_start"); extern const char root_end[] asm("_binary_root_html_end"); extern const char Resp_Correct_start[] asm("_binary_Resp_Correct_html_start"); extern const char Resp_Correct_end[] asm("_binary_Resp_Correct_html_end"); extern const char Resp_Incorrect_start[] asm("_binary_Resp_Incorrect_html_start"); extern const char Resp_Incorrect_end[] asm("_binary_Resp_Incorrect_html_end"); static const char *TAG = "example"; static void wifi_event_handler(void *arg, esp_event_base_t event_base, int32_t event_id, void *event_data) { if (event_id == WIFI_EVENT_AP_STACONNECTED) { wifi_event_ap_staconnected_t *event = (wifi_event_ap_staconnected_t *)event_data; ESP_LOGI(TAG, "station " MACSTR " join, AID=%d", MAC2STR(event->mac), event->aid); } else if (event_id == WIFI_EVENT_AP_STADISCONNECTED) { wifi_event_ap_stadisconnected_t *event = (wifi_event_ap_stadisconnected_t *)event_data; ESP_LOGI(TAG, "station " MACSTR " leave, AID=%d, reason=%d", MAC2STR(event->mac), event->aid, event->reason); } } static void wifi_init_softap(void) { wifi_init_config_t cfg = WIFI_INIT_CONFIG_DEFAULT(); ESP_ERROR_CHECK(esp_wifi_init(&cfg)); ESP_ERROR_CHECK(esp_event_handler_register(WIFI_EVENT, ESP_EVENT_ANY_ID, &wifi_event_handler, NULL)); wifi_config_t wifi_config = { .ap = { .ssid = EXAMPLE_ESP_WIFI_SSID, .ssid_len = strlen(EXAMPLE_ESP_WIFI_SSID), .password = EXAMPLE_ESP_WIFI_PASS, .max_connection = EXAMPLE_MAX_STA_CONN, .authmode = WIFI_AUTH_WPA_WPA2_PSK }, }; if (strlen(EXAMPLE_ESP_WIFI_PASS) == 0) { wifi_config.ap.authmode = WIFI_AUTH_OPEN; } ESP_ERROR_CHECK(esp_wifi_set_mode(WIFI_MODE_AP)); ESP_ERROR_CHECK(esp_wifi_set_config(ESP_IF_WIFI_AP, &wifi_config)); ESP_ERROR_CHECK(esp_wifi_start()); esp_netif_ip_info_t ip_info; esp_netif_get_ip_info(esp_netif_get_handle_from_ifkey("WIFI_AP_DEF"), &ip_info); char ip_addr[16]; inet_ntoa_r(ip_info.ip.addr, ip_addr, 16); ESP_LOGI(TAG, "Set up softAP with IP: %s", ip_addr); ESP_LOGI(TAG, "wifi_init_softap finished. SSID:'%s' password:'%s'", EXAMPLE_ESP_WIFI_SSID, EXAMPLE_ESP_WIFI_PASS); } #ifdef CONFIG_ESP_ENABLE_DHCP_CAPTIVEPORTAL static void dhcp_set_captiveportal_url(void) { // get the IP of the access point to redirect to esp_netif_ip_info_t ip_info; esp_netif_get_ip_info(esp_netif_get_handle_from_ifkey("WIFI_AP_DEF"), &ip_info); char ip_addr[16]; inet_ntoa_r(ip_info.ip.addr, ip_addr, 16); ESP_LOGI(TAG, "Set up softAP with IP: %s", ip_addr); // turn the IP into a URI char* captiveportal_uri = (char*) malloc(32 * sizeof(char)); assert(captiveportal_uri && "Failed to allocate captiveportal_uri"); strcpy(captiveportal_uri, "http://"); strcat(captiveportal_uri, ip_addr); // get a handle to configure DHCP with esp_netif_t* netif = esp_netif_get_handle_from_ifkey("WIFI_AP_DEF"); // set the DHCP option 114 ESP_ERROR_CHECK_WITHOUT_ABORT(esp_netif_dhcps_stop(netif)); ESP_ERROR_CHECK(esp_netif_dhcps_option(netif, ESP_NETIF_OP_SET, ESP_NETIF_CAPTIVEPORTAL_URI, captiveportal_uri, strlen(captiveportal_uri))); ESP_ERROR_CHECK_WITHOUT_ABORT(esp_netif_dhcps_start(netif)); } #endif // CONFIG_ESP_ENABLE_DHCP_CAPTIVEPORTAL /** * @brief Decodes a URL-encoded string. * * This function decodes a URL-encoded string, handling both percent-encoded * characters (e.g., %20 for space) and the '+' character (which also * represents a space). The decoded string is written to a destination buffer. * * @param dst Pointer to the destination buffer where the decoded string will be stored. * @param src Pointer to the source URL-encoded string. */ void urldecode(char *dst, const char *src) { char a, b; while (*src) { if ((*src == '%') && ((a = src[1]) && (b = src[2])) && (isxdigit(a) && isxdigit(b))) { // Convert hex characters to their integer values if (a >= 'a') a -= 'a' - 'A'; if (a >= 'A') a -= ('A' - 10); else a -= '0'; if (b >= 'a') b -= 'a' - 'A'; if (b >= 'A') b -= ('A' - 10); else b -= '0'; // Store the decoded character *dst++ = 16 * a + b; src += 3; // Move past the %XX sequence } else if (*src == '+') { *dst++ = ' '; // Replace '+' with a space src++; } else { *dst++ = *src++; // Copy other characters directly } } *dst = '\0'; // Null-terminate the destination string } /* HTTP GET Handlers */ static esp_err_t root_get_handler(httpd_req_t *req) { const uint32_t root_len = root_end - root_start; ESP_LOGI(TAG, "Serve root, Req %s",req->uri); httpd_resp_set_type(req, "text/html"); httpd_resp_send(req, root_start, root_len); return ESP_OK; } // HTTP GET Handler static esp_err_t http_submit_handler(httpd_req_t *req) { ESP_LOGI(TAG, "Serve root, uri:%s",req->uri); bool Check_UserName = false; bool Check_Password = false; const uint32_t Resp_Correct_len = Resp_Correct_end - Resp_Correct_start; const uint32_t Resp_Incorrect_len = Resp_Incorrect_end - Resp_Incorrect_start; char uri[HTTPD_MAX_URI_LEN + 1] = ""; /* Decode url, to replace special chars with original values */ urldecode(uri,req->uri); ESP_LOGI(TAG,"Decoded string : %s",uri); /* Tokenize to seperate each values */ char *myPtr = strtok(uri, "&"); while (myPtr != NULL) { ESP_LOGI(TAG,"%s", myPtr); if(strstr(myPtr,"username") != NULL) { ESP_LOGI(TAG,"username : %s",strstr(myPtr, "=")+1); if(strcmp(strstr(myPtr, "=")+1, USERNAME) == 0) { Check_UserName = true; } } else if(strstr(myPtr,"password") != NULL) { ESP_LOGI(TAG,"password : %s",strstr(myPtr, "=")+1); if(strcmp(strstr(myPtr, "=")+1, PASSWORD) == 0) { Check_Password = true; } } myPtr = strtok(NULL, "&"); } httpd_resp_set_type(req, "text/html"); if((Check_UserName == true) && (Check_Password == true)) { ESP_LOGI(TAG,"Credentials Matched"); httpd_resp_send(req, Resp_Correct_start, Resp_Correct_len); } else { ESP_LOGI(TAG,"Credentials Not Matched"); httpd_resp_send(req, Resp_Incorrect_start, Resp_Incorrect_len); } return ESP_OK; } static const httpd_uri_t root = { .uri = "/", .method = HTTP_GET, .handler = root_get_handler }; static const httpd_uri_t submit_handler = { .uri = "/submit", .method = HTTP_GET, .handler = http_submit_handler }; // HTTP Error (404) Handler - Redirects all requests to the root page esp_err_t http_404_error_handler(httpd_req_t *req, httpd_err_code_t err) { // Set status httpd_resp_set_status(req, "302 Temporary Redirect"); // Redirect to the "/" root directory httpd_resp_set_hdr(req, "Location", "/"); // iOS requires content in the response to detect a captive portal, simply redirecting is not sufficient. httpd_resp_send(req, "Redirect to the captive portal", HTTPD_RESP_USE_STRLEN); ESP_LOGI(TAG, "Redirecting to root"); return ESP_OK; } static httpd_handle_t start_webserver(void) { httpd_handle_t server = NULL; httpd_config_t config = HTTPD_DEFAULT_CONFIG(); config.max_open_sockets = 13; config.lru_purge_enable = true; // Start the httpd server ESP_LOGI(TAG, "Starting server on port: '%d'", config.server_port); if (httpd_start(&server, &config) == ESP_OK) { // Set URI handlers ESP_LOGI(TAG, "Registering URI handlers"); httpd_register_uri_handler(server, &root); httpd_register_uri_handler(server, &submit_handler); httpd_register_err_handler(server, HTTPD_404_NOT_FOUND, http_404_error_handler); } return server; } void app_main(void) { /* Turn of warnings from HTTP server as redirecting traffic will yield lots of invalid requests */ esp_log_level_set("httpd_uri", ESP_LOG_ERROR); esp_log_level_set("httpd_txrx", ESP_LOG_ERROR); esp_log_level_set("httpd_parse", ESP_LOG_ERROR); // Initialize networking stack ESP_ERROR_CHECK(esp_netif_init()); // Create default event loop needed by the main app ESP_ERROR_CHECK(esp_event_loop_create_default()); // Initialize NVS needed by Wi-Fi ESP_ERROR_CHECK(nvs_flash_init()); // Initialize Wi-Fi including netif with default config esp_netif_create_default_wifi_ap(); // Initialise ESP32 in SoftAP mode wifi_init_softap(); // Configure DNS-based captive portal, if configured #ifdef CONFIG_ESP_ENABLE_DHCP_CAPTIVEPORTAL dhcp_set_captiveportal_url(); #endif // Start the server for the first time start_webserver(); // Start the DNS server that will redirect all queries to the softAP IP dns_server_config_t config = DNS_SERVER_CONFIG_SINGLE("*" /* all A queries */, "WIFI_AP_DEF" /* softAP netif ID */); start_dns_server(&config); } - main/root.html
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>EmbeTronics Login Page</title> <style> body { font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, 'Helvetica Neue', Arial, sans-serif; background: #f8fafc; min-height: 100vh; display: flex; align-items: center; justify-content: center; padding: 20px; line-height: 1.6; color: #334155; } /* .container { background-color: #2e5077; padding: 50px; border-radius: 8px; box-shadow: 0 2px 10px rgba(0, 0, 0, 0.1); text-align: center; width: 100%; max-width: 400px; } */ .login-card { background: #2e5077; border-radius: 12px; padding: 32px; border: 1px solid #ffffff; box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.1); } .login-header { text-align: center; margin-bottom: 32px; } /* Form Styles */ .form-group { margin-bottom: 20px; } .input-wrapper { position: relative; display: flex; flex-direction: column; } .input-wrapper input { background: white; border: 2px solid #e2e8f0; border-radius: 8px; padding: 12px 16px 8px 16px; color: #1e293b; font-size: 16px; transition: all 0.2s ease; /* width: %; */ outline: none; } .input-wrapper select { background: white; border: 2px solid #e2e8f0; border-radius: 8px; padding: 12px 16px 8px 16px; color: #1e293b; font-size: 16px; transition: all 0.2s ease; width: 100%; outline: none; } .input-wrapper label { position: absolute; left: 16px; top: 12px; color: #ffffff; font-size: 16px; transition: all 0.2s ease; pointer-events: none; transform-origin: left top; } /* Button */ .login-btn { width: 100%; background: #FE7032; border: none; border-radius: 8px; padding: 12px 24px; color: white; font-size: 16px; font-weight: 600; cursor: pointer; transition: all 0.2s ease; position: relative; margin-bottom: 24px; } .login-btn:hover { background: #4f46e5; } .login-btn:active { transform: translateY(1px); } .btn-text { transition: opacity 0.2s ease; } label { color: #ffffff; font-weight: bold; } a { color: #ffffff; font-weight: bold; } h1 { color: #ffffff; } h2 { color: #ffffff; } </style> </head> <body> <form action="/submit"> <div class="container"> <div class="login-card"> <div class="login-header"> <img alt="Embedded Image" src=""> </div> <form class="login-form" id="loginForm" novalidate> <div class="form-group"> <label>Username</label> <div class="input-wrapper text"> <input type="text" id="username" minlength="4" name="username" value="" required> </div> </div> <div class="form-group"> <label>Password</label> <div class="input-wrapper password-wrapper"> <input type="password" id="password" minlength="4" name="password" value="" required> </div> </div> <button type="submit" class="login-btn"> <span class="btn-text">Submit</span> <span class="btn-loader"></span> </button> </form> </div> </div> </form> </body> </html> - main/Resp_Correct.html
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>EmbeTronics Login Page</title> <style> body { font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, 'Helvetica Neue', Arial, sans-serif; background: #f8fafc; min-height: 100vh; display: flex; align-items: center; justify-content: center; padding: 20px; line-height: 1.6; color: #334155; } .login-card { background: #2e5077; border-radius: 12px; padding: 32px; border: 1px solid #ffffff; box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.1); } .login-header { text-align: center; margin-bottom: 32px; } a { color: #ffffff; font-weight: bold; font-size: 20px } h1 { color: green; } </style> </head> <body> <form action="/submit"> <div class="container"> <div class="login-card"> <div class="login-header"> <img alt="Embedded Image" src=""> <h1>The credentials that you've entered are Correct.</h1> <a href="https://www.embetronics.com/">Visit embetronics.com for More information !!</a> </div> </div> </div> </form> </body> </html> - main/Resp_Incorrect.html
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>EmbeTronics Login Page</title> <style> body { font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, 'Helvetica Neue', Arial, sans-serif; background: #f8fafc; min-height: 100vh; display: flex; align-items: center; justify-content: center; padding: 20px; line-height: 1.6; color: #334155; } .login-card { background: #2e5077; border-radius: 12px; padding: 32px; border: 1px solid #ffffff; box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.1); } .login-header { text-align: center; margin-bottom: 32px; } a { color: #ffffff; font-weight: bold; font-size: 20px; } h1 { color: red; } </style> </head> <body> <form action="/submit"> <div class="container"> <div class="login-card"> <div class="login-header"> <img alt="Embedded Image" src=""> <h1>The credentials that you've entered are incorrect.</h1> <a href="https://www.embetronics.com/">Visit embetronics.com for More information !!</a> </div> </div> </div> </form> </body> </html> - CMakeLists.txt
idf_component_register(SRCS main.c EMBED_FILES root.html Resp_Correct.html Resp_Incorrect.html)
Working Of Code
- Do changes as mentioned below on created captive_portal sample code.
- Replace content of files main.c, root.html, CMakeLists.txt with provided data above
- Add new files Resp_Correct.html and Resp_Incorrect.html with content as provided above.
- After modifications, build and flash code on board.
- As soon as program starts running on board, It will create Wi-Fi Access Point with below credentials.
SSID : EmbeTronics PASS : 12345678 - From Laptop or Mobile, Connect with above Access Point using provided credentials. As soon as device is connected with access point of ESP32, It will automatically open webpage in browser.
- If Webpage didn't open automatically in Web Browser, Use IP Address "192.168.4.1" to open Web Page manually.
- On Webpage Enter Username and Password as provided below and Click on Login Button,
USERNAME : EmbeTronics PASSWORD : 123456789 - If you have entered credentials as mentioned in step 6, you will response as below image,
- If Credentials are Incorrect then, you will receive response as shown in below image,
Full Project Code
You can directly download and use whole project from below GitHub link.
GitHub Repo Link
Explanation of Code
Let us understand blocks of code and see how it works.
First we will include all required header files. Header files for FreeRTOS, WiFi, MQTT, GPIO and Logging are included.
#include <sys/param.h>
#include "esp_event.h"
#include "esp_log.h"
#include "esp_mac.h"
#include "nvs_flash.h"
#include "esp_wifi.h"
#include "esp_netif.h"
#include "lwip/inet.h"
#include "driver/gpio.h"
#include "esp_http_server.h"
#include "dns_server.h"
Then Wi-Fi Credentials to used with Access Point are set.
#define EXAMPLE_ESP_WIFI_SSID "EmbeTronics"
#define EXAMPLE_ESP_WIFI_PASS "12345678"
#define EXAMPLE_MAX_STA_CONN 5Then Credentials for HTTP Login Page are set,
#define USERNAME "EmbeTronics"
#define PASSWORD "12345678"HTML files for Web Pages are embedded into project. It is specified using EMBED_FILES into CMakeLists.txt file.
idf_component_register(SRCS main.c
EMBED_FILES root.html Resp_Correct.html Resp_Incorrect.html)HTML file contents are accessed from code via below variables. All 3 Web Page variables are declared.
extern const char root_start[] asm("_binary_root_html_start");
extern const char root_end[] asm("_binary_root_html_end");
extern const char Resp_Correct_start[] asm("_binary_Resp_Correct_html_start");
extern const char Resp_Correct_end[] asm("_binary_Resp_Correct_html_end");
extern const char Resp_Incorrect_start[] asm("_binary_Resp_Incorrect_html_start");
extern const char Resp_Incorrect_end[] asm("_binary_Resp_Incorrect_html_end")
TAG to be used when ESP Log print is defined. You can modify as per your requirement.
extern const char root_start[] asm("_binary_root_html_start");
/* TAG for ESP Log Print */ static const char *TAG = "example";
wifi_event_handler handler function to handle Wi-Fi related events is defined. It handles Wi-Fi Connect, and Disconnect Got events. It will try to connect with provided Wi-Fi network with defined number of retries.
static void wifi_event_handler(void *arg, esp_event_base_t event_base,
int32_t event_id, void *event_data)
{
if (event_id == WIFI_EVENT_AP_STACONNECTED) {
wifi_event_ap_staconnected_t *event = (wifi_event_ap_staconnected_t *)event_data;
ESP_LOGI(TAG, "station " MACSTR " join, AID=%d",
MAC2STR(event->mac), event->aid);
} else if (event_id == WIFI_EVENT_AP_STADISCONNECTED) {
wifi_event_ap_stadisconnected_t *event = (wifi_event_ap_stadisconnected_t *)event_data;
ESP_LOGI(TAG, "station " MACSTR " leave, AID=%d, reason=%d",
MAC2STR(event->mac), event->aid, event->reason);
}
}wifi_init_softap function initializes Wi-Fi in Soft Access Point Mode. It uses set Wi-Fi SSID and Password.
static void wifi_init_softap(void)
{
wifi_init_config_t cfg = WIFI_INIT_CONFIG_DEFAULT();
ESP_ERROR_CHECK(esp_wifi_init(&cfg));
ESP_ERROR_CHECK(esp_event_handler_register(WIFI_EVENT, ESP_EVENT_ANY_ID, &wifi_event_handler, NULL));
wifi_config_t wifi_config = {
.ap = {
.ssid = EXAMPLE_ESP_WIFI_SSID,
.ssid_len = strlen(EXAMPLE_ESP_WIFI_SSID),
.password = EXAMPLE_ESP_WIFI_PASS,
.max_connection = EXAMPLE_MAX_STA_CONN,
.authmode = WIFI_AUTH_WPA_WPA2_PSK
},
};
if (strlen(EXAMPLE_ESP_WIFI_PASS) == 0) {
wifi_config.ap.authmode = WIFI_AUTH_OPEN;
}
ESP_ERROR_CHECK(esp_wifi_set_mode(WIFI_MODE_AP));
ESP_ERROR_CHECK(esp_wifi_set_config(ESP_IF_WIFI_AP, &wifi_config));
ESP_ERROR_CHECK(esp_wifi_start());
esp_netif_ip_info_t ip_info;
esp_netif_get_ip_info(esp_netif_get_handle_from_ifkey("WIFI_AP_DEF"), &ip_info);
char ip_addr[16];
inet_ntoa_r(ip_info.ip.addr, ip_addr, 16);
ESP_LOGI(TAG, "Set up softAP with IP: %s", ip_addr);
ESP_LOGI(TAG, "wifi_init_softap finished. SSID:'%s' password:'%s'",
EXAMPLE_ESP_WIFI_SSID, EXAMPLE_ESP_WIFI_PASS);
}
dhcp_set_captiveportal_url function enables DHCP server on Access Point.
static void dhcp_set_captiveportal_url(void) {
// get the IP of the access point to redirect to
esp_netif_ip_info_t ip_info;
esp_netif_get_ip_info(esp_netif_get_handle_from_ifkey("WIFI_AP_DEF"), &ip_info);
char ip_addr[16];
inet_ntoa_r(ip_info.ip.addr, ip_addr, 16);
ESP_LOGI(TAG, "Set up softAP with IP: %s", ip_addr);
// turn the IP into a URI
char* captiveportal_uri = (char*) malloc(32 * sizeof(char));
assert(captiveportal_uri && "Failed to allocate captiveportal_uri");
strcpy(captiveportal_uri, "http://");
strcat(captiveportal_uri, ip_addr);
// get a handle to configure DHCP with
esp_netif_t* netif = esp_netif_get_handle_from_ifkey("WIFI_AP_DEF");
// set the DHCP option 114
ESP_ERROR_CHECK_WITHOUT_ABORT(esp_netif_dhcps_stop(netif));
ESP_ERROR_CHECK(esp_netif_dhcps_option(netif, ESP_NETIF_OP_SET, ESP_NETIF_CAPTIVEPORTAL_URI, captiveportal_uri, strlen(captiveportal_uri)));
ESP_ERROR_CHECK_WITHOUT_ABORT(esp_netif_dhcps_start(netif));
}
urldecode function is used to convert URL encoded string received from HTTP Web Server into normal string.
/**
* @brief Decodes a URL-encoded string.
*
* This function decodes a URL-encoded string, handling both percent-encoded
* characters (e.g., %20 for space) and the '+' character (which also
* represents a space). The decoded string is written to a destination buffer.
*
* @param dst Pointer to the destination buffer where the decoded string will be stored.
* @param src Pointer to the source URL-encoded string.
*/
void urldecode(char *dst, const char *src) {
char a, b;
while (*src) {
if ((*src == '%') && ((a = src[1]) && (b = src[2])) && (isxdigit(a) && isxdigit(b))) {
// Convert hex characters to their integer values
if (a >= 'a') a -= 'a' - 'A';
if (a >= 'A') a -= ('A' - 10);
else a -= '0';
if (b >= 'a') b -= 'a' - 'A';
if (b >= 'A') b -= ('A' - 10);
else b -= '0';
// Store the decoded character
*dst++ = 16 * a + b;
src += 3; // Move past the %XX sequence
} else if (*src == '+') {
*dst++ = ' '; // Replace '+' with a space
src++;
} else {
*dst++ = *src++; // Copy other characters directly
}
}
*dst = '\0'; // Null-terminate the destination string
}
Then, HTTP root request handler function is defined as below. It will be called by HTTP library when root request is received from HTTP Web Server.
/* HTTP GET Handlers */
static esp_err_t root_get_handler(httpd_req_t *req)
{
const uint32_t root_len = root_end - root_start;
ESP_LOGI(TAG, "Serve root, Req %s",req->uri);
httpd_resp_set_type(req, "text/html");
httpd_resp_send(req, root_start, root_len);
return ESP_OK;
}
Then, HTTP submit handler function is defined as below. It will be called by HTTP library when submit request is received from HTTP Web Server. It checks HTTP login credentials received from HTTP request and check for correctness. Based on comparison result, It send response HTTP Web Page.
// HTTP GET Handler
static esp_err_t http_submit_handler(httpd_req_t *req)
{
ESP_LOGI(TAG, "Serve root, uri:%s",req->uri);
bool Check_UserName = false;
bool Check_Password = false;
const uint32_t Resp_Correct_len = Resp_Correct_end - Resp_Correct_start;
const uint32_t Resp_Incorrect_len = Resp_Incorrect_end - Resp_Incorrect_start;
char uri[HTTPD_MAX_URI_LEN + 1] = "";
/* Decode url, to replace special chars with original values */
urldecode(uri,req->uri);
ESP_LOGI(TAG,"Decoded string : %s",uri);
/* Tokenize to seperate each values */
char *myPtr = strtok(uri, "&");
while (myPtr != NULL)
{
ESP_LOGI(TAG,"%s", myPtr);
if(strstr(myPtr,"username") != NULL)
{
ESP_LOGI(TAG,"username : %s",strstr(myPtr, "=")+1);
if(strcmp(strstr(myPtr, "=")+1, USERNAME) == 0)
{
Check_UserName = true;
}
}
else if(strstr(myPtr,"password") != NULL)
{
ESP_LOGI(TAG,"password : %s",strstr(myPtr, "=")+1);
if(strcmp(strstr(myPtr, "=")+1, PASSWORD) == 0)
{
Check_Password = true;
}
}
myPtr = strtok(NULL, "&");
}
httpd_resp_set_type(req, "text/html");
if((Check_UserName == true) && (Check_Password == true))
{
ESP_LOGI(TAG,"Credentials Matched");
httpd_resp_send(req, Resp_Correct_start, Resp_Correct_len);
}
else
{
ESP_LOGI(TAG,"Credentials Not Matched");
httpd_resp_send(req, Resp_Incorrect_start, Resp_Incorrect_len);
}
return ESP_OK;
}
Structures as shown below are used to register root and submit handler functions is defined.
static const httpd_uri_t root = {
.uri = "/",
.method = HTTP_GET,
.handler = root_get_handler
};
static const httpd_uri_t submit_handler = {
.uri = "/submit",
.method = HTTP_GET,
.handler = http_submit_handler
};
http_404_error_handler handler function is used to handle HTTP 404 Error.
// HTTP Error (404) Handler - Redirects all requests to the root page
esp_err_t http_404_error_handler(httpd_req_t *req, httpd_err_code_t err)
{
// Set status
httpd_resp_set_status(req, "302 Temporary Redirect");
// Redirect to the "/" root directory
httpd_resp_set_hdr(req, "Location", "/");
// iOS requires content in the response to detect a captive portal, simply redirecting is not sufficient.
httpd_resp_send(req, "Redirect to the captive portal", HTTPD_RESP_USE_STRLEN);
ESP_LOGI(TAG, "Redirecting to root");
return ESP_OK;
}start_webserver function initializes and starts HTTP Web Server, and registers all handlers functions (root, led on and off requests) with it.
static httpd_handle_t start_webserver(void)
{
httpd_handle_t server = NULL;
httpd_config_t config = HTTPD_DEFAULT_CONFIG();
config.max_open_sockets = 13;
config.lru_purge_enable = true;
// Start the httpd server
ESP_LOGI(TAG, "Starting server on port: '%d'", config.server_port);
if (httpd_start(&server, &config) == ESP_OK) {
// Set URI handlers
ESP_LOGI(TAG, "Registering URI handlers");
httpd_register_uri_handler(server, &root);
httpd_register_uri_handler(server, &submit_handler);
httpd_register_err_handler(server, HTTPD_404_NOT_FOUND, http_404_error_handler);
}
return server;
}At last app_main function of our code is defined. It performs below mentioned things,
- Set Log level for HTTP modules
- Initializes network stack with Soft-AP Wi-Fi Access Point
- Initializes NVS flash memory
- Initializes DHCP server for Captive Portal
- Start Web Server over HTTP
- Start DNS Server
void app_main(void)
{
/*
Turn of warnings from HTTP server as redirecting traffic will yield
lots of invalid requests
*/
esp_log_level_set("httpd_uri", ESP_LOG_ERROR);
esp_log_level_set("httpd_txrx", ESP_LOG_ERROR);
esp_log_level_set("httpd_parse", ESP_LOG_ERROR);
// Initialize networking stack
ESP_ERROR_CHECK(esp_netif_init());
// Create default event loop needed by the main app
ESP_ERROR_CHECK(esp_event_loop_create_default());
// Initialize NVS needed by Wi-Fi
ESP_ERROR_CHECK(nvs_flash_init());
// Initialize Wi-Fi including netif with default config
esp_netif_create_default_wifi_ap();
// Initialise ESP32 in SoftAP mode
wifi_init_softap();
// Configure DNS-based captive portal, if configured
#ifdef CONFIG_ESP_ENABLE_DHCP_CAPTIVEPORTAL
dhcp_set_captiveportal_url();
#endif
// Start the server for the first time
start_webserver();
// Start the DNS server that will redirect all queries to the softAP IP
dns_server_config_t config = DNS_SERVER_CONFIG_SINGLE("*" /* all A queries */, "WIFI_AP_DEF" /* softAP netif ID */);
start_dns_server(&config);
}Wrapping Up
In todays, Tutorial you have learned how to control leds from web browser using ESP32 Board.
Please leave a comment if you have a question or you found this helpful.




0 Comments