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 * This code sets up multiple easy handles that transfer a single file from 2613498266Sopenharmony_ci * the same URL, in a serial manner after each other. Due to the connection 2713498266Sopenharmony_ci * sharing within the multi handle all transfers are performed on the same 2813498266Sopenharmony_ci * persistent connection. 2913498266Sopenharmony_ci * 3013498266Sopenharmony_ci * This source code is used for lib526, lib527 and lib532 with only #ifdefs 3113498266Sopenharmony_ci * controlling the small differences. 3213498266Sopenharmony_ci * 3313498266Sopenharmony_ci * - lib526 closes all easy handles after 3413498266Sopenharmony_ci * they all have transferred the file over the single connection 3513498266Sopenharmony_ci * - lib527 closes each easy handle after each single transfer. 3613498266Sopenharmony_ci * - lib532 uses only a single easy handle that is removed, reset and then 3713498266Sopenharmony_ci * re-added for each transfer 3813498266Sopenharmony_ci * 3913498266Sopenharmony_ci * Test case 526, 527 and 532 use FTP, while test 528 uses the lib526 tool but 4013498266Sopenharmony_ci * with HTTP. 4113498266Sopenharmony_ci */ 4213498266Sopenharmony_ci 4313498266Sopenharmony_ci#include "test.h" 4413498266Sopenharmony_ci 4513498266Sopenharmony_ci#include <fcntl.h> 4613498266Sopenharmony_ci 4713498266Sopenharmony_ci#include "testutil.h" 4813498266Sopenharmony_ci#include "warnless.h" 4913498266Sopenharmony_ci#include "memdebug.h" 5013498266Sopenharmony_ci 5113498266Sopenharmony_ci#define TEST_HANG_TIMEOUT 60 * 1000 5213498266Sopenharmony_ci 5313498266Sopenharmony_ci#define NUM_HANDLES 4 5413498266Sopenharmony_ci 5513498266Sopenharmony_ciint test(char *URL) 5613498266Sopenharmony_ci{ 5713498266Sopenharmony_ci int res = 0; 5813498266Sopenharmony_ci CURL *curl[NUM_HANDLES]; 5913498266Sopenharmony_ci int running; 6013498266Sopenharmony_ci CURLM *m = NULL; 6113498266Sopenharmony_ci int current = 0; 6213498266Sopenharmony_ci int i; 6313498266Sopenharmony_ci 6413498266Sopenharmony_ci for(i = 0; i < NUM_HANDLES; i++) 6513498266Sopenharmony_ci curl[i] = NULL; 6613498266Sopenharmony_ci 6713498266Sopenharmony_ci start_test_timing(); 6813498266Sopenharmony_ci 6913498266Sopenharmony_ci global_init(CURL_GLOBAL_ALL); 7013498266Sopenharmony_ci 7113498266Sopenharmony_ci /* get NUM_HANDLES easy handles */ 7213498266Sopenharmony_ci for(i = 0; i < NUM_HANDLES; i++) { 7313498266Sopenharmony_ci easy_init(curl[i]); 7413498266Sopenharmony_ci /* specify target */ 7513498266Sopenharmony_ci easy_setopt(curl[i], CURLOPT_URL, URL); 7613498266Sopenharmony_ci /* go verbose */ 7713498266Sopenharmony_ci easy_setopt(curl[i], CURLOPT_VERBOSE, 1L); 7813498266Sopenharmony_ci } 7913498266Sopenharmony_ci 8013498266Sopenharmony_ci multi_init(m); 8113498266Sopenharmony_ci 8213498266Sopenharmony_ci multi_add_handle(m, curl[current]); 8313498266Sopenharmony_ci 8413498266Sopenharmony_ci fprintf(stderr, "Start at URL 0\n"); 8513498266Sopenharmony_ci 8613498266Sopenharmony_ci for(;;) { 8713498266Sopenharmony_ci struct timeval interval; 8813498266Sopenharmony_ci fd_set rd, wr, exc; 8913498266Sopenharmony_ci int maxfd = -99; 9013498266Sopenharmony_ci 9113498266Sopenharmony_ci interval.tv_sec = 1; 9213498266Sopenharmony_ci interval.tv_usec = 0; 9313498266Sopenharmony_ci 9413498266Sopenharmony_ci multi_perform(m, &running); 9513498266Sopenharmony_ci 9613498266Sopenharmony_ci abort_on_test_timeout(); 9713498266Sopenharmony_ci 9813498266Sopenharmony_ci if(!running) { 9913498266Sopenharmony_ci#ifdef LIB527 10013498266Sopenharmony_ci /* NOTE: this code does not remove the handle from the multi handle 10113498266Sopenharmony_ci here, which would be the nice, sane and documented way of working. 10213498266Sopenharmony_ci This however tests that the API survives this abuse gracefully. */ 10313498266Sopenharmony_ci curl_easy_cleanup(curl[current]); 10413498266Sopenharmony_ci curl[current] = NULL; 10513498266Sopenharmony_ci#endif 10613498266Sopenharmony_ci if(++current < NUM_HANDLES) { 10713498266Sopenharmony_ci fprintf(stderr, "Advancing to URL %d\n", current); 10813498266Sopenharmony_ci#ifdef LIB532 10913498266Sopenharmony_ci /* first remove the only handle we use */ 11013498266Sopenharmony_ci curl_multi_remove_handle(m, curl[0]); 11113498266Sopenharmony_ci 11213498266Sopenharmony_ci /* make us reuse the same handle all the time, and try resetting 11313498266Sopenharmony_ci the handle first too */ 11413498266Sopenharmony_ci curl_easy_reset(curl[0]); 11513498266Sopenharmony_ci easy_setopt(curl[0], CURLOPT_URL, URL); 11613498266Sopenharmony_ci /* go verbose */ 11713498266Sopenharmony_ci easy_setopt(curl[0], CURLOPT_VERBOSE, 1L); 11813498266Sopenharmony_ci 11913498266Sopenharmony_ci /* re-add it */ 12013498266Sopenharmony_ci multi_add_handle(m, curl[0]); 12113498266Sopenharmony_ci#else 12213498266Sopenharmony_ci multi_add_handle(m, curl[current]); 12313498266Sopenharmony_ci#endif 12413498266Sopenharmony_ci } 12513498266Sopenharmony_ci else { 12613498266Sopenharmony_ci break; /* done */ 12713498266Sopenharmony_ci } 12813498266Sopenharmony_ci } 12913498266Sopenharmony_ci 13013498266Sopenharmony_ci FD_ZERO(&rd); 13113498266Sopenharmony_ci FD_ZERO(&wr); 13213498266Sopenharmony_ci FD_ZERO(&exc); 13313498266Sopenharmony_ci 13413498266Sopenharmony_ci multi_fdset(m, &rd, &wr, &exc, &maxfd); 13513498266Sopenharmony_ci 13613498266Sopenharmony_ci /* At this point, maxfd is guaranteed to be greater or equal than -1. */ 13713498266Sopenharmony_ci 13813498266Sopenharmony_ci select_test(maxfd + 1, &rd, &wr, &exc, &interval); 13913498266Sopenharmony_ci 14013498266Sopenharmony_ci abort_on_test_timeout(); 14113498266Sopenharmony_ci } 14213498266Sopenharmony_ci 14313498266Sopenharmony_citest_cleanup: 14413498266Sopenharmony_ci 14513498266Sopenharmony_ci#if defined(LIB526) 14613498266Sopenharmony_ci 14713498266Sopenharmony_ci /* test 526 and 528 */ 14813498266Sopenharmony_ci /* proper cleanup sequence - type PB */ 14913498266Sopenharmony_ci 15013498266Sopenharmony_ci for(i = 0; i < NUM_HANDLES; i++) { 15113498266Sopenharmony_ci curl_multi_remove_handle(m, curl[i]); 15213498266Sopenharmony_ci curl_easy_cleanup(curl[i]); 15313498266Sopenharmony_ci } 15413498266Sopenharmony_ci curl_multi_cleanup(m); 15513498266Sopenharmony_ci curl_global_cleanup(); 15613498266Sopenharmony_ci 15713498266Sopenharmony_ci#elif defined(LIB527) 15813498266Sopenharmony_ci 15913498266Sopenharmony_ci /* test 527 */ 16013498266Sopenharmony_ci 16113498266Sopenharmony_ci /* Upon non-failure test flow the easy's have already been cleanup'ed. In 16213498266Sopenharmony_ci case there is a failure we arrive here with easy's that have not been 16313498266Sopenharmony_ci cleanup'ed yet, in this case we have to cleanup them or otherwise these 16413498266Sopenharmony_ci will be leaked, let's use undocumented cleanup sequence - type UB */ 16513498266Sopenharmony_ci 16613498266Sopenharmony_ci if(res) 16713498266Sopenharmony_ci for(i = 0; i < NUM_HANDLES; i++) 16813498266Sopenharmony_ci curl_easy_cleanup(curl[i]); 16913498266Sopenharmony_ci 17013498266Sopenharmony_ci curl_multi_cleanup(m); 17113498266Sopenharmony_ci curl_global_cleanup(); 17213498266Sopenharmony_ci 17313498266Sopenharmony_ci#elif defined(LIB532) 17413498266Sopenharmony_ci 17513498266Sopenharmony_ci /* test 532 */ 17613498266Sopenharmony_ci /* undocumented cleanup sequence - type UB */ 17713498266Sopenharmony_ci 17813498266Sopenharmony_ci for(i = 0; i < NUM_HANDLES; i++) 17913498266Sopenharmony_ci curl_easy_cleanup(curl[i]); 18013498266Sopenharmony_ci curl_multi_cleanup(m); 18113498266Sopenharmony_ci curl_global_cleanup(); 18213498266Sopenharmony_ci 18313498266Sopenharmony_ci#endif 18413498266Sopenharmony_ci 18513498266Sopenharmony_ci return res; 18613498266Sopenharmony_ci} 187