Simple ESP32 Web Server Login Page - ESP-IDF

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.

ESP-IDF ESP32 Web Server Login Page


To Create and Run Web Server we will use HTTP protocol. To understand what is HTTP protocol and what are its features please check my previous tutorial,

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).

ESP32-C3-DevKitC-02 is an entry-level development board based on ESP32-C3-WROOM-02 or ESP32-C3-WROOM-02U, general-purpose modules with 4 MB SPI flash. This board integrates complete Wi-Fi and Bluetooth® Low Energy functions.


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.
  1. 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 RED_LED_PIN                         2
    #define GREEN_LED_PIN                       3
    
    #define USERNAME                            "EmbeTronics"
    #define PASSWORD                            "123456789"
    
    
    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);
    
        /* Init LED GPIO Pins */
        gpio_config_t io_conf = {
            .pin_bit_mask = ((1ULL << RED_LED_PIN) | (1ULL << GREEN_LED_PIN)),      // Select Led GPIO Pins
            .mode = GPIO_MODE_OUTPUT,            // Set as output
            .pull_up_en = GPIO_PULLUP_DISABLE,  // Disable pull-up
            .pull_down_en = GPIO_PULLDOWN_DISABLE,  // Disable pull-down
            .intr_type = GPIO_INTR_DISABLE             // Disable interrupts
        };
    
        gpio_config(&io_conf);
    
        // 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);
    }
    
  2. 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>
  3. 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>
    
  4. 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>
    
  5. CMakeLists.txt 
    idf_component_register(SRCS main.c
                           EMBED_FILES root.html Resp_Correct.html Resp_Incorrect.html)
    


Working Of Code

    1. 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. 
    2. After modifications, build and flash code on board.
    3. As soon as program starts running on board, It will create Wi-Fi Access Point with below credentials.
      SSID : EmbeTronics
      PASS : 12345678
      
      
    4. 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.
    5. If Webpage didn't open automatically in Web Browser, Use IP Address "192.168.4.1" to open Web Page manually.
    6. On Webpage Enter Username and Password as provided below and Click on Login Button,

      USERNAME : EmbeTronics
      PASSWORD : 123456789
      

      ESP-IDF ESP32 HTTP Web Server Login Page



    7. If you have entered credentials as mentioned in step 6, you will response as below image,

      ESP-IDF ESP32 HTTP Web Server Login Page Correct Credentials


    8. If Credentials are Incorrect then, you will receive response as shown in below image,

      ESP-IDF ESP32 HTTP Web Server Login Page Wrong Credentials


    Full Project Code

    You can directly download and use whole project from below GitHub link.

    GitHub Repo Link


    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.

      Post a Comment

      0 Comments