2014-10-04 3 views
6

Пожалуйста, переместите/закройте это, если вопрос не имеет значения.Как отправить простой HTTP-запрос с помощью стека lwIP?

Core: Cortex-M4

Микропроцессор: TI TM4C1294NCPDT.

IP Stack: LwIP 1.4.1

Я использую этот микропроцессор, чтобы сделать некоторые записи данных, и я хочу, чтобы отправить информацию на отдельный веб-сервер с помощью запроса HTTP в виде:

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

и я хочу, чтобы процессор мог видеть заголовок ответа (то есть, если он был 200 OK или что-то пошло не так) - ему не нужно отображать/получать фактическое содержимое.

lwIP имеет http-сервер для микропроцессора, но я после противоположного (микропроцессор - клиент).

Я не уверен, как пакеты соотносятся с заголовками запроса/ответа, поэтому я не уверен, как я собираюсь отправлять/получать информацию.

ответ

12

Это оказалось довольно простым в реализации, забыл обновить этот вопрос.

Я в значительной степени следовал инструкциям, приведенным на сайте this, который является документацией Raw/TCP.

В принципе, HTTP-запрос закодирован в TCP-пакетах, поэтому для отправки данных на мой PHP-сервер я отправил HTTP-запрос с использованием TCP-пакетов (lwIP выполняет всю работу).

HTTP-пакет Я хочу послать выглядит так:

ГОЛОВА /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 "; 

Обратите внимание, что конец имеет две серии «\ г \ п»

Вы можете использовать GET или HEAD, а потому, что я не забочусь о HTML сайте мой PHP сервер, я использовал HEAD (он возвращает 200 OK при успехе или другой код при ошибке).

LwIP raw/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-сервер установлен, ' функция connectCallback»вызывается LwIP:

/* 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-> полезная нагрузка содержит фактический «200 OK» и т.п. информация. Надеюсь, это помогает кому-то.

Я пропустил некоторые ошибки в моем коде выше, чтобы упростить ответ.

0

Посмотрите на HTTP example в Википедии. Клиент отправит линии GET и HOST. Для ответа на сервер будет отвечать множество строк. Первая строка будет содержать код ответа.

+0

Да, я смотрел на это много раз. Я пошел за более конкретным ответом на это, используя библиотеку lwIP. Я думаю, что я почти понял это через raw/tcp api - скоро проверит его. – tgun926