src/server/proxy/httpclient.c

Sat, 14 Mar 2026 13:36:39 +0100

author
Olaf Wintermann <olaf.wintermann@gmail.com>
date
Sat, 14 Mar 2026 13:36:39 +0100
changeset 732
2aeaf6f32861
parent 730
c190ce0aab98
permissions
-rw-r--r--

fix httpclient termination in case an external event has called the last http_client_process

662
70fdf948b642 refactor HttpParser to support parsing of Http responses
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
1 /*
665
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
2 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
3 *
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
4 * Copyright 2026 Olaf Wintermann. All rights reserved.
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
5 *
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
6 * Redistribution and use in source and binary forms, with or without
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
7 * modification, are permitted provided that the following conditions are met:
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
8 *
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
9 * 1. Redistributions of source code must retain the above copyright
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
10 * notice, this list of conditions and the following disclaimer.
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
11 *
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
12 * 2. Redistributions in binary form must reproduce the above copyright
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
13 * notice, this list of conditions and the following disclaimer in the
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
14 * documentation and/or other materials provided with the distribution.
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
15 *
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
16 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
17 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
19 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
20 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
21 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
22 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
23 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
24 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
25 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
26 * POSSIBILITY OF SUCH DAMAGE.
662
70fdf948b642 refactor HttpParser to support parsing of Http responses
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
27 */
70fdf948b642 refactor HttpParser to support parsing of Http responses
Olaf Wintermann <olaf.wintermann@gmail.com>
parents:
diff changeset
28
665
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
29 #include "httpclient.h"
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
30
673
144bdc33fdb6 add socket utils
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 672
diff changeset
31 #include "../util/socket.h"
144bdc33fdb6 add socket utils
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 672
diff changeset
32
666
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
33 #include <cx/buffer.h>
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
34 #include <cx/string.h>
690
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
35 #include <cx/printf.h>
669
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
36 #include <stdlib.h>
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
37 #include <string.h>
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
38 #include <errno.h>
666
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
39
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
40 static int client_connected(EventHandler *ev, Event *event);
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
41 static int client_io(EventHandler *ev, Event *event);
717
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
42 static int client_ws_io(EventHandler *ev, Event *event);
707
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
43 static int client_process(HttpClient *client, Event *event);
717
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
44 static int client_ws_process(HttpClient *client, Event *event);
672
226bfd584075 minimally working httpclient
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 671
diff changeset
45 static int client_finished(EventHandler *ev, Event *event);
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
46
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
47 static int client_send_buf(HttpClient *client);
680
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
48 static int client_send_request_body(HttpClient *client);
683
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
49 static int client_read_response_header(HttpClient *client);
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
50 static int client_read_response_body(HttpClient *client);
666
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
51
665
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
52 HttpClient* http_client_new(EventHandler *ev) {
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
53 CxMempool *mp = cxMempoolCreate(32, CX_MEMPOOL_TYPE_PURE);
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
54 if(!mp) {
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
55 return NULL;
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
56 }
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
57
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
58 HttpClient *client = malloc(sizeof(HttpClient));
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
59 HeaderArray *req_headers = header_array_create();
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
60 HeaderArray *resp_headers = header_array_create();
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
61 if(!client || !req_headers || !resp_headers) {
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
62 free(client);
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
63 header_array_free(req_headers);
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
64 header_array_free(resp_headers);
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
65 cxMempoolFree(mp);
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
66 return NULL;
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
67 }
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
68
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
69 memset(client, 0, sizeof(HttpClient));
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
70 client->ev = ev;
672
226bfd584075 minimally working httpclient
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 671
diff changeset
71 client->socketfd = -1;
665
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
72 client->request_headers = req_headers;
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
73 client->response_headers = resp_headers;
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
74
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
75 client->buffer.maxsize = HTTP_CLIENT_BUFFER_SIZE;
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
76 client->buffer.inbuf = malloc(HTTP_CLIENT_BUFFER_SIZE);
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
77 HttpParser *parser = http_parser_new2(1, &client->buffer, resp_headers);
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
78 if(!parser || !client->buffer.inbuf) {
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
79 http_client_free(client);
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
80 return NULL;
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
81 }
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
82 client->parser = parser;
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
83
665
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
84 return client;
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
85 }
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
86
669
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
87 void http_client_free(HttpClient *client) {
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
88 cxMempoolFree(client->mp);
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
89 header_array_free(client->request_headers);
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
90 http_parser_free(client->parser);
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
91 if(client->stream) {
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
92 client->stream->st.free(&client->stream->st);
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
93 }
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
94 free(client->buffer.inbuf);
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
95 free(client->transfer_buffer);
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
96 free(client->transfer2_buffer);
669
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
97 free(client->addr);
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
98 free(client->method);
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
99 free(client->uri);
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
100 free(client);
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
101 }
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
102
700
658f4c02b4c5 fix proxy connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 698
diff changeset
103 int http_client_set_addr(HttpClient *client, int domain, const struct sockaddr *addr, socklen_t addrlen) {
665
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
104 free(client->addr);
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
105 client->addr = NULL;
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
106 client->addrlen = 0;
700
658f4c02b4c5 fix proxy connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 698
diff changeset
107 client->domain = domain;
665
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
108
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
109 void *newaddr = malloc(addrlen);
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
110 if(!newaddr) {
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
111 return 1;
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
112 }
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
113 memcpy(newaddr, addr, addrlen);
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
114 client->addr = newaddr;
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
115 client->addrlen = addrlen;
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
116
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
117 return 0;
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
118 }
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
119
701
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
120 int http_client_set_socket(HttpClient *client, int socketfd) {
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
121 client->socketfd = socketfd;
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
122 if(util_socket_setnonblock(socketfd, 1)) {
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
123 client->socketfd = -1;
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
124 return 1;
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
125 }
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
126 return 0;
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
127 }
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
128
669
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
129 int http_client_set_method(HttpClient *client, const char *method) {
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
130 return http_client_set_method_len(client, method, method ? strlen(method) : 0);
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
131 }
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
132
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
133 int http_client_set_uri(HttpClient *client, const char *uri) {
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
134 return http_client_set_uri_len(client, uri, uri ? strlen(uri) : 0);
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
135 }
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
136
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
137 static int client_set_str(char **ptr, const char *str, size_t len) {
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
138 free(*ptr);
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
139 if(str) {
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
140 char *newvalue = malloc(len+1);
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
141 if(!newvalue) {
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
142 *ptr = NULL;
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
143 return 1;
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
144 }
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
145 memcpy(newvalue, str, len);
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
146 newvalue[len] = 0;
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
147 *ptr = newvalue;
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
148 } else {
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
149 *ptr = NULL;
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
150 }
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
151 return 0;
669
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
152 }
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
153
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
154 int http_client_set_method_len(HttpClient *client, const char *method, size_t len) {
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
155 return client_set_str(&client->method, method, len);
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
156 }
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
157
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
158 int http_client_set_uri_len(HttpClient *client, const char *uri, size_t len) {
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
159 return client_set_str(&client->uri, uri, len);
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
160 }
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
161
665
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
162 int http_client_add_request_header(HttpClient *client, cxmutstr name, cxmutstr value) {
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
163 return header_array_add(client->request_headers, name, value);
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
164 }
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
165
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
166 int http_client_add_request_header_copy(HttpClient *client, cxstring name, cxstring value) {
675
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
167 if(!client->mp) {
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
168 client->mp = cxMempoolCreate(64, CX_MEMPOOL_TYPE_PURE);
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
169 if(!client->mp) {
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
170 return 1;
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
171 }
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
172 }
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
173
665
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
174 cxmutstr n = cx_strdup_a(client->mp->allocator, name);
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
175 cxmutstr v = cx_strdup_a(client->mp->allocator, value);
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
176
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
177 int err = 1;
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
178 if(n.ptr && v.ptr) {
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
179 err = http_client_add_request_header(client, n, v);
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
180 }
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
181 if(err) {
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
182 cxFree(client->mp->allocator, n.ptr);
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
183 cxFree(client->mp->allocator, v.ptr);
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
184 }
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
185 return err;
b8d5b797d090 add first http client code
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 662
diff changeset
186 }
666
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
187
675
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
188 int http_client_set_content_length(HttpClient *client, int64_t contentlength) {
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
189 client->req_content_length = contentlength;
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
190 char ctlen_buf[32];
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
191 size_t len = snprintf(ctlen_buf, 32, "%" PRId64, contentlength);
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
192 return http_client_add_request_header_copy(client, cx_str("content-length"), cx_strn(ctlen_buf, len));
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
193 }
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
194
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
195 int http_client_enable_chunked_transfer_encoding(HttpClient *client) {
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
196 client->req_content_length = -1;
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
197 return http_client_add_request_header(client, cx_mutstr("transfer-encoding"), cx_mutstr("chunked"));
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
198 }
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
199
704
778dcf4ad63c fix multiple reads could corrupt the httpclient buffer content
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 703
diff changeset
200 static int client_start_poll(HttpClient *client) {
701
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
201 client->event.fn = client_connected;
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
202 client->event.finish = client_finished;
704
778dcf4ad63c fix multiple reads could corrupt the httpclient buffer content
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 703
diff changeset
203 return ev_poll(client->ev, client->socketfd, &client->event);
701
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
204 }
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
205
666
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
206 int http_client_start(HttpClient *client) {
704
778dcf4ad63c fix multiple reads could corrupt the httpclient buffer content
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 703
diff changeset
207 client->event.events = EVENT_POLLOUT;
701
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
208 client->event.cookie = client;
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
209 if(client->socketfd != -1) {
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
210 int ret = client_connected(client->ev, &client->event);
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
211 if(ret != 0) {
704
778dcf4ad63c fix multiple reads could corrupt the httpclient buffer content
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 703
diff changeset
212 return client_start_poll(client);
701
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
213 }
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
214 return 0;
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
215 }
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
216
700
658f4c02b4c5 fix proxy connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 698
diff changeset
217 int socketfd = socket(client->domain, SOCK_STREAM, 0);
666
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
218 if(socketfd < 0) {
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
219 return 1;
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
220 }
673
144bdc33fdb6 add socket utils
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 672
diff changeset
221 if(util_socket_setnonblock(socketfd, 1)) {
666
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
222 return 1;
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
223 }
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
224 client->socketfd = socketfd;
704
778dcf4ad63c fix multiple reads could corrupt the httpclient buffer content
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 703
diff changeset
225
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
226 int ret = 1;
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
227 if(connect(socketfd, client->addr, client->addrlen)) {
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
228 int err = errno;
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
229 if(err == EINPROGRESS) {
704
778dcf4ad63c fix multiple reads could corrupt the httpclient buffer content
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 703
diff changeset
230 ret = client_start_poll(client);
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
231 } else {
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
232 log_ereport(LOG_FAILURE, "http-client-start: connect failed: %s", strerror(err));
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
233 }
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
234 } else {
691
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
235 ret = 0; // TODO
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
236 }
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
237
666
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
238 if(ret) {
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
239 close(socketfd);
701
936e5487418a add first full httpclient test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 700
diff changeset
240 client->socketfd = -1;
666
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
241 }
669
ccdc97fd8204 setup HttpClient in proxy SAF
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 668
diff changeset
242 return ret;
666
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
243 }
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
244
707
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
245 int http_client_process(HttpClient *client, Event *event) {
720
8c7d08d3be2e add support for websockets in the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 719
diff changeset
246 int ret = client->stage != 2 ? client_process(client, event) : client_ws_process(client, event);
707
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
247 if(ret && client->error == 0 && client->event.fn == NULL) {
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
248 if(client_start_poll(client)) {
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
249 client->error = 1;
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
250 }
706
df64b4b79912 add error handling in proxy_request_read
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 704
diff changeset
251 }
707
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
252 return ret;
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
253 }
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
254
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
255 size_t http_client_message_buf_size_available(HttpClient *client) {
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
256 return client->transfer_buffer_alloc - client->transfer_buffer_len;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
257 }
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
258
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
259 int http_client_add_message(HttpClient *client, const void *buf, size_t size) {
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
260 size_t available = http_client_message_buf_size_available(client);
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
261 if(available == 0) {
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
262 return HTTP_CLIENT_CALLBACK_WOULD_BLOCK;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
263 }
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
264 if(size > available) {
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
265 size = available;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
266 }
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
267 memcpy(client->transfer_buffer + client->transfer_buffer_len, buf, size);
718
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
268 client->transfer_buffer_len += size;
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
269 return size;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
270 }
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
271
666
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
272 static int create_req_buffer(HttpClient *client) {
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
273 CxBuffer buf;
675
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
274 if(cxBufferInit(&buf, cxDefaultAllocator, NULL, HTTP_CLIENT_BUFFER_SIZE, CX_BUFFER_AUTO_EXTEND)) {
666
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
275 return 1;
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
276 }
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
277
726
5ad3bda4aca1 fix httpclient using the wrong method
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 722
diff changeset
278 if(!client->method) {
668
0a7d1e9ca6b8 implement HttpClient function create_req_buffer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 666
diff changeset
279 cxBufferPutString(&buf, "GET ");
0a7d1e9ca6b8 implement HttpClient function create_req_buffer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 666
diff changeset
280 } else {
0a7d1e9ca6b8 implement HttpClient function create_req_buffer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 666
diff changeset
281 cxBufferPutString(&buf, client->method);
726
5ad3bda4aca1 fix httpclient using the wrong method
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 722
diff changeset
282 cxBufferPut(&buf, ' ');
668
0a7d1e9ca6b8 implement HttpClient function create_req_buffer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 666
diff changeset
283 }
0a7d1e9ca6b8 implement HttpClient function create_req_buffer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 666
diff changeset
284 cxBufferPutString(&buf, client->uri ? client->uri : "/");
0a7d1e9ca6b8 implement HttpClient function create_req_buffer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 666
diff changeset
285 cxBufferPutString(&buf, " HTTP/1.1\r\n");
666
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
286
668
0a7d1e9ca6b8 implement HttpClient function create_req_buffer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 666
diff changeset
287 HeaderArray *hdr = client->request_headers;
0a7d1e9ca6b8 implement HttpClient function create_req_buffer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 666
diff changeset
288 while(hdr) {
0a7d1e9ca6b8 implement HttpClient function create_req_buffer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 666
diff changeset
289 for(int i=0;i<hdr->len;i++) {
0a7d1e9ca6b8 implement HttpClient function create_req_buffer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 666
diff changeset
290 cxBufferPutString(&buf, hdr->headers[i].name);
0a7d1e9ca6b8 implement HttpClient function create_req_buffer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 666
diff changeset
291 cxBufferPutString(&buf, ": ");
0a7d1e9ca6b8 implement HttpClient function create_req_buffer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 666
diff changeset
292 cxBufferPutString(&buf, hdr->headers[i].value);
0a7d1e9ca6b8 implement HttpClient function create_req_buffer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 666
diff changeset
293 cxBufferPutString(&buf, "\r\n");
0a7d1e9ca6b8 implement HttpClient function create_req_buffer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 666
diff changeset
294 }
0a7d1e9ca6b8 implement HttpClient function create_req_buffer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 666
diff changeset
295 hdr = hdr->next;
0a7d1e9ca6b8 implement HttpClient function create_req_buffer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 666
diff changeset
296 }
0a7d1e9ca6b8 implement HttpClient function create_req_buffer
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 666
diff changeset
297 cxBufferPutString(&buf, "\r\n");
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
298 client->transfer_buffer = buf.space;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
299 client->transfer_buffer_alloc = buf.capacity;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
300 client->transfer_buffer_len = buf.size;
666
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
301
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
302 return 0;
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
303 }
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
304
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
305 static int client_connected(EventHandler *ev, Event *event) {
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
306 HttpClient *client = event->cookie;
707
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
307 client->last_event = event;
706
df64b4b79912 add error handling in proxy_request_read
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 704
diff changeset
308 if(client->stage < 0) {
df64b4b79912 add error handling in proxy_request_read
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 704
diff changeset
309 return 0;
df64b4b79912 add error handling in proxy_request_read
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 704
diff changeset
310 }
666
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
311 if(create_req_buffer(client)) {
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
312 // TODO: set error
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
313 return 0; // end
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
314 }
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
315 event->fn = client_io;
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
316
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
317 return client_io(ev, event);
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
318 }
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
319
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
320 static int client_io(EventHandler *ev, Event *event) {
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
321 HttpClient *client = event->cookie;
707
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
322 return client_process(client, event);
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
323 }
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
324
717
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
325 static int client_ws_io(EventHandler *ev, Event *event) {
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
326 HttpClient *client = event->cookie;
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
327 return client_ws_process(client, event);
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
328 }
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
329
707
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
330 static int client_process(HttpClient *client, Event *event) {
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
331 client->last_event = event;
706
df64b4b79912 add error handling in proxy_request_read
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 704
diff changeset
332 if(client->stage < 0) {
df64b4b79912 add error handling in proxy_request_read
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 704
diff changeset
333 return 0;
df64b4b79912 add error handling in proxy_request_read
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 704
diff changeset
334 }
730
c190ce0aab98 add some httpclient debug logging
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 726
diff changeset
335 log_ereport(LOG_INFORM, "client_process %d", client->stage);
706
df64b4b79912 add error handling in proxy_request_read
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 704
diff changeset
336
691
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
337 if(client->stage == 0) {
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
338 if(client->transfer_buffer_pos < client->transfer_buffer_len) {
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
339 if(client_send_buf(client)) {
691
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
340 return client->error == 0;
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
341 }
675
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
342 }
691
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
343
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
344 // do we need to send a request body?
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
345 if(client->req_content_length != 0) {
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
346 if(client_send_request_body(client)) {
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
347 return client->error == 0;
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
348 }
675
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
349 }
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
350 }
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
351
672
226bfd584075 minimally working httpclient
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 671
diff changeset
352 // writing complete, switch to read events
707
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
353
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
354 if(client->event.events != EVENT_POLLIN) {
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
355 if(&client->event != event) {
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
356 // The current event, that invoked client_process, is not the
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
357 // actual HttpClient event.
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
358 // Remove the current HttpClient poll event and re-add it later
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
359 // if needed
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
360 if(ev_remove_poll(client->ev, client->socketfd)) {
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
361 client->error = 1;
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
362 return 1;
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
363 }
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
364 client->event.fn = NULL;
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
365 }
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
366 client->event.events = EVENT_POLLIN;
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
367
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
368 }
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
369
691
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
370 client->stage = 1;
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
371
704
778dcf4ad63c fix multiple reads could corrupt the httpclient buffer content
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 703
diff changeset
372 if(client_read_response_header(client)) {
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
373 return client->error == 0;
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
374 }
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
375 int ret = 0;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
376 if(client->stage == 2) {
717
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
377 return client_ws_process(client, event);
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
378 }
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
379 if(client_read_response_body(client)) {
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
380 ret = 1;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
381 }
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
382
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
383 if(ret) {
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
384 return client->error == 0;
672
226bfd584075 minimally working httpclient
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 671
diff changeset
385 }
732
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
386 log_ereport(LOG_INFORM, "client_process end: %p", event->finish);
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
387
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
388 if(event != &client->event) {
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
389 // An extern event has called client_process, therefore the
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
390 // client_finished event callback is not automatically called.
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
391 // Manually shutdown the client:
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
392 log_ereport(LOG_INFORM, "client_process: manual shutdown");
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
393 ev_remove_poll(client->ev, client->socketfd);
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
394 client_finished(client->ev, &client->event);
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
395 client->event.fn = NULL;
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
396 client->event.finish = NULL;
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
397 }
672
226bfd584075 minimally working httpclient
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 671
diff changeset
398
226bfd584075 minimally working httpclient
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 671
diff changeset
399 return 0;
226bfd584075 minimally working httpclient
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 671
diff changeset
400 }
226bfd584075 minimally working httpclient
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 671
diff changeset
401
719
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
402 static int client_ws_write(HttpClient *client) {
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
403 if(!client->ws_write) {
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
404 client->transfer2_buffer_pos = client->transfer2_buffer_len;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
405 return 0;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
406 }
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
407
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
408 int ret = 0;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
409
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
410 // pass all data from the transfer2 buffer to the ws_write callback
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
411 char *out = client->transfer2_buffer + client->transfer2_buffer_pos;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
412 size_t nbytes = client->transfer2_buffer_len - client->transfer2_buffer_pos;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
413 while(nbytes > 0) {
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
414 ssize_t w = client->ws_write(client, out, nbytes, client->ws_write_userdata);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
415 if(w == HTTP_CLIENT_CALLBACK_WOULD_BLOCK) {
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
416 ret = 1;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
417 break;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
418 } else if(w <= 0) {
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
419 client->error = 1;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
420 return 0;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
421 }
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
422 client->transfer2_buffer_pos += w;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
423
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
424 // adjust buffer
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
425 out = client->transfer2_buffer + client->transfer2_buffer_pos;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
426 nbytes = client->transfer2_buffer_len - client->transfer2_buffer_pos;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
427 }
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
428
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
429 // clear buffer
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
430 if(client->transfer2_buffer_pos == client->transfer2_buffer_len) {
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
431 client->transfer2_buffer_pos = 0;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
432 client->transfer2_buffer_len = 0;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
433 }
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
434
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
435 return ret;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
436 }
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
437
717
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
438 static int client_ws_process(HttpClient *client, Event *event) {
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
439 // send available data from the transfer buffer
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
440 int ret = client_send_buf(client);
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
441 if(client->error) {
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
442 return 0;
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
443 }
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
444 // readiness notification
718
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
445 if(ret == 0) {
720
8c7d08d3be2e add support for websockets in the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 719
diff changeset
446 client->transfer_buffer_pos = 0;
8c7d08d3be2e add support for websockets in the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 719
diff changeset
447 client->transfer_buffer_len = 0;
718
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
448 if(client->ws_msg_ready && client->ws_msg_ready(client, client->ws_msg_ready_userdata)) {
717
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
449 return 0;
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
450 }
720
8c7d08d3be2e add support for websockets in the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 719
diff changeset
451
8c7d08d3be2e add support for websockets in the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 719
diff changeset
452 if(client->transfer_buffer_pos < client->transfer_buffer_len) {
8c7d08d3be2e add support for websockets in the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 719
diff changeset
453 // ws_msg_ready has added data to the transfer buffer -> flush buffer again
8c7d08d3be2e add support for websockets in the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 719
diff changeset
454 if(client_send_buf(client) && client->error) {
8c7d08d3be2e add support for websockets in the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 719
diff changeset
455 return 0;
8c7d08d3be2e add support for websockets in the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 719
diff changeset
456 }
8c7d08d3be2e add support for websockets in the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 719
diff changeset
457 }
717
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
458 }
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
459
719
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
460 // flush transfer2 buffer
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
461 if(client_ws_write(client)) {
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
462 return 1;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
463 }
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
464
717
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
465 // read message
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
466 char *buf = client->transfer2_buffer + client->transfer2_buffer_pos;
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
467 size_t available = client->transfer2_buffer_alloc - client->transfer2_buffer_len;
719
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
468
717
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
469 ssize_t r = -1;
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
470 while(available > 0) {
719
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
471 r = net_read(client->stream, buf, available);
717
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
472 if(r <= 0) {
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
473 break;
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
474 }
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
475 client->transfer2_buffer_len += r;
719
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
476 if(client_ws_write(client)) {
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
477 break;
717
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
478 }
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
479
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
480 // adjust buffer
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
481 buf = client->transfer2_buffer + client->transfer2_buffer_pos;
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
482 available = client->transfer2_buffer_alloc - client->transfer2_buffer_len;
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
483 }
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
484
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
485 return r == 0 || client->error ? 0 : 1;
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
486 }
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
487
672
226bfd584075 minimally working httpclient
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 671
diff changeset
488 static int client_finished(EventHandler *ev, Event *event) {
226bfd584075 minimally working httpclient
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 671
diff changeset
489 HttpClient *client = event->cookie;
732
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
490 log_ereport(LOG_INFORM, "client_finished: %p", client->response_finished);
672
226bfd584075 minimally working httpclient
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 671
diff changeset
491
226bfd584075 minimally working httpclient
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 671
diff changeset
492 close(client->socketfd);
226bfd584075 minimally working httpclient
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 671
diff changeset
493 client->socketfd = -1;
707
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
494 client->stage = -1;
672
226bfd584075 minimally working httpclient
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 671
diff changeset
495
226bfd584075 minimally working httpclient
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 671
diff changeset
496 // request finished
226bfd584075 minimally working httpclient
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 671
diff changeset
497 if(client->response_finished) {
226bfd584075 minimally working httpclient
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 671
diff changeset
498 client->response_finished(client, client->response_finished_userdata);
226bfd584075 minimally working httpclient
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 671
diff changeset
499 }
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
500
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
501 return 0;
666
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
502 }
c99e0b352e36 add non-blocking http client connect
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 665
diff changeset
503
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
504 // sends the content of the transfer buffer to client->socketfd
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
505 static int client_send_buf(HttpClient *client) {
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
506 size_t nbytes = client->transfer_buffer_len - client->transfer_buffer_pos;
720
8c7d08d3be2e add support for websockets in the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 719
diff changeset
507 if(nbytes == 0) {
8c7d08d3be2e add support for websockets in the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 719
diff changeset
508 return 0;
8c7d08d3be2e add support for websockets in the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 719
diff changeset
509 }
8c7d08d3be2e add support for websockets in the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 719
diff changeset
510 //printf("reqbuf:\n\n%.*s\n\n", (int)nbytes, client->transfer_buffer + client->transfer_buffer_pos);
675
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
511 ssize_t w;
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
512 while((w = write(client->socketfd, client->transfer_buffer + client->transfer_buffer_pos, nbytes)) > 0) {
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
513 client->transfer_buffer_pos += w;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
514 nbytes = client->transfer_buffer_len - client->transfer_buffer_pos;
675
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
515 if(nbytes == 0) {
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
516 break;
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
517 }
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
518 }
edacba8beedb add support for request bodies with a fixed content length for the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 674
diff changeset
519
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
520 if(w <= 0) {
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
521 if(errno != EAGAIN) {
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
522 // TODO: log correct host
707
5fb102d2c745 fix proxy event handling
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 706
diff changeset
523 log_ereport(LOG_FAILURE, "http-client %s - %s: write failed: %s", "localhost", client->uri, strerror(errno));
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
524 client->error = 1;
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
525 }
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
526 return 1;
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
527 }
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
528
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
529 return client->transfer_buffer_pos < client->transfer_buffer_len;
671
879005903b2b implement basic http client IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 669
diff changeset
530 }
673
144bdc33fdb6 add socket utils
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 672
diff changeset
531
722
12fb6b33418c fix uninitialized memory in the reverse proxy
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 720
diff changeset
532 static int client_send_request_body(HttpClient *client) {
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
533 size_t rbody_readsize = client->transfer_buffer_alloc;
680
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
534 size_t rbody_buf_offset = 0;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
535 if(client->req_content_length == -1) {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
536 // chunked transfer encoding:
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
537 // don't fill req_buffer completely, reserve some space for
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
538 // a chunk header, that will be inserted at the beginning
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
539 rbody_readsize -= 16;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
540 rbody_buf_offset = 16;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
541 }
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
542 while(!client->request_body_complete) {
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
543 ssize_t r = client->request_body_read(client, client->transfer_buffer + rbody_buf_offset, rbody_readsize, client->request_body_read_userdata);
680
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
544 if(r <= 0) {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
545 if(r == HTTP_CLIENT_CALLBACK_WOULD_BLOCK) {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
546 return 1;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
547 } else if(r == 0) {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
548 // EOF
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
549 client->request_body_complete = 1;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
550 break;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
551 } else {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
552 // error
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
553 client->error = 1;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
554 return 1;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
555 }
681
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
556 } else if(client->req_content_length == -1 && r + 32 < rbody_readsize) {
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
557 // is it time to terminate the request body?
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
558 // try read some additional bytes, if it returns 0, we know
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
559 // the request body is complete and we can add the termination chunk
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
560 char *r2buf = client->transfer_buffer + rbody_buf_offset + r;
681
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
561 ssize_t r2 = client->request_body_read(client, r2buf, 32, client->request_body_read_userdata);
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
562 if(r > 0) {
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
563 r += r2;
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
564 } else if(r == 0) {
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
565 memcpy(r2buf, "0\r\n\r\n", 5);
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
566 r += 5;
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
567 client->request_body_complete = 1;
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
568 client->request_body_terminated = 1;
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
569 } else if(r == HTTP_CLIENT_CALLBACK_WOULD_BLOCK) {
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
570 return 1;
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
571 } else {
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
572 client->error = 1;
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
573 return 1;
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
574 }
680
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
575 }
673
144bdc33fdb6 add socket utils
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 672
diff changeset
576
680
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
577 size_t startpos = 0;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
578 if(client->req_content_length == -1) {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
579 char chunkheader[16];
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
580 int chunkheaderlen = snprintf(chunkheader, 16, "%zx\r\n", (size_t)r);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
581 startpos = 16 - chunkheaderlen;
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
582 memcpy(client->transfer_buffer + startpos, chunkheader, chunkheaderlen);
680
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
583 }
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
584
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
585 client->req_contentlength_pos += r;
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
586 client->transfer_buffer_pos = startpos;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
587 client->transfer_buffer_len = rbody_buf_offset + r;
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
588 if(client_send_buf(client)) {
680
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
589 return 1;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
590 }
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
591 }
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
592
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
593 // chunked transfer encoding: terminate
681
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
594 if(client->req_content_length == -1 && !client->request_body_terminated) {
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
595 memcpy(client->transfer_buffer, "0\r\n\r\n", 5);
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
596 client->transfer_buffer_pos = 0;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
597 client->transfer_buffer_len = 5;
681
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
598 client->request_body_terminated = 1;
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
599 if(client_send_buf(client)) {
680
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
600 return 1;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
601 }
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
602
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
603 } else if(client->req_content_length != client->req_contentlength_pos) {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
604 // incomplete request body
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
605 client->error = 1;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
606 return 1;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
607 }
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
608
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
609 return 0;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
610 }
673
144bdc33fdb6 add socket utils
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 672
diff changeset
611
691
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
612 // returns 0 success
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
613 // 1 would block or error
683
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
614 static int client_read_response_header(HttpClient *client) {
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
615 if(client->response_header_complete) {
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
616 return 0;
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
617 }
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
618
703
395c62fac7e5 add more httpclient tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 702
diff changeset
619 unsigned char *buffer = client->buffer.inbuf + client->buffer.cursize;
683
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
620 size_t nbytes = client->buffer.maxsize - client->buffer.cursize;
704
778dcf4ad63c fix multiple reads could corrupt the httpclient buffer content
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 703
diff changeset
621
683
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
622 ssize_t r;
704
778dcf4ad63c fix multiple reads could corrupt the httpclient buffer content
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 703
diff changeset
623 while((r = read(client->socketfd, client->buffer.inbuf + client->buffer.cursize, client->buffer.maxsize - client->buffer.cursize)) > 0) {
683
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
624 client->buffer.cursize += r;
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
625 if(!client->response_header_complete) {
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
626 switch(http_parser_process(client->parser)) {
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
627 case 0: { // finish
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
628 if(!http_parser_validate(client->parser)) {
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
629 client->error = 1;
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
630 return 1;
683
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
631 }
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
632 client->statuscode = client->parser->status;
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
633
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
634 client->response_header_complete = 1;
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
635 if(client->response_start) {
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
636 cxmutstr msg = client->parser->msg;
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
637 char t = msg.ptr[msg.length];
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
638 msg.ptr[msg.length] = 0;
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
639 int ret = client->response_start(client, client->statuscode, msg.ptr, client->response_start_userdata);
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
640 msg.ptr[msg.length] = t;
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
641
691
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
642 if(ret != 0) {
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
643 if(ret != HTTP_CLIENT_CALLBACK_WOULD_BLOCK) {
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
644 client->error = 1;
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
645 }
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
646 return 1;
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
647 }
683
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
648 }
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
649 break;
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
650 }
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
651 case 1: { // need more data
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
652 continue;
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
653 }
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
654 case 2: { // error
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
655 client->error = 1;
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
656 return 1;
683
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
657 }
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
658 }
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
659 }
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
660
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
661 // header complete
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
662 break;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
663 }
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
664
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
665 if(r <= 0) {
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
666 if(r == 0) {
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
667 // unexpected EOF
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
668 client->error = 1;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
669 } else if(errno != EAGAIN) {
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
670 log_ereport(LOG_FAILURE, "http-client: IO error: %s", strerror(errno));
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
671 client->error = 1;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
672 }
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
673 return 1;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
674 }
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
675
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
676 // initialize httpstream
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
677 HeaderArray *headers = client->parser->headers;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
678 long long contentlength = 0;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
679 int chunkedtransferenc = 0;
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
680 cxmutstr hdr_connection = CX_NULLSTR;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
681 cxmutstr hdr_upgrade = CX_NULLSTR;
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
682 while(headers) {
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
683 for(int i=0;i<headers->len;i++) {
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
684 cxmutstr header = headers->headers[i].name;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
685 cxmutstr hvalue = headers->headers[i].value;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
686 if(!cx_strcasecmp(header, "content-length")) {
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
687 if(!cx_strtoll(hvalue, &contentlength, 10)) {
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
688 headers = NULL;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
689 break;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
690 }
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
691 } else if(!cx_strcasecmp(header, "transfer-encoding")) {
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
692 if(!cx_strcmp(hvalue, "chunked")) {
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
693 chunkedtransferenc = 1;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
694 headers = NULL;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
695 break;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
696 }
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
697 } else if(!cx_strcasecmp(header, "connection")) {
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
698 hdr_connection = hvalue;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
699 }
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
700
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
701 if(client->statuscode == 101 && !cx_strcasecmp(header, "upgrade")) {
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
702 hdr_upgrade = hvalue;
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
703 }
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
704 }
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
705
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
706 if(headers) {
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
707 headers = headers->next;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
708 }
683
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
709 }
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
710
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
711 if(client->statuscode == 101) {
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
712 if(!cx_strcasecmp(hdr_upgrade, "websocket") && !cx_strcasecmp(hdr_connection, "upgrade")) {
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
713 client->stage = 2;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
714 client->event.events = EVENT_POLLIN|EVENT_POLLOUT;
717
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
715 client->event.fn = client_ws_io;
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
716
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
717 // prepare IO buffers for websockets
717
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
718 // transfer_buffer is used for outgoing traffic
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
719 client->transfer_buffer_len = 0;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
720 client->transfer_buffer_pos = 0;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
721
717
2edcb361b8be implement httpclient websocket IO
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 715
diff changeset
722 // transfer2_buffer is used for reading
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
723 client->transfer2_buffer = malloc(HTTP_CLIENT_BUFFER_SIZE);
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
724 if(!client->transfer2_buffer) {
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
725 client->error = 1;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
726 return 1;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
727 }
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
728 client->transfer2_buffer_alloc = HTTP_CLIENT_BUFFER_SIZE;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
729 client->transfer2_buffer_len = 0;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
730 client->transfer2_buffer_pos = 0;
715
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
731
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
732 IOStream *fd = Sysstream_new(NULL, client->socketfd);
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
733 if(!fd) {
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
734 client->error = 1;
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
735 return 1;
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
736 }
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
737 HttpStream *http = (HttpStream*)httpstream_new(NULL, fd);
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
738 if(!http) {
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
739 fd->free(fd);
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
740 }
719
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
741 http->raw = WS_TRUE;
715
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
742 if(client->buffer.pos < client->buffer.cursize) {
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
743 // bytes remaining in the buffer -> enable buffered reading
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
744 httpstream_enable_buffered_read(
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
745 &http->st,
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
746 (char*)client->buffer.inbuf,
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
747 client->buffer.maxsize,
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
748 &client->buffer.cursize,
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
749 &client->buffer.pos);
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
750 }
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
751 client->stream = http;
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
752 } else {
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
753 // error: unknown protocol
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
754 log_ereport(LOG_FAILURE, "http-client: unknown protocol upgrade: %.*s", (int)hdr_upgrade.length, hdr_upgrade.ptr);
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
755 client->error = 1;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
756 return 1;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
757 }
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
758 } else if(contentlength > 0 || chunkedtransferenc) {
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
759 IOStream *fd = Sysstream_new(NULL, client->socketfd);
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
760 if(!fd) {
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
761 client->error = 1;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
762 return 1;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
763 }
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
764 HttpStream *http = (HttpStream*)httpstream_new(NULL, fd);
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
765 if(!http) {
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
766 fd->free(fd);
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
767 }
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
768
715
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
769 // we can reuse the already allocated transfer_buffer for transfer2
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
770 client->transfer2_buffer = client->transfer_buffer;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
771 client->transfer2_buffer_alloc = client->transfer_buffer_alloc;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
772 client->transfer2_buffer_len = 0;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
773 client->transfer_buffer_pos = 0;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
774 client->transfer_buffer = NULL;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
775 client->transfer_buffer_alloc = 0;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
776
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
777 if(contentlength > 0) {
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
778 http->max_read = contentlength;
715
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
779 httpstream_enable_buffered_read(
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
780 &http->st,
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
781 (char*)client->buffer.inbuf,
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
782 client->buffer.maxsize,
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
783 &client->buffer.cursize,
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
784 &client->buffer.pos);
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
785 } else if(chunkedtransferenc) {
715
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
786 httpstream_enable_chunked_read(
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
787 &http->st,
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
788 (char*)client->buffer.inbuf,
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
789 client->buffer.maxsize,
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
790 &client->buffer.cursize,
30732c5b292c httpclient: prepare httpstream for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 708
diff changeset
791 &client->buffer.pos);
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
792 }
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
793 client->stream = http;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
794 }
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
795
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
796 return 0;
683
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
797 }
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
798
691
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
799 // uses the response_body_write callback to write the content of the
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
800 // transfer buffer
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
801 // returns 0 success
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
802 // 1 would block or error
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
803 static int client_write_response(HttpClient *client) {
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
804 while(client->transfer2_buffer_pos < client->transfer2_buffer_len) {
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
805 char *buf = client->transfer2_buffer + client->transfer2_buffer_pos;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
806 size_t len = client->transfer2_buffer_len - client->transfer2_buffer_pos;
691
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
807 int ret = client->response_body_write(client, buf, len, client->response_body_write_userdata);
730
c190ce0aab98 add some httpclient debug logging
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 726
diff changeset
808 log_ereport(LOG_INFORM, "client_write_response: %d bytes; ret: %d", (int)len, ret);
691
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
809 if(ret > 0) {
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
810 client->transfer2_buffer_pos += ret;
691
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
811 } else if(ret == 0) {
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
812 // EOF?
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
813 // check if the write is incomplete, which would be an error
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
814 client->error == client->transfer2_buffer_pos < client->transfer2_buffer_len;
691
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
815 return client->error;
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
816 } else {
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
817 if(ret != HTTP_CLIENT_CALLBACK_WOULD_BLOCK) {
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
818 client->error = 1;
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
819 }
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
820 return 1;
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
821 }
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
822 }
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
823 return 0;
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
824 }
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
825
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
826 // returns 0 success
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
827 // 1 would block or error
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
828 static int client_read_response_body(HttpClient *client) {
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
829 if(!client->stream) {
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
830 return 0; // no input stream -> no response body
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
831 }
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
832
691
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
833 // does the transfer buffer still contains bytes, that should be written?
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
834 if(client_write_response(client)) {
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
835 return 1;
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
836 }
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
837
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
838 char *buf = client->transfer2_buffer;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
839 size_t nbytes = client->transfer2_buffer_alloc;
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
840
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
841 ssize_t r;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
842 while((r = net_read(&client->stream->st, buf, nbytes)) > 0) {
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
843 client->transfer2_buffer_len = r;
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
844 client->transfer2_buffer_pos = 0;
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
845 if(client->response_body_write) {
691
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
846 if(client_write_response(client)) {
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
847 return 1;
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
848 }
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
849 }
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
850 }
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
851
732
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
852 if(r == 0) {
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
853 log_ereport(LOG_INFORM, "client_read_response_body eof");
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
854 }
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
855
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
856 if(r < 0) {
732
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
857 if(client->stream->st.io_errno == EWOULDBLOCK) {
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
858 log_ereport(LOG_INFORM, "client_read_response_body would block");
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
859 }
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
860
688
02360295ec58 fix chunkheader parser not detecting EOF correctly in some cases
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 687
diff changeset
861 if(client->stream->st.io_errno != EWOULDBLOCK) {
02360295ec58 fix chunkheader parser not detecting EOF correctly in some cases
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 687
diff changeset
862 client->error = 1;
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
863 }
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
864 return 1;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
865 }
732
2aeaf6f32861 fix httpclient termination in case an external event has called the last http_client_process
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 730
diff changeset
866 log_ereport(LOG_INFORM, "client_read_response_body finished");
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
867 return 0;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
868 }
683
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
869
673
144bdc33fdb6 add socket utils
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 672
diff changeset
870 /* --------------------------------- Tests --------------------------------- */
144bdc33fdb6 add socket utils
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 672
diff changeset
871
144bdc33fdb6 add socket utils
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 672
diff changeset
872 static CX_TEST(test_http_client_send_request) {
144bdc33fdb6 add socket utils
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 672
diff changeset
873 CX_TEST_DO {
674
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
874 EventHandler dummy;
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
875 HttpClient *client = http_client_new(&dummy);
673
144bdc33fdb6 add socket utils
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 672
diff changeset
876
674
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
877 int fds[2];
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
878 util_socketpair(fds);
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
879 util_socket_setnonblock(fds[0], 1);
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
880 util_socket_setnonblock(fds[1], 1);
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
881 client->socketfd = fds[0];
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
882 int sock = fds[1];
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
883
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
884 // create a large test buffer, that is bigger than the socket buffer
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
885 // 32mb should be enough
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
886 size_t len = 32*1024*1024;
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
887 char *str = malloc(len);
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
888 // init the buffer with random data
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
889 for(size_t i=0;i<len;i+=sizeof(int)) {
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
890 int *p = (int*)(str+i);
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
891 *p = rand();
674
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
892 }
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
893
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
894 client->transfer_buffer = str;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
895 client->transfer_buffer_len = len;
674
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
896
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
897 // test client_send_request
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
898
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
899 int ret = client_send_buf(client);
674
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
900 // It is very likely that the first client_send_request call doesn't
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
901 // fully write the request buffer to the socket
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
902 // In that case it returns 1 but without the error flag
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
903 CX_TEST_ASSERT(ret == 1 && !client->error);
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
904 CX_TEST_ASSERT(client->transfer_buffer_pos > 0);
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
905 CX_TEST_ASSERT(client->transfer_buffer_pos < len);
674
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
906
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
907 // read the request buffer from sock and continue with client_send_request
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
908 CxBuffer buf;
676
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
909 cxBufferInit(&buf, cxDefaultAllocator, NULL, len, CX_BUFFER_AUTO_EXTEND|CX_BUFFER_FREE_CONTENTS);
674
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
910 char tmpbuf[1024];
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
911 int writes = 1;
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
912 while(client->transfer_buffer_pos < client->transfer_buffer_len && writes < 2000000) {
674
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
913 ssize_t r = read(sock, tmpbuf, 1024);
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
914 CX_TEST_ASSERT(r >= 0);
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
915 cxBufferWrite(tmpbuf, 1, r, &buf);
698
fea7c3d74cc6 prepare httpclient for websockets
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 694
diff changeset
916 ret = client_send_buf(client);
674
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
917 CX_TEST_ASSERT(ret == 0 || (ret == 1 && !client->error));
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
918
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
919 writes++;
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
920 }
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
921 CX_TEST_ASSERT(client->transfer_buffer_pos == client->transfer_buffer_len);
674
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
922
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
923 // finish reading the request buffer from sock
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
924 ssize_t r;
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
925 while((r = read(sock, tmpbuf, 1024)) > 0 && writes < 2000000) {
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
926 cxBufferWrite(tmpbuf, 1, r, &buf);
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
927 writes++;
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
928 }
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
929
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
930 CX_TEST_ASSERT(buf.size == len);
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
931 CX_TEST_ASSERT(!memcmp(str, buf.space, len));
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
932
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
933 // cleanup
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
934 close(fds[0]);
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
935 close(fds[1]);
6a031133a498 add http_client_send_request test
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 673
diff changeset
936 http_client_free(client);
676
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
937 cxBufferDestroy(&buf);
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
938 }
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
939 }
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
940
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
941 typedef struct TestResponse {
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
942 int status;
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
943 char *msg;
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
944 CxBuffer *response;
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
945 int error_interval;
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
946 int error_test;
676
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
947 } TestResponse;
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
948
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
949 static int test_response_start(HttpClient *client, int status, char *msg, void *userdata) {
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
950 TestResponse *test = userdata;
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
951 test->status = status;
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
952 test->msg = strdup(msg);
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
953 return 0;
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
954 }
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
955
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
956 static ssize_t test_response_body_write(HttpClient *client, void *buf, size_t size, void *userdata) {
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
957 TestResponse *test = userdata;
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
958 if(test->error_interval > 0 && test->error_test >= test->error_interval) {
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
959 test->error_test = 0;
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
960 return HTTP_CLIENT_CALLBACK_WOULD_BLOCK;
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
961 }
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
962 test->error_test++;
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
963
676
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
964 cxBufferWrite(buf, 1, size, test->response);
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
965 return size;
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
966 }
d43f1dd8b18e add test_http_client_io_simple
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 675
diff changeset
967
680
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
968
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
969 typedef struct TestRequestBody {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
970 char *content;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
971 size_t length;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
972 size_t pos;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
973 int chunksize;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
974 int max_reads; // max number of reads until test_request_body_read returns 0
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
975 int cur_reads; // current number of read-attempts
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
976 } TestRequestBody;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
977
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
978 static ssize_t test_request_body_read(HttpClient *client, void *buf, size_t size, void *userdata) {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
979 TestRequestBody *req = userdata;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
980 req->cur_reads++;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
981 if(req->chunksize == 0 || req->cur_reads > req->max_reads) {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
982 return -1;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
983 }
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
984 size_t max = req->length - req->pos;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
985 if(max == 0) {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
986 return 0;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
987 }
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
988
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
989 size_t sz = req->chunksize > size ? size : req->chunksize;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
990 if(sz > max) {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
991 sz = max;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
992 }
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
993 memcpy(buf, req->content + req->pos, sz);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
994 req->pos += sz;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
995 return sz;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
996 }
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
997
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
998 static CX_TEST(test_http_client_send_request_body_chunked) {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
999 CX_TEST_DO {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1000 EventHandler dummy;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1001 HttpClient *client = http_client_new(&dummy);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1002 create_req_buffer(client);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1003 client->req_content_length = -1;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1004
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1005 int fds[2];
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1006 util_socketpair(fds);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1007 util_socket_setnonblock(fds[0], 1);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1008 util_socket_setnonblock(fds[1], 1);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1009 client->socketfd = fds[0];
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1010 int sock = fds[1];
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1011
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1012 // response buffer
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1013 CxBuffer buf;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1014 cxBufferInit(&buf, cxDefaultAllocator, NULL, 1024, CX_BUFFER_AUTO_EXTEND|CX_BUFFER_FREE_CONTENTS);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1015
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1016 // test
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1017 char request_body[1024];
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1018 memset(request_body, 'x', 1024);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1019 memset(request_body+128, 'y', 128);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1020 memset(request_body+384, 'z', 128);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1021 memset(request_body+640, ':', 128);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1022 memset(request_body+896, '!', 128);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1023
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1024 TestRequestBody req;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1025 req.content = request_body;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1026 req.length = 1024;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1027 req.pos = 0;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1028 req.chunksize = 16;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1029 req.max_reads = 8;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1030 req.cur_reads = 0;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1031 client->request_body_read = test_request_body_read;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1032 client->request_body_read_userdata = &req;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1033
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
1034 memset(client->transfer_buffer, '_', client->transfer_buffer_alloc);
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
1035 client->transfer_buffer_pos = 0;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
1036 client->transfer_buffer_len = 0;
680
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1037
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1038 // send the first 128 bytes
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1039 while(req.cur_reads <= req.max_reads) {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1040 int ret = client_send_request_body(client);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1041 CX_TEST_ASSERT(ret == 1);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1042 CX_TEST_ASSERT(!client->error);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1043 char buf2[1024];
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1044 ssize_t r = read(sock, buf2, 1024);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1045 if(r > 0) {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1046 cxBufferWrite(buf2, 1, r, &buf);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1047 }
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1048 }
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1049
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1050 // because we are using chunked transfer encoding, the result buffer
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1051 // (buf) should contain more than 128 bytes (additional chunk headers)
681
e9705d51866a optimize chunked transfer encoding termination in client_send_request_body
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 680
diff changeset
1052 CX_TEST_ASSERT(buf.pos > 128);
680
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1053
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1054 // change chunk size to 128
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1055 req.max_reads = 9999;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1056 req.chunksize = 128;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1057 while(req.cur_reads <= req.max_reads) {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1058 int ret = client_send_request_body(client);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1059 CX_TEST_ASSERT(!client->error);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1060 char buf2[2048];
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1061 ssize_t r;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1062 while((r = read(sock, buf2, 2048)) > 0) {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1063 cxBufferWrite(buf2, 1, r, &buf);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1064 }
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1065 if(ret == 0) {
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1066 break;
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1067 }
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1068 }
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1069 CX_TEST_ASSERT(req.cur_reads < req.max_reads);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1070
682
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1071 // verify chunks
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1072 char test_request_body[1024];
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1073 memset(test_request_body, 0, 1024);
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1074
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1075 int pos = 0;
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1076 int chunklen = 0;
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1077 char *str = buf.space;
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1078 while(str < buf.space + buf.size) {
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1079 cxstring chunkheader = cx_strn(str, 2);
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1080 if(!cx_strcmp(chunkheader, "0\r")) {
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1081 chunkheader.length = 1;
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1082 }
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1083 int ret = cx_strtoi(chunkheader, &chunklen, 16);
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1084 CX_TEST_ASSERT(ret == 0);
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1085 if(chunklen == 0) {
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1086 break;
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1087 }
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1088
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1089 char *data = str + 4;
683
db37761a8494 enable buffered reader for request bodies with fixed content-length
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 682
diff changeset
1090 CX_TEST_ASSERT(data + chunklen < buf.space + buf.size);
682
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1091 memcpy(test_request_body + pos, data, chunklen);
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1092 pos += chunklen;
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1093 str = data + chunklen;
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1094 }
f4c593a99266 verify chunks in test_http_client_send_request_body_chunked
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 681
diff changeset
1095 CX_TEST_ASSERT(!memcmp(request_body, test_request_body, 1024));
680
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1096
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1097 // cleanup
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1098 close(fds[0]);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1099 close(fds[1]);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1100 http_client_free(client);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1101 cxBufferDestroy(&buf);
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1102 }
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1103 }
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1104
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1105 static CX_TEST_SUBROUTINE(test_read_response, cxstring response_str, CxBuffer *response_body) {
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1106 EventHandler dummy;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1107 HttpClient *client = http_client_new(&dummy);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1108 create_req_buffer(client);
691
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
1109 client->transfer_buffer_pos = 0;
4d8a55a7618b handle httpclient response write errors
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 690
diff changeset
1110 client->transfer_buffer_len = 0;
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1111 client->req_content_length = -1;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1112
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1113 int fds[2];
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1114 util_socketpair(fds);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1115 util_socket_setnonblock(fds[0], 1);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1116 util_socket_setnonblock(fds[1], 1);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1117 client->socketfd = fds[0];
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1118 int sock = fds[1];
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1119
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1120 TestResponse testr = { 0 };
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1121 testr.response = response_body;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1122 client->response_body_write = test_response_body_write;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1123 client->response_body_write_userdata = &testr;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1124
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1125 // test
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1126
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1127 size_t response_pos = 0;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1128 while(response_pos < response_str.length) {
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1129 size_t nbytes = response_str.length - response_pos;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1130 ssize_t w = write(sock, response_str.ptr + response_pos, nbytes);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1131 if(w > 0) {
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1132 response_pos += w;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1133 }
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1134
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1135 if(!client->response_header_complete) {
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1136 int ret = client_read_response_header(client);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1137 CX_TEST_ASSERT(client->error == 0);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1138 if(ret == 1) {
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1139 continue;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1140 }
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1141 }
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1142
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1143 if(response_body != NULL) {
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1144 CX_TEST_ASSERT(client->stream != NULL);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1145
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1146 int ret = client_read_response_body(client);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1147 CX_TEST_ASSERT(client->error == 0);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1148 if(ret == 1) {
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1149 continue;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1150 }
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1151 } else {
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1152 CX_TEST_ASSERT(client->stream == NULL);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1153 }
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1154
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1155 break;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1156 }
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1157
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1158 // cleanup
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1159 close(fds[0]);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1160 close(fds[1]);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1161 http_client_free(client);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1162 }
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1163
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
1164 static CX_TEST(test_http_client_read_response_head) {
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
1165 CX_TEST_DO {
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
1166 char *response_str =
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
1167 "HTTP/1.1 204 OK\r\n"
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
1168 "Host: localhost\r\n"
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
1169 "Content-length: 0\r\n"
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
1170 "\r\n";
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
1171
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1172 CX_TEST_CALL_SUBROUTINE(test_read_response, cx_str(response_str), NULL);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1173
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1174 response_str =
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1175 "HTTP/1.1 204 OK\r\n"
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1176 "Host: localhost\r\n"
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1177 "\r\n";
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1178
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1179 CX_TEST_CALL_SUBROUTINE(test_read_response, cx_str(response_str), NULL);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1180 }
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1181 }
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1182
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1183 static CX_TEST(test_http_client_read_response_ctlen) {
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1184 CX_TEST_DO {
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1185 char *response_str =
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1186 "HTTP/1.1 200 OK\r\n"
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1187 "Host: localhost\r\n"
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1188 "Content-length: 13\r\n"
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1189 "\r\n"
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1190 "Hello World!\n";
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1191 CxBuffer *buf = cxBufferCreate(NULL, NULL, 1024, CX_BUFFER_AUTO_EXTEND|CX_BUFFER_FREE_CONTENTS);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1192
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1193 CX_TEST_CALL_SUBROUTINE(test_read_response, cx_str(response_str), buf);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1194 CX_TEST_ASSERT(buf->size == 13);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1195 CX_TEST_ASSERT(!cx_strcmp(cx_strn(buf->space, buf->size), "Hello World!\n"));
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1196
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1197 cxBufferFree(buf);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1198 }
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1199 }
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1200
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1201 static CX_TEST(test_http_client_read_response_ctlen_big) {
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1202 CX_TEST_DO {
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1203 // create response body
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1204 size_t len = 1024*1024*32;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1205 char *response_str = malloc(len);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1206 char *str = response_str;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1207 for(size_t i=0;i<len;i+=sizeof(int)) {
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1208 int *p = (int*)(str+i);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1209 *p = rand();
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1210 }
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1211 cxstring body = cx_strn(response_str, len);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1212
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1213 // create request string
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1214 CxBuffer *req = cxBufferCreate(NULL, NULL, 1024, CX_BUFFER_AUTO_EXTEND|CX_BUFFER_FREE_CONTENTS);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1215 cxBufferPutString(req,
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1216 "HTTP/1.1 200 OK\r\n"
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1217 "Host: localhost\r\n"
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1218 "Content-length: ");
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1219 char ctlen[32];
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1220 snprintf(ctlen, 32, "%d\r\n\r\n", (int)len);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1221 cxBufferPutString(req, ctlen);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1222 cxBufferPutString(req, body);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1223 cxBufferTerminate(req);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1224
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1225 // response buffer
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1226 CxBuffer *buf = cxBufferCreate(NULL, NULL, 1024, CX_BUFFER_AUTO_EXTEND|CX_BUFFER_FREE_CONTENTS);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1227
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1228 // test
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1229 CX_TEST_CALL_SUBROUTINE(test_read_response, cx_strn(req->space, req->size), buf);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1230 CX_TEST_ASSERT(buf->size == len);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1231 CX_TEST_ASSERT(!cx_strcmp(cx_strn(buf->space, buf->size), body));
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1232
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1233 cxBufferFree(req);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1234 cxBufferFree(buf);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1235 }
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1236 }
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1237
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1238 static CX_TEST_SUBROUTINE(test_http_client_io, cxstring response, int status_code, const char *msg, CxBuffer *out_buf, size_t write_blocksz, int error_interval) {
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1239 EventHandler dummy;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1240 HttpClient *client = http_client_new(&dummy);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1241
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1242 int fds[2];
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1243 util_socketpair(fds);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1244 util_socket_setnonblock(fds[0], 1);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1245 util_socket_setnonblock(fds[1], 1);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1246 client->socketfd = fds[0];
708
027b16665f13 adjust test_http_client_io subroutine
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 707
diff changeset
1247 client->event.cookie = client;
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1248 int sock = fds[1];
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1249
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1250 // setup client
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1251 http_client_set_uri(client, "/test/uri/");
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1252 http_client_set_method(client, "GET");
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1253 http_client_add_request_header(client, cx_mutstr("Host"), cx_mutstr("localhost"));
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1254 http_client_add_request_header(client, cx_mutstr("Test1"), cx_mutstr("value1"));
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1255 http_client_add_request_header(client, cx_mutstr("Test2"), cx_mutstr("value2"));
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1256 create_req_buffer(client);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1257
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1258 size_t req_header_len = client->transfer_buffer_len;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1259
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1260 TestResponse testr = { 0 };
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1261 testr.response = out_buf;
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1262 testr.error_interval = error_interval;
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1263 client->response_start = test_response_start;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1264 client->response_start_userdata = &testr;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1265 client->response_body_write = test_response_body_write;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1266 client->response_body_write_userdata = &testr;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1267
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1268 // test IO
708
027b16665f13 adjust test_http_client_io subroutine
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 707
diff changeset
1269 int ret = client_io(&dummy, &client->event);
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1270 CX_TEST_ASSERT(!client->error);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1271 CX_TEST_ASSERT(ret == 1);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1272
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1273 // do IO and read request until the header is processed
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1274 size_t req_header_pos = 0;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1275 char req_buf[4];
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1276 while(req_header_pos < req_header_len) {
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1277 ssize_t r = read(sock, req_buf, 4);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1278 if(r == 0) {
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
1279 break;
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
1280 }
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1281 CX_TEST_ASSERT(r > 0);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1282 req_header_pos += r;
708
027b16665f13 adjust test_http_client_io subroutine
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 707
diff changeset
1283 ret = client_io(&dummy, &client->event);
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1284 CX_TEST_ASSERT(!client->error);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1285 CX_TEST_ASSERT(ret == 1);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1286 }
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1287 CX_TEST_ASSERT(req_header_pos == req_header_len);
690
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1288
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1289 const char *response_str = response.ptr;
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1290 size_t response_str_len = response.length;
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1291 size_t response_str_pos = 0;
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1292
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1293 // send response and do IO
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1294 int in_progress = 1;
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1295 while(in_progress) {
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1296 size_t len = response_str_len - response_str_pos;
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1297 if(len > 0) {
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1298 if(len > write_blocksz) {
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1299 len = write_blocksz;
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1300 }
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1301 ssize_t w = write(sock, response_str + response_str_pos, len);
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1302 if(w == 0) {
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1303 break;
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1304 }
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1305 CX_TEST_ASSERT(w > 0);
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1306 response_str_pos += w;
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1307 }
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1308
708
027b16665f13 adjust test_http_client_io subroutine
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 707
diff changeset
1309 ret = client_io(&dummy, &client->event);
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1310 if(ret == 0) {
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1311 in_progress = 0;
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1312 }
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1313
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1314 CX_TEST_ASSERT(!client->error);
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
1315 }
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1316 CX_TEST_ASSERT(response_str_pos == response_str_len);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1317 CX_TEST_ASSERT(testr.status == status_code);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1318 CX_TEST_ASSERT(testr.msg);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1319 CX_TEST_ASSERT(!strcmp(testr.msg, msg));
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1320
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1321 // cleanup
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1322 free(testr.msg);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1323 close(fds[0]);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1324 close(fds[1]);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1325 http_client_free(client);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1326 }
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1327
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1328 static CX_TEST_SUBROUTINE(test_http_client_io_simple, size_t blocksz, int error_interval) {
690
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1329 cxstring response_str = cx_str(
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1330 "HTTP/1.1 200 OK\r\n"
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1331 "Host: localhost\r\n"
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1332 "Content-length: 13\r\n"
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1333 "\r\n"
690
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1334 "Hello World!\n");
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1335 CxBuffer *buf = cxBufferCreate(NULL, NULL, 1024, CX_BUFFER_AUTO_EXTEND|CX_BUFFER_FREE_CONTENTS);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1336
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1337 CX_TEST_CALL_SUBROUTINE(test_http_client_io, response_str, 200, "OK", buf, blocksz, error_interval);
687
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1338 CX_TEST_ASSERT(buf->size == 13);
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1339 CX_TEST_ASSERT(!cx_strcmp(cx_strn(buf->space, buf->size), "Hello World!\n"));
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1340
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1341 cxBufferFree(buf);
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
1342 }
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
1343
687
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1344 static CX_TEST(test_http_client_io_simple_1b) {
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1345 CX_TEST_DO {
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1346 CX_TEST_CALL_SUBROUTINE(test_http_client_io_simple, 1, 0);
687
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1347 }
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1348 }
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1349
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1350 static CX_TEST(test_http_client_io_simple_2b) {
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1351 CX_TEST_DO {
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1352 CX_TEST_CALL_SUBROUTINE(test_http_client_io_simple, 2, 0);
687
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1353 }
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1354 }
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1355
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1356 static CX_TEST(test_http_client_io_simple_3b) {
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1357 CX_TEST_DO {
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1358 CX_TEST_CALL_SUBROUTINE(test_http_client_io_simple, 3, 0);
687
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1359 }
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1360 }
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1361
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1362 static CX_TEST(test_http_client_io_simple_16b) {
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1363 CX_TEST_DO {
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1364 CX_TEST_CALL_SUBROUTINE(test_http_client_io_simple, 16, 0);
687
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1365 }
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1366 }
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1367
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1368 static CX_TEST(test_http_client_io_simple_512b) {
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1369 CX_TEST_DO {
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1370 CX_TEST_CALL_SUBROUTINE(test_http_client_io_simple, 512, 0);
687
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1371 }
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1372 }
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1373
690
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1374 static CX_TEST_SUBROUTINE(test_http_client_io_large, size_t blocksz) {
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1375 int ctlen = 1024*1024*32;
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1376 CxBuffer *resp = cxBufferCreate(NULL, NULL, ctlen + 128, CX_BUFFER_AUTO_EXTEND|CX_BUFFER_FREE_CONTENTS);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1377 cx_bprintf(resp,
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1378 "HTTP/1.1 200 OK\r\n"
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1379 "Host: localhost\r\n"
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1380 "Content-length: %d\r\n"
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1381 "\r\n",
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1382 ctlen);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1383 size_t content_start = resp->pos;
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1384 CX_TEST_ASSERT(resp->pos + ctlen < resp->capacity);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1385 memset(resp->space + resp->pos, 'x', ctlen);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1386 resp->pos += ctlen;
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1387 resp->size += ctlen;
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1388
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1389 CxBuffer *buf = cxBufferCreate(NULL, NULL, 1024, CX_BUFFER_AUTO_EXTEND|CX_BUFFER_FREE_CONTENTS);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1390
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1391 CX_TEST_CALL_SUBROUTINE(test_http_client_io, cx_strn(resp->space, resp->size), 200, "OK", buf, blocksz, 0);
690
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1392 CX_TEST_ASSERT(buf->size == ctlen);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1393 CX_TEST_ASSERT(!memcmp(buf->space, resp->space + content_start, ctlen));
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1394
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1395 cxBufferFree(resp);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1396 cxBufferFree(buf);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1397 }
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1398
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1399 static CX_TEST(test_http_client_io_large_128b) {
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1400 CX_TEST_DO {
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1401 CX_TEST_CALL_SUBROUTINE(test_http_client_io_large, 128);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1402 }
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1403 }
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1404
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1405 static CX_TEST(test_http_client_io_large_4096b) {
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1406 CX_TEST_DO {
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1407 CX_TEST_CALL_SUBROUTINE(test_http_client_io_large, 1024);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1408 }
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1409 }
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1410
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1411 static CX_TEST_SUBROUTINE(test_http_client_io_chunked_transfer, size_t blocksz, int error_interval) {
690
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1412 cxstring response_str = cx_str(
687
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1413 "HTTP/1.1 200 OK\r\n"
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1414 "Host: localhost\r\n"
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1415 "Transfer-encoding: chunked\r\n"
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1416 "\r\n"
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1417 "d\r\n"
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1418 "Hello World!\n"
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1419 "\r\n"
690
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1420 "0\r\n\r\n");
687
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1421 CxBuffer *buf = cxBufferCreate(NULL, NULL, 1024, CX_BUFFER_AUTO_EXTEND|CX_BUFFER_FREE_CONTENTS);
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1422
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1423 CX_TEST_CALL_SUBROUTINE(test_http_client_io, response_str, 200, "OK", buf, blocksz, error_interval);
687
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1424 CX_TEST_ASSERT(buf->size == 13);
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1425 CX_TEST_ASSERT(!cx_strcmp(cx_strn(buf->space, buf->size), "Hello World!\n"));
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1426
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1427 cxBufferFree(buf);
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1428 }
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1429
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1430 static CX_TEST(test_http_client_io_chunked_transfer_1b) {
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1431 CX_TEST_DO {
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1432 CX_TEST_CALL_SUBROUTINE(test_http_client_io_chunked_transfer, 1, 0);
687
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1433 }
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1434 }
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1435
688
02360295ec58 fix chunkheader parser not detecting EOF correctly in some cases
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 687
diff changeset
1436 static CX_TEST(test_http_client_io_chunked_transfer_2b) {
02360295ec58 fix chunkheader parser not detecting EOF correctly in some cases
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 687
diff changeset
1437 CX_TEST_DO {
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1438 CX_TEST_CALL_SUBROUTINE(test_http_client_io_chunked_transfer, 2, 0);
688
02360295ec58 fix chunkheader parser not detecting EOF correctly in some cases
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 687
diff changeset
1439 }
02360295ec58 fix chunkheader parser not detecting EOF correctly in some cases
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 687
diff changeset
1440 }
02360295ec58 fix chunkheader parser not detecting EOF correctly in some cases
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 687
diff changeset
1441
687
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1442 static CX_TEST(test_http_client_io_chunked_transfer_8b) {
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1443 CX_TEST_DO {
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1444 CX_TEST_CALL_SUBROUTINE(test_http_client_io_chunked_transfer, 16, 0);
687
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1445 }
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1446 }
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1447
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1448 static CX_TEST(test_http_client_io_chunked_transfer_64b) {
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1449 CX_TEST_DO {
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1450 CX_TEST_CALL_SUBROUTINE(test_http_client_io_chunked_transfer, 64, 0);
687
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1451 }
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1452 }
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1453
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1454 static CX_TEST_SUBROUTINE(test_http_client_io_large_chunked_transfer, size_t blocksz, int error_interval) {
690
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1455 int chunk1 = 1024*1024*2;
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1456 int chunk2 = 1024*128;
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1457 int chunk3 = 123;
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1458 int chunk4 = 1024*1024*5;
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1459 int chunk5 = 1024*16;
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1460
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1461 char *chunkbuf = malloc(chunk4);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1462
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1463 int ctlen = chunk1 + chunk2 + chunk3 + chunk4 + chunk5;
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1464 CxBuffer *resp = cxBufferCreate(NULL, NULL, ctlen * 128, CX_BUFFER_AUTO_EXTEND|CX_BUFFER_FREE_CONTENTS);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1465 cx_bprintf(resp,
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1466 "HTTP/1.1 200 OK\r\n"
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1467 "Transfer-encoding: chunked\r\n"
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1468 "\r\n",
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1469 ctlen);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1470 size_t content_start = resp->pos;
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1471 CX_TEST_ASSERT(resp->pos + ctlen < resp->capacity);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1472
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1473 memset(chunkbuf, 'x', chunk1);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1474 cx_bprintf(resp, "%x\r\n", chunk1);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1475 int chunk1_pos = resp->pos;
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1476 cxBufferWrite(chunkbuf, 1, chunk1, resp);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1477 cxBufferPutString(resp, "\r\n");
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1478
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1479 memset(chunkbuf, '2', chunk2);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1480 cx_bprintf(resp, "%x\r\n", chunk2);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1481 int chunk2_pos = resp->pos;
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1482 cxBufferWrite(chunkbuf, 1, chunk2, resp);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1483 cxBufferPutString(resp, "\r\n");
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1484
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1485 memset(chunkbuf, 'c', chunk3);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1486 cx_bprintf(resp, "%x\r\n", chunk3);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1487 int chunk3_pos = resp->pos;
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1488 cxBufferWrite(chunkbuf, 1, chunk3, resp);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1489 cxBufferPutString(resp, "\r\n");
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1490
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1491 memset(chunkbuf, '4', chunk4);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1492 cx_bprintf(resp, "%x\r\n", chunk4);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1493 int chunk4_pos = resp->pos;
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1494 cxBufferWrite(chunkbuf, 1, chunk4, resp);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1495 cxBufferPutString(resp, "\r\n");
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1496
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1497 memset(chunkbuf, 'z', chunk5);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1498 cx_bprintf(resp, "%x\r\n", chunk5);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1499 int chunk5_pos = resp->pos;
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1500 cxBufferWrite(chunkbuf, 1, chunk5, resp);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1501 cxBufferPutString(resp, "\r\n0\r\n\r\n");
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1502
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1503
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1504 CxBuffer *buf = cxBufferCreate(NULL, NULL, 1024, CX_BUFFER_AUTO_EXTEND|CX_BUFFER_FREE_CONTENTS);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1505
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1506 CX_TEST_CALL_SUBROUTINE(test_http_client_io, cx_strn(resp->space, resp->size), 200, "OK", buf, blocksz, error_interval);
690
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1507 CX_TEST_ASSERT(buf->size == ctlen);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1508 CX_TEST_ASSERT(!memcmp(buf->space, resp->space + chunk1_pos, chunk1));
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1509 CX_TEST_ASSERT(!memcmp(buf->space + chunk1, resp->space + chunk2_pos, chunk2));
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1510 CX_TEST_ASSERT(!memcmp(buf->space + chunk1 + chunk2, resp->space + chunk3_pos, chunk3));
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1511 CX_TEST_ASSERT(!memcmp(buf->space + chunk1 + chunk2 + chunk3, resp->space + chunk4_pos, chunk4));
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1512 CX_TEST_ASSERT(!memcmp(buf->space + chunk1 + chunk2 + chunk3 + chunk4, resp->space + chunk5_pos, chunk5));
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1513
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1514 cxBufferFree(resp);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1515 cxBufferFree(buf);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1516 }
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1517
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1518 static CX_TEST(test_http_client_io_large_chunked_transfer_1024b) {
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1519 CX_TEST_DO {
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1520 CX_TEST_CALL_SUBROUTINE(test_http_client_io_large_chunked_transfer, 1024, 0);
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1521 }
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1522 }
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1523
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1524 static CX_TEST(test_http_client_io_write_error1) {
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1525 CX_TEST_DO {
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1526 CX_TEST_CALL_SUBROUTINE(test_http_client_io_simple, 1, 1);
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1527 }
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1528 }
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1529
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1530 static CX_TEST(test_http_client_io_write_error2) {
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1531 CX_TEST_DO {
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1532 CX_TEST_CALL_SUBROUTINE(test_http_client_io_simple, 1, 2);
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1533 }
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1534 }
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1535
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1536 static CX_TEST(test_http_client_io_write_error3) {
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1537 CX_TEST_DO {
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1538 CX_TEST_CALL_SUBROUTINE(test_http_client_io_simple, 1, 3);
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1539 }
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1540 }
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1541
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1542 static CX_TEST(test_http_client_io_write_blsz8_error1) {
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1543 CX_TEST_DO {
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1544 CX_TEST_CALL_SUBROUTINE(test_http_client_io_simple, 8, 1);
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1545 }
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1546 }
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1547
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1548 static CX_TEST(test_http_client_io_write_blsz8_error2) {
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1549 CX_TEST_DO {
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1550 CX_TEST_CALL_SUBROUTINE(test_http_client_io_simple, 8, 2);
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1551 }
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1552 }
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1553
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1554 static CX_TEST(test_http_client_io_write_blsz8_error3) {
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1555 CX_TEST_DO {
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1556 CX_TEST_CALL_SUBROUTINE(test_http_client_io_simple, 8, 3);
690
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1557 }
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1558 }
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1559
718
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1560 static CX_TEST(test_http_client_ws_msg_out) {
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1561 CX_TEST_DO {
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1562 EventHandler dummy;
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1563 HttpClient *client = http_client_new(&dummy);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1564
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1565 int fds[2];
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1566 util_socketpair(fds);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1567 util_socket_setnonblock(fds[0], 1);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1568 util_socket_setnonblock(fds[1], 1);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1569 client->socketfd = fds[0];
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1570 client->event.cookie = client;
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1571 int sock = fds[1];
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1572
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1573 create_req_buffer(client);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1574 client->transfer_buffer_len = 0;
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1575 client->transfer_buffer_pos = 0;
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1576
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1577 IOStream *fd = Sysstream_new(NULL, client->socketfd);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1578 HttpStream *http = (HttpStream*)httpstream_new(NULL, fd);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1579 http->fd->setmode((IOStream*)http, IO_MODE_RAW);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1580 client->stream = http;
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1581
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1582 size_t out_nbytes = 1024*1024*64;
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1583 char *outbuf = malloc(out_nbytes);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1584 // init the buffer with random data
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1585 for(size_t i=0;i<out_nbytes;i+=sizeof(int)) {
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1586 int *p = (int*)(outbuf+i);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1587 *p = rand();
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1588 }
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1589 size_t out_pos = 0;
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1590
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1591 CxBuffer *inbuf = cxBufferCreate(NULL, NULL, out_nbytes, CX_BUFFER_FREE_CONTENTS);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1592
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1593 WSBool add_message_would_block = FALSE;
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1594
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1595
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1596 while(inbuf->pos < out_nbytes) {
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1597 // add outbuf data to message buffer
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1598 char *msg = outbuf + out_pos;
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1599 size_t msglen = out_nbytes - out_pos;
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1600 while(msglen > 0) {
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1601 int ret = http_client_add_message(client, msg, msglen);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1602 if(ret == HTTP_CLIENT_CALLBACK_WOULD_BLOCK) {
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1603 add_message_would_block = TRUE;
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1604 break;
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1605 }
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1606 out_pos += ret;
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1607 msg = outbuf + out_pos;
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1608 msglen = out_nbytes - out_pos;
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1609 }
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1610
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1611 // send message buffer to the socket
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1612 int ret = client_ws_io(&dummy, &client->event);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1613 CX_TEST_ASSERT(client->error == 0);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1614
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1615 // run client_ws_io again, it should do nothing
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1616 ret = client_ws_io(&dummy, &client->event);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1617 CX_TEST_ASSERT(ret == 1); // would block
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1618
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1619 ssize_t r = read(sock, inbuf->space + inbuf->pos, inbuf->capacity - inbuf->pos);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1620 if(r > 0) {
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1621 inbuf->pos += r;
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1622 inbuf->size += r;
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1623 }
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1624 }
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1625
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1626 // http_client_add_message should block at least once
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1627 CX_TEST_ASSERT(add_message_would_block);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1628 CX_TEST_ASSERT(inbuf->pos == out_nbytes);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1629 CX_TEST_ASSERT(!memcmp(inbuf->space, outbuf, out_nbytes));
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1630
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1631 cxBufferFree(inbuf);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1632 http_client_free(client);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1633 close(sock);
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1634 }
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1635 }
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1636
719
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1637 static ssize_t test_ws_write(HttpClient *client, void *buf, size_t nbytes, void *userdata) {
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1638 CxBuffer *out = (CxBuffer*)userdata;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1639 return cxBufferWrite(buf, 1, nbytes, out);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1640 }
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1641
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1642 static CX_TEST(test_http_client_ws_msg_in) {
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1643 CX_TEST_DO {
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1644 EventHandler dummy;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1645 HttpClient *client = http_client_new(&dummy);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1646
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1647 int fds[2];
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1648 util_socketpair(fds);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1649 util_socket_setnonblock(fds[0], 1);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1650 util_socket_setnonblock(fds[1], 1);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1651 client->socketfd = fds[0];
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1652 client->event.cookie = client;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1653 int sock = fds[1];
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1654
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1655 create_req_buffer(client);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1656 client->transfer_buffer_len = 0;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1657 client->transfer_buffer_pos = 0;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1658 client->transfer2_buffer_alloc = 1024*128;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1659 client->transfer2_buffer = malloc(client->transfer2_buffer_alloc);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1660
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1661 IOStream *fd = Sysstream_new(NULL, client->socketfd);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1662 HttpStream *http = (HttpStream*)httpstream_new(NULL, fd);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1663 http->raw = WS_TRUE;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1664 client->stream = http;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1665
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1666 size_t str_nbytes = 1024*1024*64;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1667 char *str = malloc(str_nbytes);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1668 // init the buffer with random data
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1669 for(size_t i=0;i<str_nbytes;i+=sizeof(int)) {
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1670 int *p = (int*)(str+i);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1671 *p = rand();
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1672 }
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1673 size_t str_pos = 0;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1674
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1675 CxBuffer *out = cxBufferCreate(NULL, NULL, str_nbytes, CX_BUFFER_AUTO_EXTEND|CX_BUFFER_FREE_CONTENTS);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1676 client->ws_write = test_ws_write;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1677 client->ws_write_userdata = out;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1678
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1679 while(out->size < str_nbytes) {
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1680 size_t slen = str_nbytes - str_pos;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1681 if(slen > 64*1024) {
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1682 slen = 64*1024;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1683 }
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1684
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1685 if(slen > 0) {
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1686 ssize_t w = write(sock, str + str_pos, slen);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1687 if(w > 0) {
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1688 str_pos += w;
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1689 }
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1690 }
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1691
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1692 // webosocket IO: this should read from the other socket and
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1693 // call the ws_write callback
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1694 int ret = client_ws_io(&dummy, &client->event);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1695 CX_TEST_ASSERT(client->error == 0);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1696
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1697 // run client_ws_io again, it should do nothing
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1698 ret = client_ws_io(&dummy, &client->event);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1699 CX_TEST_ASSERT(ret == 1); // would block
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1700 }
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1701
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1702 // http_client_add_message should block at least once
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1703 CX_TEST_ASSERT(out->pos == str_nbytes);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1704 CX_TEST_ASSERT(!memcmp(out->space, str, str_nbytes));
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1705
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1706 cxBufferFree(out);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1707 http_client_free(client);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1708 close(sock);
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1709 }
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1710 }
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1711
673
144bdc33fdb6 add socket utils
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 672
diff changeset
1712 void http_client_add_tests(CxTestSuite *suite) {
144bdc33fdb6 add socket utils
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 672
diff changeset
1713 cx_test_register(suite, test_http_client_send_request);
680
02935baa186b httpclient: add support for chunked transfer encoding for request bodies
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 679
diff changeset
1714 cx_test_register(suite, test_http_client_send_request_body_chunked);
684
48da20bde908 refactor http client io to use an HttpStream for reading the response
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 683
diff changeset
1715 cx_test_register(suite, test_http_client_read_response_head);
686
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1716 cx_test_register(suite, test_http_client_read_response_ctlen);
9adf57ddcd0f fix header iteration in client_read_response_header and add more tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 684
diff changeset
1717 cx_test_register(suite, test_http_client_read_response_ctlen_big);
687
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1718 cx_test_register(suite, test_http_client_io_simple_1b);
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1719 cx_test_register(suite, test_http_client_io_simple_2b);
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1720 cx_test_register(suite, test_http_client_io_simple_3b);
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1721 cx_test_register(suite, test_http_client_io_simple_16b);
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1722 cx_test_register(suite, test_http_client_io_simple_512b);
690
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1723 cx_test_register(suite, test_http_client_io_large_128b);
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1724 cx_test_register(suite, test_http_client_io_large_4096b);
687
4bded456b4a7 add test for http client responses with chunked transfer encoding
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 686
diff changeset
1725 cx_test_register(suite, test_http_client_io_chunked_transfer_1b);
688
02360295ec58 fix chunkheader parser not detecting EOF correctly in some cases
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 687
diff changeset
1726 cx_test_register(suite, test_http_client_io_chunked_transfer_2b);
02360295ec58 fix chunkheader parser not detecting EOF correctly in some cases
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 687
diff changeset
1727 cx_test_register(suite, test_http_client_io_chunked_transfer_8b);
02360295ec58 fix chunkheader parser not detecting EOF correctly in some cases
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 687
diff changeset
1728 cx_test_register(suite, test_http_client_io_chunked_transfer_64b);
690
c7b73000a1ed add large io http client tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 688
diff changeset
1729 cx_test_register(suite, test_http_client_io_large_chunked_transfer_1024b);
692
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1730 cx_test_register(suite, test_http_client_io_write_error1);
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1731 cx_test_register(suite, test_http_client_io_write_error2);
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1732 cx_test_register(suite, test_http_client_io_write_error3);
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1733 cx_test_register(suite, test_http_client_io_write_blsz8_error1);
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1734 cx_test_register(suite, test_http_client_io_write_blsz8_error2);
32faa1d6a744 add httpclient response write error tests
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 691
diff changeset
1735 cx_test_register(suite, test_http_client_io_write_blsz8_error3);
718
9e98618464ae add httpclient test for sending websocket messages
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 717
diff changeset
1736 cx_test_register(suite, test_http_client_ws_msg_out);
719
c4c2b8e8ddc5 add test_http_client_ws_msg_in
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 718
diff changeset
1737 cx_test_register(suite, test_http_client_ws_msg_in);
673
144bdc33fdb6 add socket utils
Olaf Wintermann <olaf.wintermann@gmail.com>
parents: 672
diff changeset
1738 }

mercurial