311 lines
9.7 KiB
C
311 lines
9.7 KiB
C
#include "http_server.h"
|
||
|
||
// HTML页面模板
|
||
#include "html_files.h"
|
||
|
||
// 解析HTTP请求
|
||
static void parse_http_request(uint8_t* buffer, uint16_t length, http_request* req)
|
||
{
|
||
char* method = strtok((char*)buffer, " ");
|
||
char* uri = strtok(NULL, " ");
|
||
|
||
if(strcmp(method, "GET") == 0) {
|
||
req->method = HTTP_GET;
|
||
} else if(strcmp(method, "POST") == 0) {
|
||
req->method = HTTP_POST;
|
||
} else {
|
||
req->method = HTTP_UNKNOWN;
|
||
}
|
||
|
||
strncpy(req->uri, uri, MAX_URI_SIZE);
|
||
|
||
// 如果是POST请求,查找请求体
|
||
if(req->method == HTTP_POST) {
|
||
char* body = strstr((char*)buffer, "\r\n\r\n");
|
||
if(body) {
|
||
body += 4;
|
||
req->body = (uint8_t*)body;
|
||
req->body_length = length - (body - (char*)buffer);
|
||
}
|
||
}
|
||
}
|
||
|
||
// 发送HTTP响应
|
||
static int send_all(uint8_t sn, const uint8_t* data, uint16_t length)
|
||
{
|
||
int total_sent = 0;
|
||
while (total_sent < length) {
|
||
int sent = send(sn, data + total_sent, length - total_sent);
|
||
if (sent <= 0) {
|
||
return -1; // 发送错误
|
||
}
|
||
total_sent += sent;
|
||
}
|
||
return total_sent;
|
||
}
|
||
|
||
static void send_chunked_response_header(uint8_t sn)
|
||
{
|
||
const char* header =
|
||
"HTTP/1.1 200 OK\r\n"
|
||
"Content-Type: text/html\r\n"
|
||
"Transfer-Encoding: chunked\r\n"
|
||
"\r\n";
|
||
|
||
send_all(sn, (uint8_t*)header, strlen(header));
|
||
}
|
||
|
||
static void send_chunk(uint8_t sn, const char* data, uint16_t length)
|
||
{
|
||
char chunk_header[10];
|
||
sprintf(chunk_header, "%X\r\n", length);
|
||
send_all(sn, (uint8_t*)chunk_header, strlen(chunk_header));
|
||
send_all(sn, (uint8_t*)data, length);
|
||
send_all(sn, (uint8_t*)"\r\n", 2);
|
||
}
|
||
|
||
static void send_chunk_end(uint8_t sn)
|
||
{
|
||
send_all(sn, (uint8_t*)"0\r\n\r\n", 5);
|
||
}
|
||
|
||
#define CHUNK_SIZE 512
|
||
|
||
// 处理配置页面请求
|
||
static void handle_config_page(uint8_t sn)
|
||
{
|
||
char buffer[CHUNK_SIZE];
|
||
|
||
// 先将完整的HTML内容格式化到一个临时缓冲区
|
||
char full_page[1024*15]; // 确保足够大
|
||
int total_len = sprintf(full_page, index_html,0,
|
||
gWIZNETINFO.mac[0], gWIZNETINFO.mac[1], gWIZNETINFO.mac[2],
|
||
gWIZNETINFO.mac[3], gWIZNETINFO.mac[4], gWIZNETINFO.mac[5],
|
||
gWIZNETINFO.ip[0], gWIZNETINFO.ip[1], gWIZNETINFO.ip[2], gWIZNETINFO.ip[3],
|
||
gWIZNETINFO.sn[0], gWIZNETINFO.sn[1], gWIZNETINFO.sn[2], gWIZNETINFO.sn[3],
|
||
gWIZNETINFO.gw[0], gWIZNETINFO.gw[1], gWIZNETINFO.gw[2], gWIZNETINFO.gw[3]);
|
||
|
||
// 发送响应头
|
||
send_chunked_response_header(sn);
|
||
|
||
// 分块发送内容
|
||
int sent = 0;
|
||
while (sent < total_len) {
|
||
int chunk_len = (total_len - sent) > CHUNK_SIZE ? CHUNK_SIZE : (total_len - sent);
|
||
memcpy(buffer, full_page + sent, chunk_len);
|
||
send_chunk(sn, buffer, chunk_len);
|
||
sent += chunk_len;
|
||
}
|
||
|
||
// 发送结束块
|
||
send_chunk_end(sn);
|
||
}
|
||
|
||
// 解析并更新网络配置
|
||
static void update_network_config(const char* body)
|
||
{
|
||
char mac[18], ip[16], subnet[16], gateway[16];
|
||
uint32_t mac_values[6], ip_values[4], subnet_values[4], gateway_values[4];
|
||
|
||
if(sscanf(body, "mac=%02X%%3A%02X%%3A%02X%%3A%02X%%3A%02X%%3A%02X&"
|
||
"ip=%d.%d.%d.%d&"
|
||
"subnet=%d.%d.%d.%d&"
|
||
"gateway=%d.%d.%d.%d",
|
||
&mac_values[0], &mac_values[1], &mac_values[2],
|
||
&mac_values[3], &mac_values[4], &mac_values[5],
|
||
&ip_values[0], &ip_values[1], &ip_values[2], &ip_values[3],
|
||
&subnet_values[0], &subnet_values[1], &subnet_values[2], &subnet_values[3],
|
||
&gateway_values[0], &gateway_values[1], &gateway_values[2], &gateway_values[3]) == 18)
|
||
{
|
||
// 更新MAC地址
|
||
for(int i = 0; i < 6; i++) {
|
||
gWIZNETINFO.mac[i] = (uint8_t)mac_values[i];
|
||
}
|
||
|
||
// 更新IP地址
|
||
for(int i = 0; i < 4; i++) {
|
||
gWIZNETINFO.ip[i] = (uint8_t)ip_values[i];
|
||
gWIZNETINFO.sn[i] = (uint8_t)subnet_values[i];
|
||
gWIZNETINFO.gw[i] = (uint8_t)gateway_values[i];
|
||
}
|
||
|
||
// 应用新的网络配置
|
||
wizchip_setnetinfo(&gWIZNETINFO);
|
||
}
|
||
}
|
||
|
||
void http_server_init(void)
|
||
{
|
||
// 创建服务器socket
|
||
socket(HTTP_SERVER_SOCKET, Sn_MR_TCP, HTTP_SERVER_PORT, 0);
|
||
listen(HTTP_SERVER_SOCKET);
|
||
}
|
||
|
||
|
||
|
||
// 三通阀结构体
|
||
typedef struct {
|
||
uint8_t angle1; // 阀门1角度 (120/210)
|
||
uint8_t angle2; // 阀门2角度 (120/210)
|
||
} ValveStatus;
|
||
|
||
// 泵结构体
|
||
typedef struct {
|
||
uint8_t status1; // 泵1运行状态 (停止/顺时针/逆时针)
|
||
uint8_t status2; // 泵2运行状态 (停止/顺时针/逆时针)
|
||
uint8_t speed1; // 泵1速度百分比 (0-100)
|
||
uint8_t speed2; // 泵2速度百分比 (0-100)
|
||
} PumpStatus;
|
||
|
||
// 设备状态结构体
|
||
// 1.下挂设备在位状态,在位为1,否则为0,当前为气泡感应器,1个字节
|
||
// 2.两个三通阀实时角度,每个角度占用1个字节,只能为210或120
|
||
// 3.泵状态,1表示泵顺时针转动中,2表示泵逆时针转动中,0表示停止,两个泵共2个字节
|
||
// 4.泵速度,2个字节,每个泵的速度占用一个字节,表示当前泵设定的速度百分比
|
||
// 5.气泡感应器读值,1表示监测到气泡,0表示没有气泡,共1个字节
|
||
// 6.急停按键状态,1个字节,1为按下,0为正常。
|
||
// 7.错误码1个字节,用于监控当前系统总体状态,如通信异常、控制异常等等,具体错误码待定
|
||
// 8.初始化状态,1个字节, 0表示初始化执行中,1表示初始化执行成功,2表示初始化执行失败。
|
||
typedef struct {
|
||
uint8_t deviceStatus; // 下挂设备状态
|
||
ValveStatus valves; // 两个三通阀状态
|
||
PumpStatus pumps; // 两个泵状态
|
||
uint8_t bubbleStatus; // 气泡状态
|
||
uint8_t stopStatus; // 急停状态
|
||
uint8_t errorCode; // 错误码
|
||
uint8_t initStatus; // 初始化状态
|
||
} DeviceStatus;
|
||
DeviceStatus deviceStatus={
|
||
.deviceStatus=1,
|
||
.valves={
|
||
.angle1=0,
|
||
.angle2=0
|
||
},
|
||
.pumps={
|
||
.status1=0,
|
||
.status2=0,
|
||
.speed1=60,
|
||
.speed2=70
|
||
},
|
||
.bubbleStatus=0,
|
||
.stopStatus=0,
|
||
.errorCode=0,
|
||
.initStatus=1
|
||
};
|
||
|
||
// 发送JSON响应头
|
||
static void send_json_response_header(uint8_t sn)
|
||
{
|
||
const char* header =
|
||
"HTTP/1.1 200 OK\r\n"
|
||
"Content-Type: application/json\r\n"
|
||
"Access-Control-Allow-Origin: *\r\n"
|
||
"Connection: close\r\n" // 明确指示关闭连接
|
||
"Cache-Control: no-cache, no-store\r\n" // 禁止缓存
|
||
"\r\n";
|
||
|
||
send_all(sn, (uint8_t*)header, strlen(header));
|
||
}
|
||
|
||
// 处理状态请求
|
||
static void handle_status_request(uint8_t sn)
|
||
{
|
||
char json_buffer[512];
|
||
// DeviceStatus deviceStatus; // 确保能访问到deviceStatus变量
|
||
|
||
// 构建JSON响应
|
||
int len = sprintf(json_buffer,
|
||
"{"
|
||
"\"deviceStatus\":%d,"
|
||
"\"valves\":{"
|
||
"\"angle1\":%d,"
|
||
"\"angle2\":%d"
|
||
"},"
|
||
"\"pumps\":{"
|
||
"\"status1\":%d,"
|
||
"\"status2\":%d,"
|
||
"\"speed1\":%d,"
|
||
"\"speed2\":%d"
|
||
"},"
|
||
"\"bubbleStatus\":%d,"
|
||
"\"stopStatus\":%d,"
|
||
"\"errorCode\":%d,"
|
||
"\"initStatus\":%d"
|
||
"}",
|
||
deviceStatus.deviceStatus,
|
||
deviceStatus.valves.angle1,
|
||
deviceStatus.valves.angle2,
|
||
deviceStatus.pumps.status1,
|
||
deviceStatus.pumps.status2,
|
||
deviceStatus.pumps.speed1,
|
||
deviceStatus.pumps.speed2,
|
||
deviceStatus.bubbleStatus,
|
||
deviceStatus.stopStatus,
|
||
deviceStatus.errorCode,
|
||
deviceStatus.initStatus
|
||
);
|
||
|
||
// 发送响应
|
||
send_json_response_header(sn);
|
||
send_all(sn, (uint8_t*)json_buffer, len);
|
||
|
||
// 发送完成后主动关闭连接
|
||
disconnect(sn);
|
||
printf("send status success!\r\n");
|
||
}
|
||
|
||
void http_server_task(void)
|
||
{
|
||
uint8_t buffer[2048];
|
||
uint16_t size;
|
||
uint8_t status = getSn_SR(HTTP_SERVER_SOCKET);
|
||
|
||
switch(status)
|
||
{
|
||
case SOCK_INIT:
|
||
listen(HTTP_SERVER_SOCKET);
|
||
break;
|
||
|
||
case SOCK_LISTEN:
|
||
// 等待连接
|
||
break;
|
||
|
||
case SOCK_ESTABLISHED:
|
||
{
|
||
if((size = getSn_RX_RSR(HTTP_SERVER_SOCKET)) > 0)
|
||
{
|
||
size = recv(HTTP_SERVER_SOCKET, buffer, sizeof(buffer));
|
||
if(size > 0)
|
||
{
|
||
http_request req;
|
||
parse_http_request(buffer, size, &req);
|
||
|
||
if(req.method == HTTP_GET) {
|
||
if(strcmp(req.uri, "/status") == 0) {
|
||
handle_status_request(HTTP_SERVER_SOCKET);
|
||
} else {
|
||
handle_config_page(HTTP_SERVER_SOCKET);
|
||
}
|
||
}
|
||
else if(req.method == HTTP_POST && strcmp(req.uri, "/config") == 0) {
|
||
update_network_config((char*)req.body);
|
||
send_chunked_response_header(HTTP_SERVER_SOCKET);
|
||
send_chunk(HTTP_SERVER_SOCKET,
|
||
"<html><head><meta http-equiv='refresh' content='0;url=/'></head></html>",
|
||
strlen("<html><head><meta http-equiv='refresh' content='0;url=/'></head></html>"));
|
||
send_chunk_end(HTTP_SERVER_SOCKET);
|
||
}
|
||
}
|
||
}
|
||
break;
|
||
}
|
||
|
||
case SOCK_CLOSE_WAIT:
|
||
disconnect(HTTP_SERVER_SOCKET);
|
||
break;
|
||
|
||
case SOCK_CLOSED:
|
||
socket(HTTP_SERVER_SOCKET, Sn_MR_TCP, HTTP_SERVER_PORT, 0);
|
||
break;
|
||
}
|
||
}
|