|  | /*************************************************************************** | 
|  | *                                  _   _ ____  _ | 
|  | *  Project                     ___| | | |  _ \| | | 
|  | *                             / __| | | | |_) | | | 
|  | *                            | (__| |_| |  _ <| |___ | 
|  | *                             \___|\___/|_| \_\_____| | 
|  | * | 
|  | * Copyright (C) 1998 - 2022, Daniel Stenberg, <daniel@haxx.se>, et al. | 
|  | * | 
|  | * This software is licensed as described in the file COPYING, which | 
|  | * you should have received as part of this distribution. The terms | 
|  | * are also available at https://curl.se/docs/copyright.html. | 
|  | * | 
|  | * You may opt to use, copy, modify, merge, publish, distribute and/or sell | 
|  | * copies of the Software, and permit persons to whom the Software is | 
|  | * furnished to do so, under the terms of the COPYING file. | 
|  | * | 
|  | * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY | 
|  | * KIND, either express or implied. | 
|  | * | 
|  | * SPDX-License-Identifier: curl | 
|  | * | 
|  | ***************************************************************************/ | 
|  | #include "test.h" | 
|  |  | 
|  | /* test case and code based on https://github.com/curl/curl/issues/3927 */ | 
|  |  | 
|  | #include "testutil.h" | 
|  | #include "warnless.h" | 
|  | #include "memdebug.h" | 
|  |  | 
|  | static int dload_progress_cb(void *a, curl_off_t b, curl_off_t c, | 
|  | curl_off_t d, curl_off_t e) | 
|  | { | 
|  | (void)a; | 
|  | (void)b; | 
|  | (void)c; | 
|  | (void)d; | 
|  | (void)e; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static size_t write_cb(char *d, size_t n, size_t l, void *p) | 
|  | { | 
|  | /* take care of the data here, ignored in this example */ | 
|  | (void)d; | 
|  | (void)p; | 
|  | return n*l; | 
|  | } | 
|  |  | 
|  | static CURLcode run(CURL *hnd, long limit, long time) | 
|  | { | 
|  | curl_easy_setopt(hnd, CURLOPT_LOW_SPEED_LIMIT, limit); | 
|  | curl_easy_setopt(hnd, CURLOPT_LOW_SPEED_TIME, time); | 
|  | return curl_easy_perform(hnd); | 
|  | } | 
|  |  | 
|  | int test(char *URL) | 
|  | { | 
|  | CURLcode ret; | 
|  | CURL *hnd; | 
|  | char buffer[CURL_ERROR_SIZE]; | 
|  | curl_global_init(CURL_GLOBAL_ALL); | 
|  | hnd = curl_easy_init(); | 
|  | curl_easy_setopt(hnd, CURLOPT_URL, URL); | 
|  | curl_easy_setopt(hnd, CURLOPT_WRITEFUNCTION, write_cb); | 
|  | curl_easy_setopt(hnd, CURLOPT_ERRORBUFFER, buffer); | 
|  | curl_easy_setopt(hnd, CURLOPT_NOPROGRESS, 0L); | 
|  | curl_easy_setopt(hnd, CURLOPT_XFERINFOFUNCTION, dload_progress_cb); | 
|  |  | 
|  | printf("Start: %d\n", time(NULL)); | 
|  | ret = run(hnd, 1, 2); | 
|  | if(ret) | 
|  | fprintf(stderr, "error %d: %s\n", ret, buffer); | 
|  |  | 
|  | ret = run(hnd, 12000, 1); | 
|  | if(ret != CURLE_OPERATION_TIMEDOUT) | 
|  | fprintf(stderr, "error %d: %s\n", ret, buffer); | 
|  | else | 
|  | ret = CURLE_OK; | 
|  |  | 
|  | printf("End: %d\n", time(NULL)); | 
|  | curl_easy_cleanup(hnd); | 
|  | curl_global_cleanup(); | 
|  |  | 
|  | return (int)ret; | 
|  | } |