ESP32系列四:搭建http的webserver的服务器

最近在使用ESP32搭建web服务器测试,发现esp32搭建这类开发环境还是比较方便的。具体的http协议这里就不再赘述,我们主要说一下如何使用ESP32提供的API来搭建我们的http web。

一、web服务器搭建过程

1、配置web服务器

在ESP-IDF中,Web服务器使用httpd组件实现。我们需要先创建httpd_config_t结构体,指定服务器的端口、最大并发连接数、URI匹配处理器等选项。然后,我们通过调用httpd_start函数来启动Web服务器。(使用默认的配置就可以,包括端口号都已经默认配置好了)

httpd_config_t config = HTTPD_DEFAULT_CONFIG();
httpd_handle_t server = NULL;// 设置服务器端口为80
config.server_port = 80;// 创建HTTP服务器句柄
if (httpd_start(&server, &config) != ESP_OK) {printf("Error starting server!\n");return;
}

2、 注册 URI处理器
在Web服务器启动后,我们需要为不同的URI注册处理器函数。当Web服务器接收到请求时,会根据请求的URI选择相应的处理器函数进行处理。在ESP-IDF中,我们可以使用httpd_register_uri_handler函数注册URI处理器。该函数的原型如下

esp_err_t httpd_register_uri_handler(httpd_handle_t hd, const httpd_uri_t *uri)

其中,hd参数为HTTP服务器句柄;uri参数为包含URI路径、HTTP方法、处理函数等信息的结构体指针。例如:

static const httpd_uri_t echo = {.uri       = "/",.method    = HTTP_POST,.handler   = echo_post_handler,.user_ctx  = NULL
};
static esp_err_t echo_post_handler(httpd_req_t *req)
{char buf[100];// char ssid[10];// char pswd[10];int ret, remaining = req->content_len;while (remaining > 0) {/* Read the data for the request */if ((ret = httpd_req_recv(req, buf,MIN(remaining, sizeof(buf)))) <= 0) {if (ret == HTTPD_SOCK_ERR_TIMEOUT) {/* Retry receiving if timeout occurred */continue;}return ESP_FAIL;}/* Send back the same data */httpd_resp_send_chunk(req, buf, ret);remaining -= ret;esp_err_t e = httpd_query_key_value(buf,"ssid",wifi_name,sizeof(wifi_name));if(e == ESP_OK) {printf("ssid = %s\r\n",wifi_name);}else {printf("error = %d\r\n",e);}e = httpd_query_key_value(buf,"password",wifi_password,sizeof(wifi_password));if(e == ESP_OK) {printf("pswd = %s\r\n",wifi_password);}else {printf("error = %d\r\n",e);}/* Log data received */ESP_LOGI(TAG, "=========== RECEIVED DATA ==========");ESP_LOGI(TAG, "%.*s", ret, buf);ESP_LOGI(TAG, "====================================");}// End responsehttpd_resp_send_chunk(req, NULL, 0);if(strcmp(wifi_name ,"\0")!=0 && strcmp(wifi_password,"\0")!=0){xSemaphoreGive(ap_sem);ESP_LOGI(TAG, "set wifi name and password successfully! goto station mode");}return ESP_OK;
}

html的网页如下:这个网页包含了按钮的定义,以及发送json格式的数据。

最后送json数据的时候,要用JSON.stringify方法格式化data,否则esp32解析json会报错,此处一定要注意!!!
整体html界面非常简单,没有基础的也很容易读懂,里面写了一个js函数,该函数是在点击按钮的时候触发,功能主要是读取文本框输入的数据,将数据封装为json格式,然后post发送数据,xhttp.open(“POST”, “/wifi_data”, true);中的url “/wifi_data”和esp32服务端中的定义要一致,否则是无法成功的。

<!DOCTYPE html>
<head><meta charset="utf-8"><title>Web server system</title>
</head>
<table class="fixed" border="5"><col width="1000px" /><col width="500px" /><tr><td><h2 style=" text-align:center;"> **** Web Server ***</h2><h3>wifi 密码配置</h3><div><label for="name">wifi名称</label><input type="text" id="wifi" name="car_name" placeholder="ssid"><br><label for="type">密码</label><input type="text" id="code" name="car_type" placeholder="password"><br><button id ="send_WIFI" type="button" onclick="send_wifi()">提交</button></div></td><td><table border="10"><tr><td><label for="newfile">Upload a file</label></td><td colspan="2"><input id="newfile" type="file" onchange="setpath()" style="width:100%;"></td></tr><tr><td><label for="filepath">Set path on server</label></td><td><input id="filepath" type="text" style="width:100%;"></td><td><button id="upload" type="button" onclick="upload()">Upload</button></td></tr></table></td></tr>
</table>
<script>
function setpath() {var default_path = document.getElementById("newfile").files[0].name;document.getElementById("filepath").value = default_path;
}
function upload() {var filePath = document.getElementById("filepath").value;var upload_path = "/upload/" + filePath;var fileInput = document.getElementById("newfile").files;/* Max size of an individual file. Make sure this* value is same as that set in file_server.c */var MAX_FILE_SIZE = 200*1024;var MAX_FILE_SIZE_STR = "200KB";if (fileInput.length == 0) {alert("No file selected!");} else if (filePath.length == 0) {alert("File path on server is not set!");} else if (filePath.indexOf(' ') >= 0) {alert("File path on server cannot have spaces!");} else if (filePath[filePath.length-1] == '/') {alert("File name not specified after path!");} else if (fileInput[0].size > 200*1024) {alert("File size must be less than 200KB!");} else {document.getElementById("newfile").disabled = true;document.getElementById("filepath").disabled = true;document.getElementById("upload").disabled = true;var file = fileInput[0];var xhttp = new XMLHttpRequest();xhttp.onreadystatechange = function() {if (xhttp.readyState == 4) {if (xhttp.status == 200) {document.open();document.write(xhttp.responseText);document.close();} else if (xhttp.status == 0) {alert("Server closed the connection abruptly!");location.reload()} else {alert(xhttp.status + " Error!\n" + xhttp.responseText);location.reload()}}};xhttp.open("POST", upload_path, true);xhttp.send(file);}
}function send_wifi() {var input_ssid = document.getElementById("wifi").value;var input_code = document.getElementById("code").value;var xhttp = new XMLHttpRequest();xhttp.open("POST", "/wifi_data", true);xhttp.onreadystatechange = function() {if (xhttp.readyState == 4) {if (xhttp.status == 200) {console.log(xhttp.responseText);} else if (xhttp.status == 0) {alert("Server closed the connection abruptly!");location.reload()} else {alert(xhttp.status + " Error!\n" + xhttp.responseText);location.reload()}}};var data = {"wifi_name":input_ssid,"wifi_code":input_code}xhttp.send(JSON.stringify(data));
}</script>

static esp_err_t html_default_get_handler(httpd_req_t *req)
{// /* Send HTML file header */// httpd_resp_sendstr_chunk(req, "<!DOCTYPE html><html><body>");/* Get handle to embedded file upload script */extern const unsigned char upload_script_start[] asm("_binary_upload_script_html_start");extern const unsigned char upload_script_end[] asm("_binary_upload_script_html_end");const size_t upload_script_size = (upload_script_end - upload_script_start);/* Add file upload form and script which on execution sends a POST request to /upload */httpd_resp_send_chunk(req, (const char *)upload_script_start, upload_script_size);/* Send remaining chunk of HTML file to complete it */httpd_resp_sendstr_chunk(req, "</body></html>");/* Send empty chunk to signal HTTP response completion */httpd_resp_sendstr_chunk(req, NULL);return ESP_OK;
}httpd_uri_t html_default = {.uri = "/", // Match all URIs of type /path/to/file.method = HTTP_GET,.handler = html_default_get_handler,.user_ctx = "html" // Pass server data as context};httpd_register_uri_handler(server, &html_default);

这里要特别注意:

1)、.uri = "/",这个表示当你打开网页的IP地址后第一个要显示的页面,也就是要放到根目录下("/"也就是根目录)。

2)、ESP32可以直接将html的网页编译进来不需要转为数组,但在CMakeList.txt文件中需要EMBED_FILES upload_script.html样的形式引入网页文件。这个html编译出出来的文本文件怎么使用呢。wifi.html编译出来一般名称是默认的_binary_名称_类型_start。这个指针代编译出来文件的起始地址。_binary_名称_类型_end,代表结束地址。wifi.html的引用方式如下。通过以下的方式就可以获得html这个大数组。

    extern const unsigned char upload_script_start[] asm("_binary_upload_script_html_start");extern const unsigned char upload_script_end[] asm("_binary_upload_script_html_end");const size_t upload_script_size = (upload_script_end - upload_script_start);/* Add file upload form and script which on execution sends a POST request to /upload */httpd_resp_send_chunk(req, (const char *)upload_script_start, upload_script_size);

3、实现URI处理函数

在注册URI处理器后,我们需要实现对应的处理器函数。URI处理器函数的原型为:

typedef esp_err_t (*httpd_uri_func_t)(httpd_req_t *req);

如上面示例中的:

static esp_err_t html_default_get_handler(httpd_req_t *req)

4、处理HTTP请求
在URI处理器函数中,我们可以通过HTTP请求信息结构体指针httpd_req_t获取HTTP请求的各种参数和数据。以下是一些常用的HTTP请求处理函数:

httpd_req_get_hdr_value_str:获取HTTP请求头中指定字段的值(字符串格式)
httpd_req_get_url_query_str:获取HTTP请求URL中的查询参数(字符串格式)
httpd_query_key_value:解析HTTP请求URL中的查询参数,获取指定参数名的值(字符串格式)
httpd_req_recv:从HTTP请求接收数据
httpd_req_send:发送HTTP响应数据
httpd_resp_set_type:设置HTTP响应内容的MIME类型
httpd_resp_send_chunk:分块发送HTTP响应数据。
例如,以下是一个URI处理器函数的示例,用于处理/echo路径的POST请求:

static esp_err_t echo_post_handler(httpd_req_t *req)
{char buf[1024];int ret, remaining = req->content_len;// 从HTTP请求中接收数据while (remaining > 0) {ret = httpd_req_recv(req, buf, MIN(remaining, sizeof(buf)));if (ret <= 0) {if (ret == HTTPD_SOCK_ERR_TIMEOUT) {// 处理超时httpd_resp_send_408(req);}return ESP_FAIL;}// 处理接收到的数据// ...remaining -= ret;}// 发送HTTP响应httpd_resp_set_type(req, HTTPD_TYPE_TEXT);httpd_resp_send(req, "Received data: ", -1);httpd_resp_send_chunk(req, buf, req->content_len);httpd_resp_send_chunk(req, NULL, 0);return ESP_OK;
}

5、 发送响应
在URI处理函数中,可以使用httpd_resp_send()函数将响应发送回客户端。该函数需要传入一个httpd_req_t结构体作为参数,该结构体表示HTTP请求和响应。

例如,在上面的hello_get_handler处理函数中,可以使用httpd_resp_send()函数将“Hello, World!”字符串作为响应发送回客户端:

static esp_err_t hello_get_handler(httpd_req_t *req)
{const char* resp_str = "Hello, World!";httpd_resp_send(req, resp_str, strlen(resp_str));return ESP_OK;
}

二、主要使用API的说明
1. httpd_register_uri_handler
用于将HTTP请求的URI路由到处理程序。这个函数接收两个参数:httpd_handle_t类型的HTTP服务器句柄和httpd_uri_t类型的URI配置。

2. httpd_handle_t
httpd_handle_t是HTTP服务器的一个句柄,它是通过httpd_start函数创建的。而httpd_uri_t则定义了HTTP请求的URI信息,包括URI路径、HTTP请求方法和处理函数等。

3. httpd_query_key_value获取变量值
httpd_query_key_value 用于从查询字符串中获取指定键的值。查询字符串是指URL中?后面的部分,包含多个键值对,每个键值对之间使用&分隔。例如,对于以下URL:
http://192.168.1.1/path/to/handler?key1=value1&key2=value2
获取其中的:
esp_err_t httpd_query_key_value(const char *query, const char *key, char *buf, size_t buf_len);

这是一个使用示例

char query_str[] = "key1=value1&key2=value2";
char key[] = "key1";
char value[16];if (httpd_query_key_value(query_str, key, value, sizeof(value)) == ESP_OK) {printf("value=%s\n", value);
} else {printf("key not found\n");
}

4. 获取get参数示例

下面定义的 handler 演示了如何从请求参数里解析 字符串param1和整型变量param2:

 esp_err_t index_handler(httpd_req_t *req)
{char* query_str = NULL;char param1_value[10] = {0};int param2_value=0;query_str = strstr(req->uri, "?");if(query_str!=NULL){query_str ++;httpd_query_key_value(query_str, "param1", param1_value, sizeof(param1_value));char param2_str[10] = {0};httpd_query_key_value(query_str, "param2", param2_str, sizeof(param2_str));param2_value = atoi(param2_str);}char resp_str[50] = {0};snprintf(resp_str, sizeof(resp_str), "param1=%s, param2=%d", param1_value, param2_value);httpd_resp_send(req, resp_str, strlen(resp_str));return ESP_OK;
}

5. 获取post参数示例

下面的示例代码中根据httpd_req_t的content_len来分配一个缓冲区,并解析请求中的POST参数:

esp_err_t post_demo_handler(httpd_req_t *req)
{char post_string[64];int post_int=0;if (req->content_len > 0){// 从请求体中读取POST参数char *buf = malloc(req->content_len + 1);int ret = httpd_req_recv(req, buf, req->content_len);if (ret <= 0){// 接收数据出错free(buf);return ESP_FAIL;}buf[req->content_len] = '\0';// 解析POST参数char *param_str;param_str = strtok(buf, "&");while (param_str != NULL){char *value_str = strchr(param_str, '=');if (value_str != NULL){*value_str++ = '\0';if (strcmp(param_str, "post_string") == 0){strncpy(post_string, value_str, sizeof(post_string));}else if (strcmp(param_str, "post_int") == 0){post_int = atoi(value_str);}}param_str = strtok(NULL, "&");}free(buf);}// 将结果打印输出printf("post_string=%s, post_int=%d\n", post_string, post_int);// 返回成功httpd_resp_send(req, NULL, 0);return ESP_OK;
}httpd_uri_t post_uri = {.uri = "/post",.method = HTTP_POST,.handler = post_demo_handler,.user_ctx = NULL};

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/731449.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

JavaWeb笔记 --- 一JDBC

一、JDBC JDBC就是Java操作关系型数据库的一种API DriverManager 注册驱动可以不写 Class.forName("com.mysql.jdbc.Driver"); Connection Statement ResultSet PrepareStatement 密码输入一个SQL脚本&#xff0c;直接登录 预编译开启在url中 数据库连接池

HTML5+CSS3+JS小实例:暗紫色Tabbar

实例:暗紫色Tabbar 技术栈:HTML+CSS+JS 效果: 源码: 【HTML】 <!DOCTYPE html> <html lang="zh-CN"><head><meta charset="UTF-8" /><meta name="viewport" content="width=device-width, initial-scal…

【梳理】k8s使用Operator搭建Flink集群

文章目录 架构图安装cert-manager依赖helm 安装operator运行集群实例k8s上的两种模式&#xff1a;Native和Standalone两种CRDemo1&#xff1a;Application 单任务Demo2&#xff1a;Session 多任务创建ingress 总结 架构图 参考&#xff1a;部署验证demo 安装cert-manager依赖 …

SSL证书是什么

SSL 证书就是遵守 SSL协议&#xff0c;由受信任的数字证书颁发机构CA&#xff0c;在验证服务器身份后颁发&#xff0c;具有服务器身份验证和数据传输加密功能。 Secure socket layer(SSL)安全协议是由Netscape Communication公司设计开发。 该安全协议主要用来提供对用户和服…

CSS中 ,有哪些方式可以隐藏页面元素

文章目录 CSS中 &#xff0c;有哪些方式可以隐藏页面元素实现方式display&#xff1a;nonevisibility:hiddenopacity:0设置height 、width属性为0position:absoluteclip-path小结 CSS中 &#xff0c;有哪些方式可以隐藏页面元素 实现方式 通过 css 实现隐藏元素方法有如下 : …

Testing Library - 简介

testing-library 软件包系列以用户为中心的方式帮助您测试 UI 组件。 你的测试越接近你的软件使用方式&#xff0c;它们能给你提供的信心就越多。 核心库 DOM Testing Library&#xff0c;是一个轻量级的解决方案&#xff0c;用于通过查询和与DOM节点&#xff08;无论是使用JSD…

【面试准备日常】从头复习mysql--20240308

1.mysql数据类型 a.数值类型 分类类型大小有符号(SIGNED)范围无符号(UNSIGNED)范围描述数值类型TINYINT1 byte(-128&#xff0c;127)(0&#xff0c;255)小整数值SMALLINT2 bytes(-32768&#xff0c;32767)(0&#xff0c;65535)大整数值MEDIUMINT3 bytes(-8388608&#xff0c;…

[蓝桥杯]接龙数列(C语言)

目录 题目链接 题目理解 解题思路 完整代码 重难点解答 *dp数组的具体用法 *对于dp[b]dp[a]1>dp[b]?dp[a]1:dp[b]的解释 题目链接 [蓝桥杯 2023 省 B] 接龙数列 - 洛谷 题目理解 这道题让我们求任给的一串数字&#xff0c;若想让其变成接龙数列最少需要删除的数字…

【程序员经常使用的算法】讲解

程序员经常使用的算法 程序员经常使用的一些算法包括&#xff1a; 1. 排序算法&#xff08;Sorting Algorithms&#xff09;: 冒泡排序&#xff08;Bubble Sort&#xff09;选择排序&#xff08;Selection Sort&#xff09;插入排序&#xff08;Insertion Sort&#xff09;归…

uniapp列表进入动画

app列表入场动画 - DCloud 插件市场 列表入场动画https://ext.dcloud.net.cn/plugin?id16957

中医把脉笔记

目录 寸关尺对应的五脏六腑自己给自己把脉把脉五布法定寸关尺分浮中沉分快慢辨阴阳看虚实 参考文章 寸关尺对应的五脏六腑 自己给自己把脉 up主道道总是睡不着的把脉教学视频 用中指按住小骨头下面一点&#xff0c;这是关脉&#xff0c;左手的关脉对应肝脏。 把脉五布法 定…

网络基础aaa

三次握手 四次挥手 网络模型 TCP or UDP 的特点 如何理解 TCP 的5层协议 TCP的5层协议是指计算机网络体系结构中&#xff0c;与TCP&#xff08;传输控制协议&#xff09;相关的五个层次。这五个层次从高到低依次是&#xff1a;应用层、传输层、网络层、数据链路层和物理层。每…

java注释的详尽解析

一、什么是注解 (1).注解的作用 ①&#xff1a;注解一般用于对程序的说明&#xff0c;就像注释一样&#xff0c;但是区别是注释是给人看的&#xff0c;但是注解是给程序看的。 ②&#xff1a;让编译器进行编译检查的作用&#xff0c;比如下边这个Override注解是重写的意思&am…

前端缓存使用规范

一、Cookie使用规范 cookie的存储空间非常有限且会携带在请求头中会浪费不必要的流量&#xff0c;如果仅仅是为存储数据&#xff0c;可以采用其他替代方案&#xff0c;例如 webStorage&#xff0c;非必要不使用cookie。 1、使用方法 注意&#xff1a;过期时间时需转换成UTC格…

内存安全的编程语言

美国政府新颁布《回归基础构件&#xff1a;通往安全软件之路》 《回归基础构件&#xff1a;通往安全软件之路》中&#xff0c;白宫国家网络主任办公室&#xff08;ONCD&#xff09;呼吁开发者使用「内存安全的编程语言」 内存安全的编程语言 根据NSA的建议&#xff0c;内存…

sqlyog社区版下载,数据库客户端,mysql

Downloads webyog/sqlyog-community Wiki GitHubhttps://github.com/webyog/sqlyog-community/wiki/Downloadssqlyog社区版下载

liteIDE 解决go root报错 go: cannot find GOROOT directory: c:\go

liteIDE环境配置 我使用的liteIDE为 x36 5.9.5版本 。在查看–>选项 中可以看到 LiteEnv&#xff0c;双击LiteEnv &#xff0c;在右侧选择对应系统的env文件&#xff0c;我的是win64系统&#xff0c;所以文件名为win64.env 再双击 win64.env &#xff0c;关闭当前窗口&…

git 初始化项目并上传到github

如果还没配置过&#xff0c;需要配置账号信息 git config --global user.name "baymax-collab" git config --global user.email "baymax-collabtest.com"创建一个新的存储库 git clone gitgithub.com:xxxx cd test git switch --create main touch READ…

C++ Qt开发:QHostInfo主机地址查询组件

Qt 是一个跨平台C图形界面开发库&#xff0c;利用Qt可以快速开发跨平台窗体应用程序&#xff0c;在Qt中我们可以通过拖拽的方式将不同组件放到指定的位置&#xff0c;实现图形化开发极大的方便了开发效率&#xff0c;本章将重点介绍如何运用QHostInfo组件实现对主机地址查询功能…

ROS——VirtualBox下载

下载&安装Virtualbox Oracle VM VirtualBox 根据电脑系统版本下载。 注意&#xff1a;前提是电脑cpu要开启虚拟化 根据自己的需求下载 双击开始安装 浏览可以更改下载位置&#xff0c;默认在C盘 然后一直点&#xff0c;是或下一步就好了 下载拓展包 后续需要连接使…