113498266Sopenharmony_ci/***************************************************************************
213498266Sopenharmony_ci *                                  _   _ ____  _
313498266Sopenharmony_ci *  Project                     ___| | | |  _ \| |
413498266Sopenharmony_ci *                             / __| | | | |_) | |
513498266Sopenharmony_ci *                            | (__| |_| |  _ <| |___
613498266Sopenharmony_ci *                             \___|\___/|_| \_\_____|
713498266Sopenharmony_ci *
813498266Sopenharmony_ci * Copyright (C) Daniel Stenberg, <daniel@haxx.se>, et al.
913498266Sopenharmony_ci *
1013498266Sopenharmony_ci * This software is licensed as described in the file COPYING, which
1113498266Sopenharmony_ci * you should have received as part of this distribution. The terms
1213498266Sopenharmony_ci * are also available at https://curl.se/docs/copyright.html.
1313498266Sopenharmony_ci *
1413498266Sopenharmony_ci * You may opt to use, copy, modify, merge, publish, distribute and/or sell
1513498266Sopenharmony_ci * copies of the Software, and permit persons to whom the Software is
1613498266Sopenharmony_ci * furnished to do so, under the terms of the COPYING file.
1713498266Sopenharmony_ci *
1813498266Sopenharmony_ci * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
1913498266Sopenharmony_ci * KIND, either express or implied.
2013498266Sopenharmony_ci *
2113498266Sopenharmony_ci * SPDX-License-Identifier: curl
2213498266Sopenharmony_ci *
2313498266Sopenharmony_ci ***************************************************************************/
2413498266Sopenharmony_ci
2513498266Sopenharmony_ci/*
2613498266Sopenharmony_ci * This test sends data with CURLOPT_KEEP_SENDING_ON_ERROR.
2713498266Sopenharmony_ci * The server responds with an early error response.
2813498266Sopenharmony_ci * The test is successful if the connection can be reused for the next request,
2913498266Sopenharmony_ci * because this implies that the data has been sent completely to the server.
3013498266Sopenharmony_ci */
3113498266Sopenharmony_ci
3213498266Sopenharmony_ci#include "test.h"
3313498266Sopenharmony_ci
3413498266Sopenharmony_ci#include "memdebug.h"
3513498266Sopenharmony_ci
3613498266Sopenharmony_cistruct cb_data {
3713498266Sopenharmony_ci  CURL *easy_handle;
3813498266Sopenharmony_ci  int response_received;
3913498266Sopenharmony_ci  int paused;
4013498266Sopenharmony_ci  size_t remaining_bytes;
4113498266Sopenharmony_ci};
4213498266Sopenharmony_ci
4313498266Sopenharmony_ci
4413498266Sopenharmony_cistatic void reset_data(struct cb_data *data, CURL *curl)
4513498266Sopenharmony_ci{
4613498266Sopenharmony_ci  data->easy_handle = curl;
4713498266Sopenharmony_ci  data->response_received = 0;
4813498266Sopenharmony_ci  data->paused = 0;
4913498266Sopenharmony_ci  data->remaining_bytes = 3;
5013498266Sopenharmony_ci}
5113498266Sopenharmony_ci
5213498266Sopenharmony_ci
5313498266Sopenharmony_cistatic size_t read_callback(char *ptr, size_t size, size_t nitems,
5413498266Sopenharmony_ci                            void *userdata)
5513498266Sopenharmony_ci{
5613498266Sopenharmony_ci  struct cb_data *data = (struct cb_data *)userdata;
5713498266Sopenharmony_ci
5813498266Sopenharmony_ci  /* wait until the server has sent all response headers */
5913498266Sopenharmony_ci  if(data->response_received) {
6013498266Sopenharmony_ci    size_t totalsize = nitems * size;
6113498266Sopenharmony_ci
6213498266Sopenharmony_ci    size_t bytes_to_send = data->remaining_bytes;
6313498266Sopenharmony_ci    if(bytes_to_send > totalsize) {
6413498266Sopenharmony_ci      bytes_to_send = totalsize;
6513498266Sopenharmony_ci    }
6613498266Sopenharmony_ci
6713498266Sopenharmony_ci    memset(ptr, 'a', bytes_to_send);
6813498266Sopenharmony_ci    data->remaining_bytes -= bytes_to_send;
6913498266Sopenharmony_ci
7013498266Sopenharmony_ci    return bytes_to_send;
7113498266Sopenharmony_ci  }
7213498266Sopenharmony_ci  else {
7313498266Sopenharmony_ci    data->paused = 1;
7413498266Sopenharmony_ci    return CURL_READFUNC_PAUSE;
7513498266Sopenharmony_ci  }
7613498266Sopenharmony_ci}
7713498266Sopenharmony_ci
7813498266Sopenharmony_ci
7913498266Sopenharmony_cistatic size_t write_callback(char *ptr, size_t size, size_t nmemb,
8013498266Sopenharmony_ci                             void *userdata)
8113498266Sopenharmony_ci{
8213498266Sopenharmony_ci  struct cb_data *data = (struct cb_data *)userdata;
8313498266Sopenharmony_ci  size_t totalsize = nmemb * size;
8413498266Sopenharmony_ci
8513498266Sopenharmony_ci  /* unused parameter */
8613498266Sopenharmony_ci  (void)ptr;
8713498266Sopenharmony_ci
8813498266Sopenharmony_ci  /* all response headers have been received */
8913498266Sopenharmony_ci  data->response_received = 1;
9013498266Sopenharmony_ci
9113498266Sopenharmony_ci  if(data->paused) {
9213498266Sopenharmony_ci    /* continue to send request body data */
9313498266Sopenharmony_ci    data->paused = 0;
9413498266Sopenharmony_ci    curl_easy_pause(data->easy_handle, CURLPAUSE_CONT);
9513498266Sopenharmony_ci  }
9613498266Sopenharmony_ci
9713498266Sopenharmony_ci  return totalsize;
9813498266Sopenharmony_ci}
9913498266Sopenharmony_ci
10013498266Sopenharmony_ci
10113498266Sopenharmony_cistatic int perform_and_check_connections(CURL *curl, const char *description,
10213498266Sopenharmony_ci                                         long expected_connections)
10313498266Sopenharmony_ci{
10413498266Sopenharmony_ci  CURLcode res;
10513498266Sopenharmony_ci  long connections = 0;
10613498266Sopenharmony_ci
10713498266Sopenharmony_ci  res = curl_easy_perform(curl);
10813498266Sopenharmony_ci  if(res != CURLE_OK) {
10913498266Sopenharmony_ci    fprintf(stderr, "curl_easy_perform() failed with %d\n", (int)res);
11013498266Sopenharmony_ci    return TEST_ERR_MAJOR_BAD;
11113498266Sopenharmony_ci  }
11213498266Sopenharmony_ci
11313498266Sopenharmony_ci  res = curl_easy_getinfo(curl, CURLINFO_NUM_CONNECTS, &connections);
11413498266Sopenharmony_ci  if(res != CURLE_OK) {
11513498266Sopenharmony_ci    fprintf(stderr, "curl_easy_getinfo() failed\n");
11613498266Sopenharmony_ci    return TEST_ERR_MAJOR_BAD;
11713498266Sopenharmony_ci  }
11813498266Sopenharmony_ci
11913498266Sopenharmony_ci  fprintf(stderr, "%s: expected: %ld connections; actual: %ld connections\n",
12013498266Sopenharmony_ci          description, expected_connections, connections);
12113498266Sopenharmony_ci
12213498266Sopenharmony_ci  if(connections != expected_connections) {
12313498266Sopenharmony_ci    return TEST_ERR_FAILURE;
12413498266Sopenharmony_ci  }
12513498266Sopenharmony_ci
12613498266Sopenharmony_ci  return TEST_ERR_SUCCESS;
12713498266Sopenharmony_ci}
12813498266Sopenharmony_ci
12913498266Sopenharmony_ci
13013498266Sopenharmony_ciint test(char *URL)
13113498266Sopenharmony_ci{
13213498266Sopenharmony_ci  struct cb_data data;
13313498266Sopenharmony_ci  CURL *curl = NULL;
13413498266Sopenharmony_ci  int res = TEST_ERR_FAILURE;
13513498266Sopenharmony_ci
13613498266Sopenharmony_ci  if(curl_global_init(CURL_GLOBAL_ALL) != CURLE_OK) {
13713498266Sopenharmony_ci    fprintf(stderr, "curl_global_init() failed\n");
13813498266Sopenharmony_ci    return TEST_ERR_MAJOR_BAD;
13913498266Sopenharmony_ci  }
14013498266Sopenharmony_ci
14113498266Sopenharmony_ci  curl = curl_easy_init();
14213498266Sopenharmony_ci  if(!curl) {
14313498266Sopenharmony_ci    fprintf(stderr, "curl_easy_init() failed\n");
14413498266Sopenharmony_ci    curl_global_cleanup();
14513498266Sopenharmony_ci    return TEST_ERR_MAJOR_BAD;
14613498266Sopenharmony_ci  }
14713498266Sopenharmony_ci
14813498266Sopenharmony_ci  reset_data(&data, curl);
14913498266Sopenharmony_ci
15013498266Sopenharmony_ci  test_setopt(curl, CURLOPT_URL, URL);
15113498266Sopenharmony_ci  test_setopt(curl, CURLOPT_POST, 1L);
15213498266Sopenharmony_ci  test_setopt(curl, CURLOPT_POSTFIELDSIZE_LARGE,
15313498266Sopenharmony_ci              (curl_off_t)data.remaining_bytes);
15413498266Sopenharmony_ci  test_setopt(curl, CURLOPT_VERBOSE, 1L);
15513498266Sopenharmony_ci  test_setopt(curl, CURLOPT_READFUNCTION, read_callback);
15613498266Sopenharmony_ci  test_setopt(curl, CURLOPT_READDATA, &data);
15713498266Sopenharmony_ci  test_setopt(curl, CURLOPT_WRITEFUNCTION, write_callback);
15813498266Sopenharmony_ci  test_setopt(curl, CURLOPT_WRITEDATA, &data);
15913498266Sopenharmony_ci
16013498266Sopenharmony_ci  res = perform_and_check_connections(curl,
16113498266Sopenharmony_ci    "First request without CURLOPT_KEEP_SENDING_ON_ERROR", 1);
16213498266Sopenharmony_ci  if(res != TEST_ERR_SUCCESS) {
16313498266Sopenharmony_ci    goto test_cleanup;
16413498266Sopenharmony_ci  }
16513498266Sopenharmony_ci
16613498266Sopenharmony_ci  reset_data(&data, curl);
16713498266Sopenharmony_ci
16813498266Sopenharmony_ci  res = perform_and_check_connections(curl,
16913498266Sopenharmony_ci    "Second request without CURLOPT_KEEP_SENDING_ON_ERROR", 1);
17013498266Sopenharmony_ci  if(res != TEST_ERR_SUCCESS) {
17113498266Sopenharmony_ci    goto test_cleanup;
17213498266Sopenharmony_ci  }
17313498266Sopenharmony_ci
17413498266Sopenharmony_ci  test_setopt(curl, CURLOPT_KEEP_SENDING_ON_ERROR, 1L);
17513498266Sopenharmony_ci
17613498266Sopenharmony_ci  reset_data(&data, curl);
17713498266Sopenharmony_ci
17813498266Sopenharmony_ci  res = perform_and_check_connections(curl,
17913498266Sopenharmony_ci    "First request with CURLOPT_KEEP_SENDING_ON_ERROR", 1);
18013498266Sopenharmony_ci  if(res != TEST_ERR_SUCCESS) {
18113498266Sopenharmony_ci    goto test_cleanup;
18213498266Sopenharmony_ci  }
18313498266Sopenharmony_ci
18413498266Sopenharmony_ci  reset_data(&data, curl);
18513498266Sopenharmony_ci
18613498266Sopenharmony_ci  res = perform_and_check_connections(curl,
18713498266Sopenharmony_ci    "Second request with CURLOPT_KEEP_SENDING_ON_ERROR", 0);
18813498266Sopenharmony_ci  if(res != TEST_ERR_SUCCESS) {
18913498266Sopenharmony_ci    goto test_cleanup;
19013498266Sopenharmony_ci  }
19113498266Sopenharmony_ci
19213498266Sopenharmony_ci  res = TEST_ERR_SUCCESS;
19313498266Sopenharmony_ci
19413498266Sopenharmony_citest_cleanup:
19513498266Sopenharmony_ci
19613498266Sopenharmony_ci  curl_easy_cleanup(curl);
19713498266Sopenharmony_ci
19813498266Sopenharmony_ci  curl_global_cleanup();
19913498266Sopenharmony_ci
20013498266Sopenharmony_ci  return (int)res;
20113498266Sopenharmony_ci}
202