1*6236dae4SAndroid Build Coastguard Worker /***************************************************************************
2*6236dae4SAndroid Build Coastguard Worker * _ _ ____ _
3*6236dae4SAndroid Build Coastguard Worker * Project ___| | | | _ \| |
4*6236dae4SAndroid Build Coastguard Worker * / __| | | | |_) | |
5*6236dae4SAndroid Build Coastguard Worker * | (__| |_| | _ <| |___
6*6236dae4SAndroid Build Coastguard Worker * \___|\___/|_| \_\_____|
7*6236dae4SAndroid Build Coastguard Worker *
8*6236dae4SAndroid Build Coastguard Worker * Copyright (C) Daniel Stenberg, <[email protected]>, et al.
9*6236dae4SAndroid Build Coastguard Worker *
10*6236dae4SAndroid Build Coastguard Worker * This software is licensed as described in the file COPYING, which
11*6236dae4SAndroid Build Coastguard Worker * you should have received as part of this distribution. The terms
12*6236dae4SAndroid Build Coastguard Worker * are also available at https://curl.se/docs/copyright.html.
13*6236dae4SAndroid Build Coastguard Worker *
14*6236dae4SAndroid Build Coastguard Worker * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15*6236dae4SAndroid Build Coastguard Worker * copies of the Software, and permit persons to whom the Software is
16*6236dae4SAndroid Build Coastguard Worker * furnished to do so, under the terms of the COPYING file.
17*6236dae4SAndroid Build Coastguard Worker *
18*6236dae4SAndroid Build Coastguard Worker * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19*6236dae4SAndroid Build Coastguard Worker * KIND, either express or implied.
20*6236dae4SAndroid Build Coastguard Worker *
21*6236dae4SAndroid Build Coastguard Worker * SPDX-License-Identifier: curl
22*6236dae4SAndroid Build Coastguard Worker *
23*6236dae4SAndroid Build Coastguard Worker ***************************************************************************/
24*6236dae4SAndroid Build Coastguard Worker
25*6236dae4SAndroid Build Coastguard Worker /*
26*6236dae4SAndroid Build Coastguard Worker * This test sends data with CURLOPT_KEEP_SENDING_ON_ERROR.
27*6236dae4SAndroid Build Coastguard Worker * The server responds with an early error response.
28*6236dae4SAndroid Build Coastguard Worker * The test is successful if the connection can be reused for the next request,
29*6236dae4SAndroid Build Coastguard Worker * because this implies that the data has been sent completely to the server.
30*6236dae4SAndroid Build Coastguard Worker */
31*6236dae4SAndroid Build Coastguard Worker
32*6236dae4SAndroid Build Coastguard Worker #include "test.h"
33*6236dae4SAndroid Build Coastguard Worker
34*6236dae4SAndroid Build Coastguard Worker #include "memdebug.h"
35*6236dae4SAndroid Build Coastguard Worker
36*6236dae4SAndroid Build Coastguard Worker struct cb_data {
37*6236dae4SAndroid Build Coastguard Worker CURL *easy_handle;
38*6236dae4SAndroid Build Coastguard Worker int response_received;
39*6236dae4SAndroid Build Coastguard Worker int paused;
40*6236dae4SAndroid Build Coastguard Worker size_t remaining_bytes;
41*6236dae4SAndroid Build Coastguard Worker };
42*6236dae4SAndroid Build Coastguard Worker
43*6236dae4SAndroid Build Coastguard Worker
reset_data(struct cb_data * data,CURL * curl)44*6236dae4SAndroid Build Coastguard Worker static void reset_data(struct cb_data *data, CURL *curl)
45*6236dae4SAndroid Build Coastguard Worker {
46*6236dae4SAndroid Build Coastguard Worker data->easy_handle = curl;
47*6236dae4SAndroid Build Coastguard Worker data->response_received = 0;
48*6236dae4SAndroid Build Coastguard Worker data->paused = 0;
49*6236dae4SAndroid Build Coastguard Worker data->remaining_bytes = 3;
50*6236dae4SAndroid Build Coastguard Worker }
51*6236dae4SAndroid Build Coastguard Worker
52*6236dae4SAndroid Build Coastguard Worker
read_callback(char * ptr,size_t size,size_t nitems,void * userdata)53*6236dae4SAndroid Build Coastguard Worker static size_t read_callback(char *ptr, size_t size, size_t nitems,
54*6236dae4SAndroid Build Coastguard Worker void *userdata)
55*6236dae4SAndroid Build Coastguard Worker {
56*6236dae4SAndroid Build Coastguard Worker struct cb_data *data = (struct cb_data *)userdata;
57*6236dae4SAndroid Build Coastguard Worker
58*6236dae4SAndroid Build Coastguard Worker /* wait until the server has sent all response headers */
59*6236dae4SAndroid Build Coastguard Worker if(data->response_received) {
60*6236dae4SAndroid Build Coastguard Worker size_t totalsize = nitems * size;
61*6236dae4SAndroid Build Coastguard Worker
62*6236dae4SAndroid Build Coastguard Worker size_t bytes_to_send = data->remaining_bytes;
63*6236dae4SAndroid Build Coastguard Worker if(bytes_to_send > totalsize) {
64*6236dae4SAndroid Build Coastguard Worker bytes_to_send = totalsize;
65*6236dae4SAndroid Build Coastguard Worker }
66*6236dae4SAndroid Build Coastguard Worker
67*6236dae4SAndroid Build Coastguard Worker memset(ptr, 'a', bytes_to_send);
68*6236dae4SAndroid Build Coastguard Worker data->remaining_bytes -= bytes_to_send;
69*6236dae4SAndroid Build Coastguard Worker
70*6236dae4SAndroid Build Coastguard Worker return bytes_to_send;
71*6236dae4SAndroid Build Coastguard Worker }
72*6236dae4SAndroid Build Coastguard Worker else {
73*6236dae4SAndroid Build Coastguard Worker data->paused = 1;
74*6236dae4SAndroid Build Coastguard Worker return CURL_READFUNC_PAUSE;
75*6236dae4SAndroid Build Coastguard Worker }
76*6236dae4SAndroid Build Coastguard Worker }
77*6236dae4SAndroid Build Coastguard Worker
78*6236dae4SAndroid Build Coastguard Worker
write_callback(char * ptr,size_t size,size_t nmemb,void * userdata)79*6236dae4SAndroid Build Coastguard Worker static size_t write_callback(char *ptr, size_t size, size_t nmemb,
80*6236dae4SAndroid Build Coastguard Worker void *userdata)
81*6236dae4SAndroid Build Coastguard Worker {
82*6236dae4SAndroid Build Coastguard Worker struct cb_data *data = (struct cb_data *)userdata;
83*6236dae4SAndroid Build Coastguard Worker size_t totalsize = nmemb * size;
84*6236dae4SAndroid Build Coastguard Worker
85*6236dae4SAndroid Build Coastguard Worker /* unused parameter */
86*6236dae4SAndroid Build Coastguard Worker (void)ptr;
87*6236dae4SAndroid Build Coastguard Worker
88*6236dae4SAndroid Build Coastguard Worker /* all response headers have been received */
89*6236dae4SAndroid Build Coastguard Worker data->response_received = 1;
90*6236dae4SAndroid Build Coastguard Worker
91*6236dae4SAndroid Build Coastguard Worker if(data->paused) {
92*6236dae4SAndroid Build Coastguard Worker /* continue to send request body data */
93*6236dae4SAndroid Build Coastguard Worker data->paused = 0;
94*6236dae4SAndroid Build Coastguard Worker curl_easy_pause(data->easy_handle, CURLPAUSE_CONT);
95*6236dae4SAndroid Build Coastguard Worker }
96*6236dae4SAndroid Build Coastguard Worker
97*6236dae4SAndroid Build Coastguard Worker return totalsize;
98*6236dae4SAndroid Build Coastguard Worker }
99*6236dae4SAndroid Build Coastguard Worker
100*6236dae4SAndroid Build Coastguard Worker
perform_and_check_connections(CURL * curl,const char * description,long expected_connections)101*6236dae4SAndroid Build Coastguard Worker static int perform_and_check_connections(CURL *curl, const char *description,
102*6236dae4SAndroid Build Coastguard Worker long expected_connections)
103*6236dae4SAndroid Build Coastguard Worker {
104*6236dae4SAndroid Build Coastguard Worker CURLcode res;
105*6236dae4SAndroid Build Coastguard Worker long connections = 0;
106*6236dae4SAndroid Build Coastguard Worker
107*6236dae4SAndroid Build Coastguard Worker res = curl_easy_perform(curl);
108*6236dae4SAndroid Build Coastguard Worker if(res != CURLE_OK) {
109*6236dae4SAndroid Build Coastguard Worker fprintf(stderr, "curl_easy_perform() failed with %d\n", res);
110*6236dae4SAndroid Build Coastguard Worker return TEST_ERR_MAJOR_BAD;
111*6236dae4SAndroid Build Coastguard Worker }
112*6236dae4SAndroid Build Coastguard Worker
113*6236dae4SAndroid Build Coastguard Worker res = curl_easy_getinfo(curl, CURLINFO_NUM_CONNECTS, &connections);
114*6236dae4SAndroid Build Coastguard Worker if(res != CURLE_OK) {
115*6236dae4SAndroid Build Coastguard Worker fprintf(stderr, "curl_easy_getinfo() failed\n");
116*6236dae4SAndroid Build Coastguard Worker return TEST_ERR_MAJOR_BAD;
117*6236dae4SAndroid Build Coastguard Worker }
118*6236dae4SAndroid Build Coastguard Worker
119*6236dae4SAndroid Build Coastguard Worker fprintf(stderr, "%s: expected: %ld connections; actual: %ld connections\n",
120*6236dae4SAndroid Build Coastguard Worker description, expected_connections, connections);
121*6236dae4SAndroid Build Coastguard Worker
122*6236dae4SAndroid Build Coastguard Worker if(connections != expected_connections) {
123*6236dae4SAndroid Build Coastguard Worker return TEST_ERR_FAILURE;
124*6236dae4SAndroid Build Coastguard Worker }
125*6236dae4SAndroid Build Coastguard Worker
126*6236dae4SAndroid Build Coastguard Worker return TEST_ERR_SUCCESS;
127*6236dae4SAndroid Build Coastguard Worker }
128*6236dae4SAndroid Build Coastguard Worker
129*6236dae4SAndroid Build Coastguard Worker
test(char * URL)130*6236dae4SAndroid Build Coastguard Worker CURLcode test(char *URL)
131*6236dae4SAndroid Build Coastguard Worker {
132*6236dae4SAndroid Build Coastguard Worker struct cb_data data;
133*6236dae4SAndroid Build Coastguard Worker CURL *curl = NULL;
134*6236dae4SAndroid Build Coastguard Worker CURLcode res = TEST_ERR_FAILURE;
135*6236dae4SAndroid Build Coastguard Worker int result;
136*6236dae4SAndroid Build Coastguard Worker
137*6236dae4SAndroid Build Coastguard Worker if(curl_global_init(CURL_GLOBAL_ALL) != CURLE_OK) {
138*6236dae4SAndroid Build Coastguard Worker fprintf(stderr, "curl_global_init() failed\n");
139*6236dae4SAndroid Build Coastguard Worker return TEST_ERR_MAJOR_BAD;
140*6236dae4SAndroid Build Coastguard Worker }
141*6236dae4SAndroid Build Coastguard Worker
142*6236dae4SAndroid Build Coastguard Worker curl = curl_easy_init();
143*6236dae4SAndroid Build Coastguard Worker if(!curl) {
144*6236dae4SAndroid Build Coastguard Worker fprintf(stderr, "curl_easy_init() failed\n");
145*6236dae4SAndroid Build Coastguard Worker curl_global_cleanup();
146*6236dae4SAndroid Build Coastguard Worker return TEST_ERR_MAJOR_BAD;
147*6236dae4SAndroid Build Coastguard Worker }
148*6236dae4SAndroid Build Coastguard Worker
149*6236dae4SAndroid Build Coastguard Worker reset_data(&data, curl);
150*6236dae4SAndroid Build Coastguard Worker
151*6236dae4SAndroid Build Coastguard Worker test_setopt(curl, CURLOPT_URL, URL);
152*6236dae4SAndroid Build Coastguard Worker test_setopt(curl, CURLOPT_POST, 1L);
153*6236dae4SAndroid Build Coastguard Worker test_setopt(curl, CURLOPT_POSTFIELDSIZE_LARGE,
154*6236dae4SAndroid Build Coastguard Worker (curl_off_t)data.remaining_bytes);
155*6236dae4SAndroid Build Coastguard Worker test_setopt(curl, CURLOPT_VERBOSE, 1L);
156*6236dae4SAndroid Build Coastguard Worker test_setopt(curl, CURLOPT_READFUNCTION, read_callback);
157*6236dae4SAndroid Build Coastguard Worker test_setopt(curl, CURLOPT_READDATA, &data);
158*6236dae4SAndroid Build Coastguard Worker test_setopt(curl, CURLOPT_WRITEFUNCTION, write_callback);
159*6236dae4SAndroid Build Coastguard Worker test_setopt(curl, CURLOPT_WRITEDATA, &data);
160*6236dae4SAndroid Build Coastguard Worker
161*6236dae4SAndroid Build Coastguard Worker result = perform_and_check_connections(curl,
162*6236dae4SAndroid Build Coastguard Worker "First request without CURLOPT_KEEP_SENDING_ON_ERROR", 1);
163*6236dae4SAndroid Build Coastguard Worker if(result != TEST_ERR_SUCCESS) {
164*6236dae4SAndroid Build Coastguard Worker res = (CURLcode) result;
165*6236dae4SAndroid Build Coastguard Worker goto test_cleanup;
166*6236dae4SAndroid Build Coastguard Worker }
167*6236dae4SAndroid Build Coastguard Worker
168*6236dae4SAndroid Build Coastguard Worker reset_data(&data, curl);
169*6236dae4SAndroid Build Coastguard Worker
170*6236dae4SAndroid Build Coastguard Worker result = perform_and_check_connections(curl,
171*6236dae4SAndroid Build Coastguard Worker "Second request without CURLOPT_KEEP_SENDING_ON_ERROR", 1);
172*6236dae4SAndroid Build Coastguard Worker if(result != TEST_ERR_SUCCESS) {
173*6236dae4SAndroid Build Coastguard Worker res = (CURLcode) result;
174*6236dae4SAndroid Build Coastguard Worker goto test_cleanup;
175*6236dae4SAndroid Build Coastguard Worker }
176*6236dae4SAndroid Build Coastguard Worker
177*6236dae4SAndroid Build Coastguard Worker test_setopt(curl, CURLOPT_KEEP_SENDING_ON_ERROR, 1L);
178*6236dae4SAndroid Build Coastguard Worker
179*6236dae4SAndroid Build Coastguard Worker reset_data(&data, curl);
180*6236dae4SAndroid Build Coastguard Worker
181*6236dae4SAndroid Build Coastguard Worker result = perform_and_check_connections(curl,
182*6236dae4SAndroid Build Coastguard Worker "First request with CURLOPT_KEEP_SENDING_ON_ERROR", 1);
183*6236dae4SAndroid Build Coastguard Worker if(result != TEST_ERR_SUCCESS) {
184*6236dae4SAndroid Build Coastguard Worker res = (CURLcode) result;
185*6236dae4SAndroid Build Coastguard Worker goto test_cleanup;
186*6236dae4SAndroid Build Coastguard Worker }
187*6236dae4SAndroid Build Coastguard Worker
188*6236dae4SAndroid Build Coastguard Worker reset_data(&data, curl);
189*6236dae4SAndroid Build Coastguard Worker
190*6236dae4SAndroid Build Coastguard Worker result = perform_and_check_connections(curl,
191*6236dae4SAndroid Build Coastguard Worker "Second request with CURLOPT_KEEP_SENDING_ON_ERROR", 0);
192*6236dae4SAndroid Build Coastguard Worker if(result != TEST_ERR_SUCCESS) {
193*6236dae4SAndroid Build Coastguard Worker res = (CURLcode) result;
194*6236dae4SAndroid Build Coastguard Worker goto test_cleanup;
195*6236dae4SAndroid Build Coastguard Worker }
196*6236dae4SAndroid Build Coastguard Worker
197*6236dae4SAndroid Build Coastguard Worker res = TEST_ERR_SUCCESS;
198*6236dae4SAndroid Build Coastguard Worker
199*6236dae4SAndroid Build Coastguard Worker test_cleanup:
200*6236dae4SAndroid Build Coastguard Worker
201*6236dae4SAndroid Build Coastguard Worker curl_easy_cleanup(curl);
202*6236dae4SAndroid Build Coastguard Worker
203*6236dae4SAndroid Build Coastguard Worker curl_global_cleanup();
204*6236dae4SAndroid Build Coastguard Worker
205*6236dae4SAndroid Build Coastguard Worker return res;
206*6236dae4SAndroid Build Coastguard Worker }
207