如何使用lwIP堆栈发送简单的HTTP请求?

15

如果问题不相关,请将其移动/关闭。

核心:Cortex-M4

微处理器:TI TM4C1294NCPDT。

IP堆栈:lwIP 1.4.1

我正在使用这个微处理器进行一些数据记录,并希望通过HTTP请求以以下形式向单独的Web服务器发送一些信息:

http://123.456.789.012:8800/process.php?data1=foo&data2=bar&time=1234568789

我希望处理器能够查看响应头(即是否为200 OK或发生了某些错误),但它不必显示/接收实际内容。

lwIP为微处理器提供了一个http服务器,但我需要相反的(微处理器是客户端)。

我不确定数据包和请求/响应头如何关联,因此不确定如何发送/接收信息。

3个回答

24

实现起来非常简单,只是忘了更新这个问题。

我基本上按照网站上给出的说明,也就是Raw/TCP“文档”。

基本上,HTTP请求被编码在TCP数据包中,所以为了向我的PHP服务器发送数据,我使用TCP数据包发送了一个HTTP请求(lwIP完成所有工作)。

我想要发送的HTTP数据包如下:

HEAD /process.php?data1=12& data2 = 5 HTTP/1.0

主机:mywebsite.com

要将此“翻译”为HTTP服务器能够理解的文本,您必须在代码中添加“\r\n”回车符/换行符。因此它看起来像这样:

char *string = "HEAD /process.php?data1=12&data2=5 HTTP/1.0\r\nHost: mywebsite.com\r\n\r\n ";

请注意,结尾有两个“\r\n”。

你可以使用GET或HEAD,但由于我不关心PHP服务器返回的HTML网站,所以我使用了HEAD(它在成功时返回200 OK,或在失败时返回不同的代码)。

lwIP原始/ TCP基于回调函数。您基本上设置所有回调函数,然后将要发送的数据推送到TCP缓冲区(在本例中为上面指定的TCP字符串),然后告诉lwIP发送数据包。

用于建立TCP连接的函数(每次我想发送TCP数据包时,我的应用程序都会直接调用此函数):

void tcp_setup(void)
{
    uint32_t data = 0xdeadbeef;

    /* create an ip */
    struct ip_addr ip;
    IP4_ADDR(&ip, 110,777,888,999);    //IP of my PHP server

    /* create the control block */
    testpcb = tcp_new();    //testpcb is a global struct tcp_pcb
                            // as defined by lwIP


    /* dummy data to pass to callbacks*/

    tcp_arg(testpcb, &data);

    /* register callbacks with the pcb */

    tcp_err(testpcb, tcpErrorHandler);
    tcp_recv(testpcb, tcpRecvCallback);
    tcp_sent(testpcb, tcpSendCallback);

    /* now connect */
    tcp_connect(testpcb, &ip, 80, connectCallback);

}

一旦与我的PHP服务器建立连接,lwIP将调用“connectCallback”函数:

/* connection established callback, err is unused and only return 0 */
err_t connectCallback(void *arg, struct tcp_pcb *tpcb, err_t err)
{
    UARTprintf("Connection Established.\n");
    UARTprintf("Now sending a packet\n");
    tcp_send_packet();
    return 0;
}

该函数调用实际函数tcp_send_packet(),发送HTTP请求,如下:

uint32_t tcp_send_packet(void)
{
    char *string = "HEAD /process.php?data1=12&data2=5 HTTP/1.0\r\nHost: mywebsite.com\r\n\r\n ";
    uint32_t len = strlen(string);

    /* push to buffer */
        error = tcp_write(testpcb, string, strlen(string), TCP_WRITE_FLAG_COPY);

    if (error) {
        UARTprintf("ERROR: Code: %d (tcp_send_packet :: tcp_write)\n", error);
        return 1;
    }

    /* now send */
    error = tcp_output(testpcb);
    if (error) {
        UARTprintf("ERROR: Code: %d (tcp_send_packet :: tcp_output)\n", error);
        return 1;
    }
    return 0;
}

一旦TCP包已发送(如果您只想“碰运气”并且不在乎数据是否实际发送,则这就足够了),PHP服务器将返回一个TCP包(带有200 OK等以及HTML代码,如果使用GET而不是HEAD)。此代码可以在以下代码中读取和验证:

err_t tcpRecvCallback(void *arg, struct tcp_pcb *tpcb, struct pbuf *p, err_t err)
{
    UARTprintf("Data recieved.\n");
    if (p == NULL) {
        UARTprintf("The remote host closed the connection.\n");
        UARTprintf("Now I'm closing the connection.\n");
        tcp_close_con();
        return ERR_ABRT;
    } else {
        UARTprintf("Number of pbufs %d\n", pbuf_clen(p));
        UARTprintf("Contents of pbuf %s\n", (char *)p->payload);
    }

    return 0;
}

p->payload包含实际的“200 OK”等信息。希望这可以帮助某些人。

我在上面的代码中省略了一些错误检查,以简化答案。


4
太棒了!非常感谢你的帮助。lwIP文档缺少像这样的基本示例。但要小心,我猜用户代码必须在tcpRecvCallback中手动调用pbuf_free(p),否则内存不会从PBUF_POOL中释放。我也遇到了tcp_close(pcb)的一些问题,它返回ERR_OK,但pcb永远不会被释放并且无限期地停留在CLOSE_WAIT状态。我不得不在调用tcp_close之前将tcp_input_pcb = NULL作为解决方法... - Motla
这对我很有效。谢谢@tgun926。我的一个问题是,我正在使用它每5分钟向服务器发送数据。当tcp_connect调用时,它会卡在分段部分。你能帮我解决这个问题吗?我只想使用这些函数将数据发送到服务器,并且我有多个服务器。因此,我每次都必须使用不同的IP连接tcp。 - Jay Katira
我已经尝试在sys_init()和lwip_init()被调用后的主循环中使用您的代码,但似乎除非我明确提供(并绑定到)我的本地IP地址,否则无法发出HTTP请求。您有什么想法为什么会这样吗?在您提供的文档链接中,它说调用tcp_bind()是可选的(但在我的情况下不是)。 - DEls

1
请看维基百科中的HTTP example。客户端将发送GET和HOST行。服务器将响应许多响应行。第一行将包含响应代码。

是的,我已经多次看过了。我想要一个更具体的答案,关于如何通过lwIP库来完成它。我认为我几乎通过原始/ TCP API弄清楚了——很快会测试一下。 - tgun926

-1
我成功创建了一个 HTTP 客户端,用于树莓派 Pico W,使用的示例可以在这里找到。
它使用 sdk 中的 httpc_get_file 或 httpc_get_file_dns 函数。
然而,那个示例是 不完整的,因为它有一个内存泄漏问题。 你需要释放 headers 函数中由结构体 pbuf *hdr 占用的内存,以及 body 函数中的结构体 pbuf *p,分别使用 pbuf_free(hdr); 和 pbuf_free(p);。
如果没有进行这些修改,它将在大约 20 次调用后停止工作(可能取决于响应的大小)。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接