lwip1.4.0 http server实现及POST 实现

lwip1.4.0http server实现及POST 实现

 

一、HTTP SERVER的实现

 lwip默认的http server 在 apps/httpserver_raw 主要核心文件为 fs.c fs.h(读取相关html相关资源), httpd.c httpd.h httpd_structs.h http协议核心文件

 

首先在LWIP协议栈正常运行后 需要在main函数中调用httpd_init() 初始化Http 正常情况下般还需要实现SSICGI回调函数的初始工作 本人写在一个函数中 如下:

void http_start(void)

{

   http_set_ssi_handler(SSIHandler, g_pcConfigSSITags, sizeof(g_pcConfigSSITags)/sizeof (char *));

   http_set_cgi_handlers(g_psConfigCGIURIs, sizeof(g_psConfigCGIURIs)/sizeof(tCGI));

}

然在httpd_init()下调用http_start() 完成初始化SSICGI的工作。

 

其次 要使用makefsfile.exe 对网页进行编译 这个小工具可以从网上下载一个本人将编译命令写在一个makefsfile.bat批处理文件中 每次编译只要运行一下makefsfile.bat 具体命令如下:

 

echo off

makefsfile -i web_pages -o ../lwip-1.4.0/src/apps/httpserver_raw/fsdata.h -r -h

echo on

 

其中 web_pages为所包含的网页文件夹 产生的网页数据放在fsdata.h中 用于跟工程文件一起编译, -r 表示每次编译网页时重写fsdata.h  -h 表示产生的网页数据中不包含http协议头部 因为本人在HTTP中使用的是动态产生HTTP协议头。

 

至此 网页完成 在浏览器中敲入板子的IP地址 便可以看到网页了

 

二、POST方案实现

LWIP HTTP 协议中默认只支持GET方法 但是一般提交表单时都用POST方法 而LWIPPOST方案需要自己实现 不过LWIP已经需要实现的函数申明在httpd.h中了

首先将宏 LWIP_HTTPD_SUPPORT_POST 设置成表示支持HTTP POST 方法需要实现的函数分别为: httpd_post_begin(当接收到一个POST请求时会调用此函数), httpd_post_receive_data(接收HTTP POST 数据), httpd_post_finished(接收完成后 调用此函数)

具体实现如下:

err_t httpd_post_begin(void *connection, const char *uri, const char *http_request,

                       u16_t http_request_len, int content_len, char *response_uri,

                       u16_t response_uri_len, u8_t *post_auto_wnd)

{

#if LWIP_HTTPD_CGI

  int i = 0;

#endif

struct http_state *hs = (struct http_state *)connection;

 

 

 if(!uri || (uri[0] == '\0')) {

    return ERR_ARG;

 }

 

 hs->cgi_handler_index = -1;   // 此变量为本人自己在struct http_state 添加 用于保存CGI handler 索引 为-1表示无CGI handler索引

 hs->response_file = NULL; // 此变量为本人自己在struct http_state 添加 用于保存 CGI handler 处理完后返回的响应uri.

 

#if LWIP_HTTPD_CGI

 

  if (g_iNumCGIs && g_pCGIs) {

    for (i = 0; i < g_iNumCGIs; i++) {

      if (strcmp(uri, g_pCGIs[i].pcCGIName) == 0) {

         

         hs->cgi_handler_index = i; // 找到响应的 CGI handler 将其保存在cgi_handler_index 以便在httpd_post_receive_data中使用

         break;

       }

    }

  }

 

 

  if(i == g_iNumCGIs) {

    return ERR_ARG; // 未找到CGI handler 

  }

#endif

 

  return ERR_OK;

}

 

#define LWIP_HTTPD_POST_MAX_PAYLOAD_LEN     512

static char http_post_payload[LWIP_HTTPD_POST_MAX_PAYLOAD_LEN];

static u16_t http_post_payload_len = 0;

 

err_t httpd_post_receive_data(void *connection, struct pbuf *p)

{

    struct http_state *hs = (struct http_state *)connection;

    struct pbuf *q = p;

    int count;

    u32_t http_post_payload_full_flag = 0;

 

    while(q != NULL)  // 缓存接收的数据至http_post_payload

    {

     

      if(http_post_payload_len + q->len <= LWIP_HTTPD_POST_MAX_PAYLOAD_LEN) {

          MEMCPY(http_post_payload+http_post_payload_len, q->payload, q->len);

          http_post_payload_len += q->len;

      }

      else {  // 缓存溢出 置溢出标志位

        http_post_payload_full_flag = 1;

        break;

      }

      q = q->next;

    }

 

    pbuf_free(p); // 释放pbuf

 

    if(http_post_payload_full_flag) // 缓存溢出 则丢弃数据

    {

        http_post_payload_full_flag = 0;

        http_post_payload_len = 0;

        hs->cgi_handler_index = -1;

        hs->response_file = NULL;

    }

    else if(hs->post_content_len_left == 0) {  // POST数据已经接收完毕 则处理

       

        if(hs->cgi_handler_index != -1) {

            count = extract_uri_parameters(hs, http_post_payload);  // 解析

            hs->response_file = g_pCGIs[hs->cgi_handler_index].pfnCGIHandler(hs->cgi_handler_index, count, hs->params,

                                             hs->param_vals); // 调用解析函数

            http_post_payload_len = 0;

        }

       

        else {

            hs->response_file = NULL;

            http_post_payload_len = 0;

        }

    }

 

    return ERR_OK;

}

 

void httpd_post_finished(void *connection, char *response_uri, u16_t response_uri_len)

{

    struct http_state *hs = (struct http_state *)connection;

    if(hs->response_file != NULL) {

        strncpy(response_uri, hs->response_file,response_uri_len); // 拷贝uri 用于给浏览器响应相应的请求

    }

}

 

至此 HTTP SERVER 和 HTTP POST 方法实现完成


  • 9
    点赞
  • 38
    收藏
    觉得还不错? 一键收藏
  • 35
    评论
lwIP(Lightweight IP)是一个轻量级的网络协议栈,支持TCP/IP协议。它的设计目标是在小型嵌入式系统中提供高性能的网络通信能力。 下面是一个简单的lwIP HTTP服务器的代码实现: ```c #include "lwip/tcp.h" void http_server_task(void *arg) { struct tcp_pcb *pcb; struct tcp_pcb *newpcb; err_t err; err_t accept_err; u16_t port = 80; LWIP_UNUSED_ARG(arg); pcb = tcp_new(); if (pcb == NULL) { LWIP_DEBUGF(HTTPD_DEBUG, ("http_server_task: Error creating PCB. Out of Memory\n")); return; } err = tcp_bind(pcb, IP_ADDR_ANY, port); if (err != ERR_OK) { LWIP_DEBUGF(HTTPD_DEBUG, ("http_server_task: Unable to bind to port 80. Error code %d\n", err)); return; } pcb = tcp_listen(pcb); tcp_accept(pcb, http_server_accept); } void http_server_accept(void *arg, struct tcp_pcb *pcb, err_t err) { LWIP_UNUSED_ARG(arg); if (err != ERR_OK) { tcp_close(pcb); return; } tcp_arg(pcb, pcb); tcp_recv(pcb, http_server_recv); } void http_server_recv(void *arg, struct tcp_pcb *pcb, struct pbuf *p, err_t err) { LWIP_UNUSED_ARG(arg); if (err == ERR_OK && p != NULL) { /* Process HTTP request */ /* ... */ /* Send HTTP response */ /* ... */ tcp_write(pcb, response_buffer, response_length, TCP_WRITE_FLAG_COPY); tcp_close(pcb); pbuf_free(p); } else { tcp_close(pcb); } } ``` 这个代码实现了一个基本的HTTP服务器,它监听端口80并等待连接。一旦连接建立,它将接收HTTP请求并发送HTTP响应。在这个简单的例子中,我们只是发送一个硬编码的响应,但是你可以根据需要修改代码来生成动态响应。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 35
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值