|  | /*************************************************************************** | 
|  | *                                  _   _ ____  _ | 
|  | *  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 | 
|  | * | 
|  | ***************************************************************************/ | 
|  | /* | 
|  | * See https://github.com/curl/curl/issues/3371 | 
|  | * | 
|  | * This test case checks whether curl_multi_remove_handle() cancels | 
|  | * asynchronous DNS resolvers without blocking where possible.  Obviously, it | 
|  | * only tests whichever resolver cURL is actually built with. | 
|  | */ | 
|  |  | 
|  | /* We're willing to wait a very generous two seconds for the removal.  This is | 
|  | as low as we can go while still easily supporting SIGALRM timing for the | 
|  | non-threaded blocking resolver.  It doesn't matter that much because when | 
|  | the test passes, we never wait this long. */ | 
|  | #define TEST_HANG_TIMEOUT 2 * 1000 | 
|  |  | 
|  | #include "test.h" | 
|  | #include "testutil.h" | 
|  |  | 
|  | #include <sys/stat.h> | 
|  |  | 
|  | int test(char *URL) | 
|  | { | 
|  | int stillRunning; | 
|  | CURLM *multiHandle = NULL; | 
|  | CURL *curl = NULL; | 
|  | CURLcode res = CURLE_OK; | 
|  | CURLMcode mres; | 
|  | int timeout; | 
|  |  | 
|  | global_init(CURL_GLOBAL_ALL); | 
|  |  | 
|  | multi_init(multiHandle); | 
|  |  | 
|  | easy_init(curl); | 
|  |  | 
|  | easy_setopt(curl, CURLOPT_VERBOSE, 1L); | 
|  | easy_setopt(curl, CURLOPT_URL, URL); | 
|  |  | 
|  | /* Set a DNS server that hopefully will not respond when using c-ares. */ | 
|  | if(curl_easy_setopt(curl, CURLOPT_DNS_SERVERS, "0.0.0.0") == CURLE_OK) | 
|  | /* Since we could set the DNS server, presume we are working with a | 
|  | resolver that can be cancelled (i.e. c-ares).  Thus, | 
|  | curl_multi_remove_handle() should not block even when the resolver | 
|  | request is outstanding.  So, set a request timeout _longer_ than the | 
|  | test hang timeout so we will fail if the handle removal call incorrectly | 
|  | blocks. */ | 
|  | timeout = TEST_HANG_TIMEOUT * 2; | 
|  | else { | 
|  | /* If we can't set the DNS server, presume that we are configured to use a | 
|  | resolver that can't be cancelled (i.e. the threaded resolver or the | 
|  | non-threaded blocking resolver).  So, we just test that the | 
|  | curl_multi_remove_handle() call does finish well within our test | 
|  | timeout. | 
|  |  | 
|  | But, it is very unlikely that the resolver request will take any time at | 
|  | all because we haven't been able to configure the resolver to use an | 
|  | non-responsive DNS server.  At least we exercise the flow. | 
|  | */ | 
|  | fprintf(stderr, | 
|  | "CURLOPT_DNS_SERVERS not supported; " | 
|  | "assuming curl_multi_remove_handle() will block\n"); | 
|  | timeout = TEST_HANG_TIMEOUT / 2; | 
|  | } | 
|  |  | 
|  | /* Setting a timeout on the request should ensure that even if we have to | 
|  | wait for the resolver during curl_multi_remove_handle(), it won't take | 
|  | longer than this, because the resolver request inherits its timeout from | 
|  | this. */ | 
|  | easy_setopt(curl, CURLOPT_TIMEOUT_MS, timeout); | 
|  |  | 
|  | multi_add_handle(multiHandle, curl); | 
|  |  | 
|  | /* This should move the handle from INIT => CONNECT => WAITRESOLVE. */ | 
|  | fprintf(stderr, "curl_multi_perform()...\n"); | 
|  | multi_perform(multiHandle, &stillRunning); | 
|  | fprintf(stderr, "curl_multi_perform() succeeded\n"); | 
|  |  | 
|  | /* Start measuring how long it takes to remove the handle. */ | 
|  | fprintf(stderr, "curl_multi_remove_handle()...\n"); | 
|  | start_test_timing(); | 
|  | mres = curl_multi_remove_handle(multiHandle, curl); | 
|  | if(mres) { | 
|  | fprintf(stderr, "curl_multi_remove_handle() failed, " | 
|  | "with code %d\n", (int)res); | 
|  | res = TEST_ERR_MULTI; | 
|  | goto test_cleanup; | 
|  | } | 
|  | fprintf(stderr, "curl_multi_remove_handle() succeeded\n"); | 
|  |  | 
|  | /* Fail the test if it took too long to remove.  This happens after the fact, | 
|  | and says "it seems that it would have run forever", which isn't true, but | 
|  | it's close enough, and simple to do. */ | 
|  | abort_on_test_timeout(); | 
|  |  | 
|  | test_cleanup: | 
|  | curl_easy_cleanup(curl); | 
|  | curl_multi_cleanup(multiHandle); | 
|  | curl_global_cleanup(); | 
|  |  | 
|  | return (int)res; | 
|  | } |