mirror of
https://github.com/moparisthebest/curl
synced 2024-11-10 03:25:04 -05:00
2f4fe0175b
part of the official libcurl API http://curl.haxx.se/lxr/source/lib/README.curlx The documented way of using them would be to use timeval.c as a source code file. The above described method works very well when statically linking libcurl and apps, test programs, but has several drawbacks when you build a true shared libcurl (i.e. Name space clash at linkage stage as functions are defined more than once. Windows makefiles are not capable of handling this system of source-level sharing) So... Now testutil.h and testutil.c define and implement tutil_tvnow and tutil_tvdiff which replace curlx_tvnow and curlx_tvdiff for the libtest programs. Doing this we avoid the above described problems, and the code in the testsuite does not impose the need to keep those functions public in libcurl even when not part of the API.
194 lines
4.8 KiB
C
194 lines
4.8 KiB
C
/*****************************************************************************
|
|
* _ _ ____ _
|
|
* Project ___| | | | _ \| |
|
|
* / __| | | | |_) | |
|
|
* | (__| |_| | _ <| |___
|
|
* \___|\___/|_| \_\_____|
|
|
*
|
|
* $Id$
|
|
*/
|
|
|
|
#include "test.h"
|
|
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <fcntl.h>
|
|
|
|
#include "testutil.h"
|
|
|
|
#define MAIN_LOOP_HANG_TIMEOUT 300 * 1000
|
|
#define MULTI_PERFORM_HANG_TIMEOUT 120 * 1000
|
|
|
|
int test(char *URL)
|
|
{
|
|
int res = 0;
|
|
CURL *curl;
|
|
FILE *hd_src ;
|
|
int hd ;
|
|
struct_stat file_info;
|
|
int running;
|
|
char done=FALSE;
|
|
CURLM *m;
|
|
struct timeval ml_start;
|
|
struct timeval mp_start;
|
|
char ml_timedout = FALSE;
|
|
char mp_timedout = FALSE;
|
|
|
|
if (!arg2) {
|
|
fprintf(stderr, "Usage: lib525 [url] [uploadfile]\n");
|
|
return -1;
|
|
}
|
|
|
|
/* get the file size of the local file */
|
|
hd = open(arg2, O_RDONLY) ;
|
|
fstat(hd, &file_info);
|
|
close(hd) ;
|
|
|
|
/* get a FILE * of the same file, could also be made with
|
|
fdopen() from the previous descriptor, but hey this is just
|
|
an example! */
|
|
hd_src = fopen(arg2, "rb");
|
|
|
|
if (curl_global_init(CURL_GLOBAL_ALL) != CURLE_OK) {
|
|
fprintf(stderr, "curl_global_init() failed\n");
|
|
fclose(hd_src);
|
|
return TEST_ERR_MAJOR_BAD;
|
|
}
|
|
|
|
if ((curl = curl_easy_init()) == NULL) {
|
|
fprintf(stderr, "curl_easy_init() failed\n");
|
|
fclose(hd_src);
|
|
curl_global_cleanup();
|
|
return TEST_ERR_MAJOR_BAD;
|
|
}
|
|
|
|
/* enable uploading */
|
|
curl_easy_setopt(curl, CURLOPT_UPLOAD, TRUE) ;
|
|
|
|
/* specify target */
|
|
curl_easy_setopt(curl,CURLOPT_URL, URL);
|
|
|
|
/* go verbose */
|
|
curl_easy_setopt(curl, CURLOPT_VERBOSE, 1);
|
|
|
|
/* use active FTP */
|
|
curl_easy_setopt(curl, CURLOPT_FTPPORT, "-");
|
|
|
|
/* now specify which file to upload */
|
|
curl_easy_setopt(curl, CURLOPT_READDATA, hd_src);
|
|
|
|
/* NOTE: if you want this code to work on Windows with libcurl as a DLL, you
|
|
MUST also provide a read callback with CURLOPT_READFUNCTION. Failing to
|
|
do so will give you a crash since a DLL may not use the variable's memory
|
|
when passed in to it from an app like this. */
|
|
|
|
/* Set the size of the file to upload (optional). If you give a *_LARGE
|
|
option you MUST make sure that the type of the passed-in argument is a
|
|
curl_off_t. If you use CURLOPT_INFILESIZE (without _LARGE) you must
|
|
make sure that to pass in a type 'long' argument. */
|
|
curl_easy_setopt(curl, CURLOPT_INFILESIZE_LARGE,
|
|
(curl_off_t)file_info.st_size);
|
|
|
|
if ((m = curl_multi_init()) == NULL) {
|
|
fprintf(stderr, "curl_multi_init() failed\n");
|
|
curl_easy_cleanup(curl);
|
|
curl_global_cleanup();
|
|
fclose(hd_src);
|
|
return TEST_ERR_MAJOR_BAD;
|
|
}
|
|
|
|
if ((res = (int)curl_multi_add_handle(m, curl)) != CURLM_OK) {
|
|
fprintf(stderr, "curl_multi_add_handle() failed, "
|
|
"with code %d\n", res);
|
|
curl_multi_cleanup(m);
|
|
curl_easy_cleanup(curl);
|
|
curl_global_cleanup();
|
|
fclose(hd_src);
|
|
return TEST_ERR_MAJOR_BAD;
|
|
}
|
|
|
|
ml_timedout = FALSE;
|
|
ml_start = tutil_tvnow();
|
|
|
|
while (!done) {
|
|
fd_set rd, wr, exc;
|
|
int max_fd;
|
|
struct timeval interval;
|
|
|
|
interval.tv_sec = 1;
|
|
interval.tv_usec = 0;
|
|
|
|
if (tutil_tvdiff(tutil_tvnow(), ml_start) >
|
|
MAIN_LOOP_HANG_TIMEOUT) {
|
|
ml_timedout = TRUE;
|
|
break;
|
|
}
|
|
mp_timedout = FALSE;
|
|
mp_start = tutil_tvnow();
|
|
|
|
while (res == CURLM_CALL_MULTI_PERFORM) {
|
|
res = (int)curl_multi_perform(m, &running);
|
|
if (tutil_tvdiff(tutil_tvnow(), mp_start) >
|
|
MULTI_PERFORM_HANG_TIMEOUT) {
|
|
mp_timedout = TRUE;
|
|
break;
|
|
}
|
|
if (running <= 0) {
|
|
done = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
if (mp_timedout || done)
|
|
break;
|
|
|
|
if (res != CURLM_OK) {
|
|
fprintf(stderr, "not okay???\n");
|
|
break;
|
|
}
|
|
|
|
FD_ZERO(&rd);
|
|
FD_ZERO(&wr);
|
|
FD_ZERO(&exc);
|
|
max_fd = 0;
|
|
|
|
if (curl_multi_fdset(m, &rd, &wr, &exc, &max_fd) != CURLM_OK) {
|
|
fprintf(stderr, "unexpected failured of fdset.\n");
|
|
res = 189;
|
|
break;
|
|
}
|
|
|
|
if (select_test(max_fd+1, &rd, &wr, &exc, &interval) == -1) {
|
|
fprintf(stderr, "bad select??\n");
|
|
res = 195;
|
|
break;
|
|
}
|
|
|
|
res = CURLM_CALL_MULTI_PERFORM;
|
|
}
|
|
|
|
if (ml_timedout || mp_timedout) {
|
|
if (ml_timedout) fprintf(stderr, "ml_timedout\n");
|
|
if (mp_timedout) fprintf(stderr, "mp_timedout\n");
|
|
fprintf(stderr, "ABORTING TEST, since it seems "
|
|
"that it would have run forever.\n");
|
|
res = TEST_ERR_RUNS_FOREVER;
|
|
}
|
|
|
|
#ifdef LIB529
|
|
/* test 529 */
|
|
curl_multi_remove_handle(m, curl);
|
|
curl_multi_cleanup(m);
|
|
curl_easy_cleanup(curl);
|
|
#else
|
|
/* test 525 */
|
|
curl_multi_remove_handle(m, curl);
|
|
curl_easy_cleanup(curl);
|
|
curl_multi_cleanup(m);
|
|
#endif
|
|
|
|
fclose(hd_src); /* close the local file */
|
|
|
|
curl_global_cleanup();
|
|
return res;
|
|
}
|