Annotation of embedaddon/curl/src/tool_operate.c, revision 1.1
1.1 ! misho 1: /***************************************************************************
! 2: * _ _ ____ _
! 3: * Project ___| | | | _ \| |
! 4: * / __| | | | |_) | |
! 5: * | (__| |_| | _ <| |___
! 6: * \___|\___/|_| \_\_____|
! 7: *
! 8: * Copyright (C) 1998 - 2020, Daniel Stenberg, <daniel@haxx.se>, et al.
! 9: *
! 10: * This software is licensed as described in the file COPYING, which
! 11: * you should have received as part of this distribution. The terms
! 12: * are also available at https://curl.haxx.se/docs/copyright.html.
! 13: *
! 14: * You may opt to use, copy, modify, merge, publish, distribute and/or sell
! 15: * copies of the Software, and permit persons to whom the Software is
! 16: * furnished to do so, under the terms of the COPYING file.
! 17: *
! 18: * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
! 19: * KIND, either express or implied.
! 20: *
! 21: ***************************************************************************/
! 22: #include "tool_setup.h"
! 23:
! 24: #ifdef HAVE_FCNTL_H
! 25: # include <fcntl.h>
! 26: #endif
! 27:
! 28: #ifdef HAVE_LOCALE_H
! 29: # include <locale.h>
! 30: #endif
! 31:
! 32: #ifdef HAVE_SYS_SELECT_H
! 33: # include <sys/select.h>
! 34: #elif defined(HAVE_UNISTD_H)
! 35: # include <unistd.h>
! 36: #endif
! 37:
! 38: #ifdef __VMS
! 39: # include <fabdef.h>
! 40: #endif
! 41:
! 42: #ifdef __AMIGA__
! 43: # include <proto/dos.h>
! 44: #endif
! 45:
! 46: #include "strcase.h"
! 47:
! 48: #define ENABLE_CURLX_PRINTF
! 49: /* use our own printf() functions */
! 50: #include "curlx.h"
! 51:
! 52: #include "tool_binmode.h"
! 53: #include "tool_cfgable.h"
! 54: #include "tool_cb_dbg.h"
! 55: #include "tool_cb_hdr.h"
! 56: #include "tool_cb_prg.h"
! 57: #include "tool_cb_rea.h"
! 58: #include "tool_cb_see.h"
! 59: #include "tool_cb_wrt.h"
! 60: #include "tool_dirhie.h"
! 61: #include "tool_doswin.h"
! 62: #include "tool_easysrc.h"
! 63: #include "tool_filetime.h"
! 64: #include "tool_getparam.h"
! 65: #include "tool_helpers.h"
! 66: #include "tool_homedir.h"
! 67: #include "tool_libinfo.h"
! 68: #include "tool_main.h"
! 69: #include "tool_metalink.h"
! 70: #include "tool_msgs.h"
! 71: #include "tool_operate.h"
! 72: #include "tool_operhlp.h"
! 73: #include "tool_paramhlp.h"
! 74: #include "tool_parsecfg.h"
! 75: #include "tool_setopt.h"
! 76: #include "tool_sleep.h"
! 77: #include "tool_urlglob.h"
! 78: #include "tool_util.h"
! 79: #include "tool_writeout.h"
! 80: #include "tool_xattr.h"
! 81: #include "tool_vms.h"
! 82: #include "tool_help.h"
! 83: #include "tool_hugehelp.h"
! 84: #include "tool_progress.h"
! 85:
! 86: #include "memdebug.h" /* keep this as LAST include */
! 87:
! 88: #ifdef CURLDEBUG
! 89: /* libcurl's debug builds provide an extra function */
! 90: CURLcode curl_easy_perform_ev(CURL *easy);
! 91: #endif
! 92:
! 93: #define CURLseparator "--_curl_--"
! 94:
! 95: #ifndef O_BINARY
! 96: /* since O_BINARY as used in bitmasks, setting it to zero makes it usable in
! 97: source code but yet it doesn't ruin anything */
! 98: # define O_BINARY 0
! 99: #endif
! 100:
! 101: #define CURL_CA_CERT_ERRORMSG \
! 102: "More details here: https://curl.haxx.se/docs/sslcerts.html\n\n" \
! 103: "curl failed to verify the legitimacy of the server and therefore " \
! 104: "could not\nestablish a secure connection to it. To learn more about " \
! 105: "this situation and\nhow to fix it, please visit the web page mentioned " \
! 106: "above.\n"
! 107:
! 108: static CURLcode single_transfer(struct GlobalConfig *global,
! 109: struct OperationConfig *config,
! 110: CURLSH *share,
! 111: bool capath_from_env,
! 112: bool *added);
! 113: static CURLcode create_transfer(struct GlobalConfig *global,
! 114: CURLSH *share,
! 115: bool *added);
! 116:
! 117: static bool is_fatal_error(CURLcode code)
! 118: {
! 119: switch(code) {
! 120: case CURLE_FAILED_INIT:
! 121: case CURLE_OUT_OF_MEMORY:
! 122: case CURLE_UNKNOWN_OPTION:
! 123: case CURLE_FUNCTION_NOT_FOUND:
! 124: case CURLE_BAD_FUNCTION_ARGUMENT:
! 125: /* critical error */
! 126: return TRUE;
! 127: default:
! 128: break;
! 129: }
! 130:
! 131: /* no error or not critical */
! 132: return FALSE;
! 133: }
! 134:
! 135: /*
! 136: * Check if a given string is a PKCS#11 URI
! 137: */
! 138: static bool is_pkcs11_uri(const char *string)
! 139: {
! 140: if(curl_strnequal(string, "pkcs11:", 7)) {
! 141: return TRUE;
! 142: }
! 143: else {
! 144: return FALSE;
! 145: }
! 146: }
! 147:
! 148: #ifdef __VMS
! 149: /*
! 150: * get_vms_file_size does what it takes to get the real size of the file
! 151: *
! 152: * For fixed files, find out the size of the EOF block and adjust.
! 153: *
! 154: * For all others, have to read the entire file in, discarding the contents.
! 155: * Most posted text files will be small, and binary files like zlib archives
! 156: * and CD/DVD images should be either a STREAM_LF format or a fixed format.
! 157: *
! 158: */
! 159: static curl_off_t vms_realfilesize(const char *name,
! 160: const struct_stat *stat_buf)
! 161: {
! 162: char buffer[8192];
! 163: curl_off_t count;
! 164: int ret_stat;
! 165: FILE * file;
! 166:
! 167: /* !checksrc! disable FOPENMODE 1 */
! 168: file = fopen(name, "r"); /* VMS */
! 169: if(file == NULL) {
! 170: return 0;
! 171: }
! 172: count = 0;
! 173: ret_stat = 1;
! 174: while(ret_stat > 0) {
! 175: ret_stat = fread(buffer, 1, sizeof(buffer), file);
! 176: if(ret_stat != 0)
! 177: count += ret_stat;
! 178: }
! 179: fclose(file);
! 180:
! 181: return count;
! 182: }
! 183:
! 184: /*
! 185: *
! 186: * VmsSpecialSize checks to see if the stat st_size can be trusted and
! 187: * if not to call a routine to get the correct size.
! 188: *
! 189: */
! 190: static curl_off_t VmsSpecialSize(const char *name,
! 191: const struct_stat *stat_buf)
! 192: {
! 193: switch(stat_buf->st_fab_rfm) {
! 194: case FAB$C_VAR:
! 195: case FAB$C_VFC:
! 196: return vms_realfilesize(name, stat_buf);
! 197: break;
! 198: default:
! 199: return stat_buf->st_size;
! 200: }
! 201: }
! 202: #endif /* __VMS */
! 203:
! 204: #define BUFFER_SIZE (100*1024)
! 205:
! 206: struct per_transfer *transfers; /* first node */
! 207: static struct per_transfer *transfersl; /* last node */
! 208:
! 209: /* add_per_transfer creates a new 'per_transfer' node in the linked
! 210: list of transfers */
! 211: static CURLcode add_per_transfer(struct per_transfer **per)
! 212: {
! 213: struct per_transfer *p;
! 214: p = calloc(sizeof(struct per_transfer), 1);
! 215: if(!p)
! 216: return CURLE_OUT_OF_MEMORY;
! 217: if(!transfers)
! 218: /* first entry */
! 219: transfersl = transfers = p;
! 220: else {
! 221: /* make the last node point to the new node */
! 222: transfersl->next = p;
! 223: /* make the new node point back to the formerly last node */
! 224: p->prev = transfersl;
! 225: /* move the last node pointer to the new entry */
! 226: transfersl = p;
! 227: }
! 228: *per = p;
! 229: all_xfers++; /* count total number of transfers added */
! 230: return CURLE_OK;
! 231: }
! 232:
! 233: /* Remove the specified transfer from the list (and free it), return the next
! 234: in line */
! 235: static struct per_transfer *del_per_transfer(struct per_transfer *per)
! 236: {
! 237: struct per_transfer *n;
! 238: struct per_transfer *p;
! 239: DEBUGASSERT(transfers);
! 240: DEBUGASSERT(transfersl);
! 241: DEBUGASSERT(per);
! 242:
! 243: n = per->next;
! 244: p = per->prev;
! 245:
! 246: if(p)
! 247: p->next = n;
! 248: else
! 249: transfers = n;
! 250:
! 251: if(n)
! 252: n->prev = p;
! 253: else
! 254: transfersl = p;
! 255:
! 256: free(per);
! 257:
! 258: return n;
! 259: }
! 260:
! 261: static CURLcode pre_transfer(struct GlobalConfig *global,
! 262: struct per_transfer *per)
! 263: {
! 264: curl_off_t uploadfilesize = -1;
! 265: struct_stat fileinfo;
! 266: CURLcode result = CURLE_OK;
! 267:
! 268: if(per->separator_err)
! 269: fprintf(global->errors, "%s\n", per->separator_err);
! 270: if(per->separator)
! 271: printf("%s\n", per->separator);
! 272:
! 273: if(per->uploadfile && !stdin_upload(per->uploadfile)) {
! 274: /* VMS Note:
! 275: *
! 276: * Reading binary from files can be a problem... Only FIXED, VAR
! 277: * etc WITHOUT implied CC will work Others need a \n appended to a
! 278: * line
! 279: *
! 280: * - Stat gives a size but this is UNRELIABLE in VMS As a f.e. a
! 281: * fixed file with implied CC needs to have a byte added for every
! 282: * record processed, this can by derived from Filesize & recordsize
! 283: * for VARiable record files the records need to be counted! for
! 284: * every record add 1 for linefeed and subtract 2 for the record
! 285: * header for VARIABLE header files only the bare record data needs
! 286: * to be considered with one appended if implied CC
! 287: */
! 288: #ifdef __VMS
! 289: /* Calculate the real upload size for VMS */
! 290: per->infd = -1;
! 291: if(stat(per->uploadfile, &fileinfo) == 0) {
! 292: fileinfo.st_size = VmsSpecialSize(uploadfile, &fileinfo);
! 293: switch(fileinfo.st_fab_rfm) {
! 294: case FAB$C_VAR:
! 295: case FAB$C_VFC:
! 296: case FAB$C_STMCR:
! 297: per->infd = open(per->uploadfile, O_RDONLY | O_BINARY);
! 298: break;
! 299: default:
! 300: per->infd = open(per->uploadfile, O_RDONLY | O_BINARY,
! 301: "rfm=stmlf", "ctx=stm");
! 302: }
! 303: }
! 304: if(per->infd == -1)
! 305: #else
! 306: per->infd = open(per->uploadfile, O_RDONLY | O_BINARY);
! 307: if((per->infd == -1) || fstat(per->infd, &fileinfo))
! 308: #endif
! 309: {
! 310: helpf(global->errors, "Can't open '%s'!\n", per->uploadfile);
! 311: if(per->infd != -1) {
! 312: close(per->infd);
! 313: per->infd = STDIN_FILENO;
! 314: }
! 315: return CURLE_READ_ERROR;
! 316: }
! 317: per->infdopen = TRUE;
! 318:
! 319: /* we ignore file size for char/block devices, sockets, etc. */
! 320: if(S_ISREG(fileinfo.st_mode))
! 321: uploadfilesize = fileinfo.st_size;
! 322:
! 323: if(uploadfilesize != -1)
! 324: my_setopt(per->curl, CURLOPT_INFILESIZE_LARGE, uploadfilesize);
! 325: per->input.fd = per->infd;
! 326: }
! 327: return result;
! 328: }
! 329:
! 330: /*
! 331: * Call this after a transfer has completed.
! 332: */
! 333: static CURLcode post_per_transfer(struct GlobalConfig *global,
! 334: struct per_transfer *per,
! 335: CURLcode result,
! 336: bool *retryp)
! 337: {
! 338: struct OutStruct *outs = &per->outs;
! 339: CURL *curl = per->curl;
! 340: struct OperationConfig *config = per->config;
! 341:
! 342: if(!curl || !config)
! 343: return result;
! 344:
! 345: *retryp = FALSE;
! 346:
! 347: if(per->infdopen)
! 348: close(per->infd);
! 349:
! 350: #ifdef __VMS
! 351: if(is_vms_shell()) {
! 352: /* VMS DCL shell behavior */
! 353: if(!global->showerror)
! 354: vms_show = VMSSTS_HIDE;
! 355: }
! 356: else
! 357: #endif
! 358: if(config->synthetic_error) {
! 359: ;
! 360: }
! 361: else if(result && global->showerror) {
! 362: fprintf(global->errors, "curl: (%d) %s\n", result,
! 363: (per->errorbuffer[0]) ? per->errorbuffer :
! 364: curl_easy_strerror(result));
! 365: if(result == CURLE_PEER_FAILED_VERIFICATION)
! 366: fputs(CURL_CA_CERT_ERRORMSG, global->errors);
! 367: }
! 368:
! 369: /* Set file extended attributes */
! 370: if(!result && config->xattr && outs->fopened && outs->stream) {
! 371: int rc = fwrite_xattr(curl, fileno(outs->stream));
! 372: if(rc)
! 373: warnf(config->global, "Error setting extended attributes: %s\n",
! 374: strerror(errno));
! 375: }
! 376:
! 377: if(!result && !outs->stream && !outs->bytes) {
! 378: /* we have received no data despite the transfer was successful
! 379: ==> force cration of an empty output file (if an output file
! 380: was specified) */
! 381: long cond_unmet = 0L;
! 382: /* do not create (or even overwrite) the file in case we get no
! 383: data because of unmet condition */
! 384: curl_easy_getinfo(curl, CURLINFO_CONDITION_UNMET, &cond_unmet);
! 385: if(!cond_unmet && !tool_create_output_file(outs, config))
! 386: result = CURLE_WRITE_ERROR;
! 387: }
! 388:
! 389: if(!outs->s_isreg && outs->stream) {
! 390: /* Dump standard stream buffered data */
! 391: int rc = fflush(outs->stream);
! 392: if(!result && rc) {
! 393: /* something went wrong in the writing process */
! 394: result = CURLE_WRITE_ERROR;
! 395: fprintf(global->errors, "(%d) Failed writing body\n", result);
! 396: }
! 397: }
! 398:
! 399: #ifdef USE_METALINK
! 400: if(per->metalink && !per->metalink_next_res)
! 401: fprintf(global->errors, "Metalink: fetching (%s) from (%s) OK\n",
! 402: per->mlfile->filename, per->this_url);
! 403:
! 404: if(!per->metalink && config->use_metalink && result == CURLE_OK) {
! 405: int rv = parse_metalink(config, outs, per->this_url);
! 406: if(!rv) {
! 407: fprintf(config->global->errors, "Metalink: parsing (%s) OK\n",
! 408: per->this_url);
! 409: }
! 410: else if(rv == -1)
! 411: fprintf(config->global->errors, "Metalink: parsing (%s) FAILED\n",
! 412: per->this_url);
! 413: }
! 414: else if(per->metalink && result == CURLE_OK && !per->metalink_next_res) {
! 415: int rv;
! 416: (void)fflush(outs->stream);
! 417: rv = metalink_check_hash(global, per->mlfile, outs->filename);
! 418: if(!rv)
! 419: per->metalink_next_res = 1;
! 420: }
! 421: #endif /* USE_METALINK */
! 422:
! 423: #ifdef USE_METALINK
! 424: if(outs->metalink_parser)
! 425: metalink_parser_context_delete(outs->metalink_parser);
! 426: #endif /* USE_METALINK */
! 427:
! 428: if(outs->is_cd_filename && outs->stream && !global->mute &&
! 429: outs->filename)
! 430: printf("curl: Saved to filename '%s'\n", outs->filename);
! 431:
! 432: /* if retry-max-time is non-zero, make sure we haven't exceeded the
! 433: time */
! 434: if(per->retry_numretries &&
! 435: (!config->retry_maxtime ||
! 436: (tvdiff(tvnow(), per->retrystart) <
! 437: config->retry_maxtime*1000L)) ) {
! 438: enum {
! 439: RETRY_NO,
! 440: RETRY_TIMEOUT,
! 441: RETRY_CONNREFUSED,
! 442: RETRY_HTTP,
! 443: RETRY_FTP,
! 444: RETRY_LAST /* not used */
! 445: } retry = RETRY_NO;
! 446: long response;
! 447: if((CURLE_OPERATION_TIMEDOUT == result) ||
! 448: (CURLE_COULDNT_RESOLVE_HOST == result) ||
! 449: (CURLE_COULDNT_RESOLVE_PROXY == result) ||
! 450: (CURLE_FTP_ACCEPT_TIMEOUT == result))
! 451: /* retry timeout always */
! 452: retry = RETRY_TIMEOUT;
! 453: else if(config->retry_connrefused &&
! 454: (CURLE_COULDNT_CONNECT == result)) {
! 455: long oserrno;
! 456: curl_easy_getinfo(curl, CURLINFO_OS_ERRNO, &oserrno);
! 457: if(ECONNREFUSED == oserrno)
! 458: retry = RETRY_CONNREFUSED;
! 459: }
! 460: else if((CURLE_OK == result) ||
! 461: (config->failonerror &&
! 462: (CURLE_HTTP_RETURNED_ERROR == result))) {
! 463: /* If it returned OK. _or_ failonerror was enabled and it
! 464: returned due to such an error, check for HTTP transient
! 465: errors to retry on. */
! 466: long protocol;
! 467: curl_easy_getinfo(curl, CURLINFO_PROTOCOL, &protocol);
! 468: if((protocol == CURLPROTO_HTTP) || (protocol == CURLPROTO_HTTPS)) {
! 469: /* This was HTTP(S) */
! 470: curl_easy_getinfo(curl, CURLINFO_RESPONSE_CODE, &response);
! 471:
! 472: switch(response) {
! 473: case 429: /* Too Many Requests (RFC6585) */
! 474: case 500: /* Internal Server Error */
! 475: case 502: /* Bad Gateway */
! 476: case 503: /* Service Unavailable */
! 477: case 504: /* Gateway Timeout */
! 478: retry = RETRY_HTTP;
! 479: /*
! 480: * At this point, we have already written data to the output
! 481: * file (or terminal). If we write to a file, we must rewind
! 482: * or close/re-open the file so that the next attempt starts
! 483: * over from the beginning.
! 484: *
! 485: * TODO: similar action for the upload case. We might need
! 486: * to start over reading from a previous point if we have
! 487: * uploaded something when this was returned.
! 488: */
! 489: break;
! 490: }
! 491: }
! 492: } /* if CURLE_OK */
! 493: else if(result) {
! 494: long protocol;
! 495:
! 496: curl_easy_getinfo(curl, CURLINFO_RESPONSE_CODE, &response);
! 497: curl_easy_getinfo(curl, CURLINFO_PROTOCOL, &protocol);
! 498:
! 499: if((protocol == CURLPROTO_FTP || protocol == CURLPROTO_FTPS) &&
! 500: response / 100 == 4)
! 501: /*
! 502: * This is typically when the FTP server only allows a certain
! 503: * amount of users and we are not one of them. All 4xx codes
! 504: * are transient.
! 505: */
! 506: retry = RETRY_FTP;
! 507: }
! 508:
! 509: if(retry) {
! 510: long sleeptime = 0;
! 511: curl_off_t retry_after = 0;
! 512: static const char * const m[]={
! 513: NULL,
! 514: "timeout",
! 515: "connection refused",
! 516: "HTTP error",
! 517: "FTP error"
! 518: };
! 519:
! 520: sleeptime = per->retry_sleep;
! 521: if(RETRY_HTTP == retry) {
! 522: curl_easy_getinfo(curl, CURLINFO_RETRY_AFTER, &retry_after);
! 523: if(retry_after) {
! 524: /* store in a 'long', make sure it doesn't overflow */
! 525: if(retry_after > LONG_MAX/1000)
! 526: sleeptime = LONG_MAX;
! 527: else
! 528: sleeptime = (long)retry_after * 1000; /* milliseconds */
! 529: }
! 530: }
! 531: warnf(config->global, "Transient problem: %s "
! 532: "Will retry in %ld seconds. "
! 533: "%ld retries left.\n",
! 534: m[retry], sleeptime/1000L, per->retry_numretries);
! 535:
! 536: per->retry_numretries--;
! 537: tool_go_sleep(sleeptime);
! 538: if(!config->retry_delay) {
! 539: per->retry_sleep *= 2;
! 540: if(per->retry_sleep > RETRY_SLEEP_MAX)
! 541: per->retry_sleep = RETRY_SLEEP_MAX;
! 542: }
! 543: if(outs->bytes && outs->filename && outs->stream) {
! 544: int rc;
! 545: /* We have written data to a output file, we truncate file
! 546: */
! 547: if(!global->mute)
! 548: fprintf(global->errors, "Throwing away %"
! 549: CURL_FORMAT_CURL_OFF_T " bytes\n",
! 550: outs->bytes);
! 551: fflush(outs->stream);
! 552: /* truncate file at the position where we started appending */
! 553: #ifdef HAVE_FTRUNCATE
! 554: if(ftruncate(fileno(outs->stream), outs->init)) {
! 555: /* when truncate fails, we can't just append as then we'll
! 556: create something strange, bail out */
! 557: if(!global->mute)
! 558: fprintf(global->errors,
! 559: "failed to truncate, exiting\n");
! 560: return CURLE_WRITE_ERROR;
! 561: }
! 562: /* now seek to the end of the file, the position where we
! 563: just truncated the file in a large file-safe way */
! 564: rc = fseek(outs->stream, 0, SEEK_END);
! 565: #else
! 566: /* ftruncate is not available, so just reposition the file
! 567: to the location we would have truncated it. This won't
! 568: work properly with large files on 32-bit systems, but
! 569: most of those will have ftruncate. */
! 570: rc = fseek(outs->stream, (long)outs->init, SEEK_SET);
! 571: #endif
! 572: if(rc) {
! 573: if(!global->mute)
! 574: fprintf(global->errors,
! 575: "failed seeking to end of file, exiting\n");
! 576: return CURLE_WRITE_ERROR;
! 577: }
! 578: outs->bytes = 0; /* clear for next round */
! 579: }
! 580: *retryp = TRUE; /* curl_easy_perform loop */
! 581: return CURLE_OK;
! 582: }
! 583: } /* if retry_numretries */
! 584: else if(per->metalink) {
! 585: /* Metalink: Decide to try the next resource or not. Try the next resource
! 586: if download was not successful. */
! 587: long response;
! 588: if(CURLE_OK == result) {
! 589: /* TODO We want to try next resource when download was
! 590: not successful. How to know that? */
! 591: char *effective_url = NULL;
! 592: curl_easy_getinfo(curl, CURLINFO_EFFECTIVE_URL, &effective_url);
! 593: if(effective_url &&
! 594: curl_strnequal(effective_url, "http", 4)) {
! 595: /* This was HTTP(S) */
! 596: curl_easy_getinfo(curl, CURLINFO_RESPONSE_CODE, &response);
! 597: if(response != 200 && response != 206) {
! 598: per->metalink_next_res = 1;
! 599: fprintf(global->errors,
! 600: "Metalink: fetching (%s) from (%s) FAILED "
! 601: "(HTTP status code %ld)\n",
! 602: per->mlfile->filename, per->this_url, response);
! 603: }
! 604: }
! 605: }
! 606: else {
! 607: per->metalink_next_res = 1;
! 608: fprintf(global->errors,
! 609: "Metalink: fetching (%s) from (%s) FAILED (%s)\n",
! 610: per->mlfile->filename, per->this_url,
! 611: curl_easy_strerror(result));
! 612: }
! 613: }
! 614:
! 615: if((global->progressmode == CURL_PROGRESS_BAR) &&
! 616: per->progressbar.calls)
! 617: /* if the custom progress bar has been displayed, we output a
! 618: newline here */
! 619: fputs("\n", per->progressbar.out);
! 620:
! 621: if(config->writeout)
! 622: ourWriteOut(per->curl, &per->outs, config->writeout);
! 623:
! 624: /* Close the outs file */
! 625: if(outs->fopened && outs->stream) {
! 626: int rc = fclose(outs->stream);
! 627: if(!result && rc) {
! 628: /* something went wrong in the writing process */
! 629: result = CURLE_WRITE_ERROR;
! 630: fprintf(global->errors, "(%d) Failed writing body\n", result);
! 631: }
! 632: }
! 633:
! 634: /* File time can only be set _after_ the file has been closed */
! 635: if(!result && config->remote_time && outs->s_isreg && outs->filename) {
! 636: /* Ask libcurl if we got a remote file time */
! 637: curl_off_t filetime = -1;
! 638: curl_easy_getinfo(curl, CURLINFO_FILETIME_T, &filetime);
! 639: setfiletime(filetime, outs->filename, config->global->errors);
! 640: }
! 641:
! 642: /* Close function-local opened file descriptors */
! 643: if(per->heads.fopened && per->heads.stream)
! 644: fclose(per->heads.stream);
! 645:
! 646: if(per->heads.alloc_filename)
! 647: Curl_safefree(per->heads.filename);
! 648:
! 649: if(per->etag_save.fopened && per->etag_save.stream)
! 650: fclose(per->etag_save.stream);
! 651:
! 652: if(per->etag_save.alloc_filename)
! 653: Curl_safefree(per->etag_save.filename);
! 654:
! 655: curl_easy_cleanup(per->curl);
! 656: if(outs->alloc_filename)
! 657: free(outs->filename);
! 658: free(per->this_url);
! 659: free(per->separator_err);
! 660: free(per->separator);
! 661: free(per->outfile);
! 662: free(per->uploadfile);
! 663:
! 664: return CURLE_OK;
! 665: }
! 666:
! 667: static void single_transfer_cleanup(struct OperationConfig *config)
! 668: {
! 669: if(config) {
! 670: struct State *state = &config->state;
! 671: if(state->urls) {
! 672: /* Free list of remaining URLs */
! 673: glob_cleanup(state->urls);
! 674: state->urls = NULL;
! 675: }
! 676: Curl_safefree(state->outfiles);
! 677: Curl_safefree(state->httpgetfields);
! 678: Curl_safefree(state->uploadfile);
! 679: if(state->inglob) {
! 680: /* Free list of globbed upload files */
! 681: glob_cleanup(state->inglob);
! 682: state->inglob = NULL;
! 683: }
! 684: }
! 685: }
! 686:
! 687: /* create the next (singular) transfer */
! 688:
! 689: static CURLcode single_transfer(struct GlobalConfig *global,
! 690: struct OperationConfig *config,
! 691: CURLSH *share,
! 692: bool capath_from_env,
! 693: bool *added)
! 694: {
! 695: CURLcode result = CURLE_OK;
! 696: struct getout *urlnode;
! 697: metalinkfile *mlfile_last = NULL;
! 698: bool orig_noprogress = global->noprogress;
! 699: bool orig_isatty = global->isatty;
! 700: struct State *state = &config->state;
! 701: char *httpgetfields = state->httpgetfields;
! 702: *added = FALSE; /* not yet */
! 703:
! 704: if(config->postfields) {
! 705: if(config->use_httpget) {
! 706: if(!httpgetfields) {
! 707: /* Use the postfields data for a http get */
! 708: httpgetfields = state->httpgetfields = strdup(config->postfields);
! 709: Curl_safefree(config->postfields);
! 710: if(!httpgetfields) {
! 711: errorf(global, "out of memory\n");
! 712: result = CURLE_OUT_OF_MEMORY;
! 713: }
! 714: else if(SetHTTPrequest(config,
! 715: (config->no_body?HTTPREQ_HEAD:HTTPREQ_GET),
! 716: &config->httpreq)) {
! 717: result = CURLE_FAILED_INIT;
! 718: }
! 719: }
! 720: }
! 721: else {
! 722: if(SetHTTPrequest(config, HTTPREQ_SIMPLEPOST, &config->httpreq))
! 723: result = CURLE_FAILED_INIT;
! 724: }
! 725: if(result) {
! 726: single_transfer_cleanup(config);
! 727: return result;
! 728: }
! 729: }
! 730: if(!state->urlnode) {
! 731: /* first time caller, setup things */
! 732: state->urlnode = config->url_list;
! 733: state->infilenum = 1;
! 734: }
! 735:
! 736: while(config->state.urlnode) {
! 737: char *infiles; /* might be a glob pattern */
! 738: URLGlob *inglob = state->inglob;
! 739: bool metalink = FALSE; /* metalink download? */
! 740: metalinkfile *mlfile;
! 741: metalink_resource *mlres;
! 742:
! 743: urlnode = config->state.urlnode;
! 744: if(urlnode->flags & GETOUT_METALINK) {
! 745: metalink = 1;
! 746: if(mlfile_last == NULL) {
! 747: mlfile_last = config->metalinkfile_list;
! 748: }
! 749: mlfile = mlfile_last;
! 750: mlfile_last = mlfile_last->next;
! 751: mlres = mlfile->resource;
! 752: }
! 753: else {
! 754: mlfile = NULL;
! 755: mlres = NULL;
! 756: }
! 757:
! 758: /* urlnode->url is the full URL (it might be NULL) */
! 759:
! 760: if(!urlnode->url) {
! 761: /* This node has no URL. Free node data without destroying the
! 762: node itself nor modifying next pointer and continue to next */
! 763: Curl_safefree(urlnode->outfile);
! 764: Curl_safefree(urlnode->infile);
! 765: urlnode->flags = 0;
! 766: config->state.urlnode = urlnode->next;
! 767: state->up = 0;
! 768: continue; /* next URL please */
! 769: }
! 770:
! 771: /* save outfile pattern before expansion */
! 772: if(urlnode->outfile && !state->outfiles) {
! 773: state->outfiles = strdup(urlnode->outfile);
! 774: if(!state->outfiles) {
! 775: errorf(global, "out of memory\n");
! 776: result = CURLE_OUT_OF_MEMORY;
! 777: break;
! 778: }
! 779: }
! 780:
! 781: infiles = urlnode->infile;
! 782:
! 783: if(!config->globoff && infiles && !inglob) {
! 784: /* Unless explicitly shut off */
! 785: result = glob_url(&inglob, infiles, &state->infilenum,
! 786: global->showerror?global->errors:NULL);
! 787: if(result)
! 788: break;
! 789: config->state.inglob = inglob;
! 790: }
! 791:
! 792: {
! 793: int separator;
! 794: unsigned long urlnum;
! 795:
! 796: if(!state->up && !infiles)
! 797: Curl_nop_stmt;
! 798: else {
! 799: if(!state->uploadfile) {
! 800: if(inglob) {
! 801: result = glob_next_url(&state->uploadfile, inglob);
! 802: if(result == CURLE_OUT_OF_MEMORY)
! 803: errorf(global, "out of memory\n");
! 804: }
! 805: else if(!state->up) {
! 806: state->uploadfile = strdup(infiles);
! 807: if(!state->uploadfile) {
! 808: errorf(global, "out of memory\n");
! 809: result = CURLE_OUT_OF_MEMORY;
! 810: }
! 811: }
! 812: }
! 813: if(result)
! 814: break;
! 815: }
! 816:
! 817: if(!state->urlnum) {
! 818: if(metalink) {
! 819: /* For Metalink download, we don't use glob. Instead we use
! 820: the number of resources as urlnum. */
! 821: urlnum = count_next_metalink_resource(mlfile);
! 822: }
! 823: else if(!config->globoff) {
! 824: /* Unless explicitly shut off, we expand '{...}' and '[...]'
! 825: expressions and return total number of URLs in pattern set */
! 826: result = glob_url(&state->urls, urlnode->url, &state->urlnum,
! 827: global->showerror?global->errors:NULL);
! 828: if(result)
! 829: break;
! 830: urlnum = state->urlnum;
! 831: }
! 832: else
! 833: urlnum = 1; /* without globbing, this is a single URL */
! 834: }
! 835: else
! 836: urlnum = state->urlnum;
! 837:
! 838: /* if multiple files extracted to stdout, insert separators! */
! 839: separator = ((!state->outfiles ||
! 840: !strcmp(state->outfiles, "-")) && urlnum > 1);
! 841:
! 842: if(state->up < state->infilenum) {
! 843: struct per_transfer *per;
! 844: struct OutStruct *outs;
! 845: struct InStruct *input;
! 846: struct OutStruct *heads;
! 847: struct OutStruct *etag_save;
! 848: struct HdrCbData *hdrcbdata = NULL;
! 849: CURL *curl = curl_easy_init();
! 850: result = add_per_transfer(&per);
! 851: if(result || !curl) {
! 852: curl_easy_cleanup(curl);
! 853: result = CURLE_OUT_OF_MEMORY;
! 854: break;
! 855: }
! 856: if(state->uploadfile) {
! 857: per->uploadfile = strdup(state->uploadfile);
! 858: if(!per->uploadfile) {
! 859: curl_easy_cleanup(curl);
! 860: result = CURLE_OUT_OF_MEMORY;
! 861: break;
! 862: }
! 863: }
! 864: *added = TRUE;
! 865: per->config = config;
! 866: per->curl = curl;
! 867:
! 868: /* default headers output stream is stdout */
! 869: heads = &per->heads;
! 870: heads->stream = stdout;
! 871:
! 872: /* Single header file for all URLs */
! 873: if(config->headerfile) {
! 874: /* open file for output: */
! 875: if(strcmp(config->headerfile, "-")) {
! 876: FILE *newfile;
! 877: newfile = fopen(config->headerfile, per->prev == NULL?"wb":"ab");
! 878: if(!newfile) {
! 879: warnf(config->global, "Failed to open %s\n", config->headerfile);
! 880: result = CURLE_WRITE_ERROR;
! 881: break;
! 882: }
! 883: else {
! 884: heads->filename = config->headerfile;
! 885: heads->s_isreg = TRUE;
! 886: heads->fopened = TRUE;
! 887: heads->stream = newfile;
! 888: }
! 889: }
! 890: else {
! 891: /* always use binary mode for protocol header output */
! 892: set_binmode(heads->stream);
! 893: }
! 894: }
! 895:
! 896: hdrcbdata = &per->hdrcbdata;
! 897:
! 898: outs = &per->outs;
! 899: input = &per->input;
! 900:
! 901: per->outfile = NULL;
! 902: per->infdopen = FALSE;
! 903: per->infd = STDIN_FILENO;
! 904:
! 905: /* default output stream is stdout */
! 906: outs->stream = stdout;
! 907:
! 908: /* --etag-compare */
! 909: if(config->etag_compare_file) {
! 910: char *etag_from_file = NULL;
! 911: char *header = NULL;
! 912:
! 913: /* open file for reading: */
! 914: FILE *file = fopen(config->etag_compare_file, FOPEN_READTEXT);
! 915: if(!file && !config->etag_save_file) {
! 916: errorf(config->global,
! 917: "Failed to open %s\n", config->etag_compare_file);
! 918: result = CURLE_READ_ERROR;
! 919: break;
! 920: }
! 921:
! 922: if((PARAM_OK == file2string(&etag_from_file, file)) &&
! 923: etag_from_file) {
! 924: header = aprintf("If-None-Match: \"%s\"", etag_from_file);
! 925: Curl_safefree(etag_from_file);
! 926: }
! 927: else
! 928: header = aprintf("If-None-Match: \"\"");
! 929:
! 930: if(!header) {
! 931: if(file)
! 932: fclose(file);
! 933: errorf(config->global,
! 934: "Failed to allocate memory for custom etag header\n");
! 935: result = CURLE_OUT_OF_MEMORY;
! 936: break;
! 937: }
! 938:
! 939: /* add Etag from file to list of custom headers */
! 940: add2list(&config->headers, header);
! 941:
! 942: Curl_safefree(header);
! 943:
! 944: if(file) {
! 945: fclose(file);
! 946: }
! 947: }
! 948:
! 949: /* --etag-save */
! 950: etag_save = &per->etag_save;
! 951: etag_save->stream = stdout;
! 952:
! 953: if(config->etag_save_file) {
! 954: /* open file for output: */
! 955: if(strcmp(config->etag_save_file, "-")) {
! 956: FILE *newfile = fopen(config->etag_save_file, "wb");
! 957: if(!newfile) {
! 958: warnf(
! 959: config->global,
! 960: "Failed to open %s\n", config->etag_save_file);
! 961:
! 962: result = CURLE_WRITE_ERROR;
! 963: break;
! 964: }
! 965: else {
! 966: etag_save->filename = config->etag_save_file;
! 967: etag_save->s_isreg = TRUE;
! 968: etag_save->fopened = TRUE;
! 969: etag_save->stream = newfile;
! 970: }
! 971: }
! 972: else {
! 973: /* always use binary mode for protocol header output */
! 974: set_binmode(etag_save->stream);
! 975: }
! 976: }
! 977:
! 978: if(metalink) {
! 979: /* For Metalink download, use name in Metalink file as
! 980: filename. */
! 981: per->outfile = strdup(mlfile->filename);
! 982: if(!per->outfile) {
! 983: result = CURLE_OUT_OF_MEMORY;
! 984: break;
! 985: }
! 986: per->this_url = strdup(mlres->url);
! 987: if(!per->this_url) {
! 988: result = CURLE_OUT_OF_MEMORY;
! 989: break;
! 990: }
! 991: per->mlfile = mlfile;
! 992: }
! 993: else {
! 994: if(state->urls) {
! 995: result = glob_next_url(&per->this_url, state->urls);
! 996: if(result)
! 997: break;
! 998: }
! 999: else if(!state->li) {
! 1000: per->this_url = strdup(urlnode->url);
! 1001: if(!per->this_url) {
! 1002: result = CURLE_OUT_OF_MEMORY;
! 1003: break;
! 1004: }
! 1005: }
! 1006: else
! 1007: per->this_url = NULL;
! 1008: if(!per->this_url)
! 1009: break;
! 1010:
! 1011: if(state->outfiles) {
! 1012: per->outfile = strdup(state->outfiles);
! 1013: if(!per->outfile) {
! 1014: result = CURLE_OUT_OF_MEMORY;
! 1015: break;
! 1016: }
! 1017: }
! 1018: }
! 1019:
! 1020: if(((urlnode->flags&GETOUT_USEREMOTE) ||
! 1021: (per->outfile && strcmp("-", per->outfile))) &&
! 1022: (metalink || !config->use_metalink)) {
! 1023:
! 1024: /*
! 1025: * We have specified a file name to store the result in, or we have
! 1026: * decided we want to use the remote file name.
! 1027: */
! 1028:
! 1029: if(!per->outfile) {
! 1030: /* extract the file name from the URL */
! 1031: result = get_url_file_name(&per->outfile, per->this_url);
! 1032: if(result)
! 1033: break;
! 1034: if(!*per->outfile && !config->content_disposition) {
! 1035: errorf(global, "Remote file name has no length!\n");
! 1036: result = CURLE_WRITE_ERROR;
! 1037: break;
! 1038: }
! 1039: }
! 1040: else if(state->urls) {
! 1041: /* fill '#1' ... '#9' terms from URL pattern */
! 1042: char *storefile = per->outfile;
! 1043: result = glob_match_url(&per->outfile, storefile, state->urls);
! 1044: Curl_safefree(storefile);
! 1045: if(result) {
! 1046: /* bad globbing */
! 1047: warnf(config->global, "bad output glob!\n");
! 1048: break;
! 1049: }
! 1050: }
! 1051:
! 1052: /* Create the directory hierarchy, if not pre-existent to a multiple
! 1053: file output call */
! 1054:
! 1055: if(config->create_dirs || metalink) {
! 1056: result = create_dir_hierarchy(per->outfile, global->errors);
! 1057: /* create_dir_hierarchy shows error upon CURLE_WRITE_ERROR */
! 1058: if(result)
! 1059: break;
! 1060: }
! 1061:
! 1062: if((urlnode->flags & GETOUT_USEREMOTE)
! 1063: && config->content_disposition) {
! 1064: /* Our header callback MIGHT set the filename */
! 1065: DEBUGASSERT(!outs->filename);
! 1066: }
! 1067:
! 1068: if(config->resume_from || config->resume_from_current) {
! 1069: #ifdef __VMS
! 1070: /* open file for output, forcing VMS output format into stream
! 1071: mode which is needed for stat() call above to always work. */
! 1072: FILE *file = fopen(outfile, "ab",
! 1073: "ctx=stm", "rfm=stmlf", "rat=cr", "mrs=0");
! 1074: #else
! 1075: /* open file for output: */
! 1076: FILE *file = fopen(per->outfile, "ab");
! 1077: #endif
! 1078: if(!file) {
! 1079: errorf(global, "Can't open '%s'!\n", per->outfile);
! 1080: result = CURLE_WRITE_ERROR;
! 1081: break;
! 1082: }
! 1083: outs->fopened = TRUE;
! 1084: outs->stream = file;
! 1085:
! 1086: if(config->resume_from_current) {
! 1087: /* We're told to continue from where we are now. Get the size
! 1088: of the file as it is now */
! 1089: struct_stat fileinfo;
! 1090: if(0 == fstat(fileno(outs->stream), &fileinfo))
! 1091: /* set offset to current file size: */
! 1092: config->resume_from = fileinfo.st_size;
! 1093: else
! 1094: /* let offset be 0 */
! 1095: config->resume_from = 0;
! 1096: }
! 1097:
! 1098: outs->init = config->resume_from;
! 1099: }
! 1100: else {
! 1101: outs->stream = NULL; /* open when needed */
! 1102: }
! 1103: outs->filename = per->outfile;
! 1104: outs->s_isreg = TRUE;
! 1105: }
! 1106:
! 1107: if(per->uploadfile && !stdin_upload(per->uploadfile)) {
! 1108: /*
! 1109: * We have specified a file to upload and it isn't "-".
! 1110: */
! 1111: char *nurl = add_file_name_to_url(per->this_url, per->uploadfile);
! 1112: if(!nurl) {
! 1113: result = CURLE_OUT_OF_MEMORY;
! 1114: break;
! 1115: }
! 1116: per->this_url = nurl;
! 1117: }
! 1118: else if(per->uploadfile && stdin_upload(per->uploadfile)) {
! 1119: /* count to see if there are more than one auth bit set
! 1120: in the authtype field */
! 1121: int authbits = 0;
! 1122: int bitcheck = 0;
! 1123: while(bitcheck < 32) {
! 1124: if(config->authtype & (1UL << bitcheck++)) {
! 1125: authbits++;
! 1126: if(authbits > 1) {
! 1127: /* more than one, we're done! */
! 1128: break;
! 1129: }
! 1130: }
! 1131: }
! 1132:
! 1133: /*
! 1134: * If the user has also selected --anyauth or --proxy-anyauth
! 1135: * we should warn him/her.
! 1136: */
! 1137: if(config->proxyanyauth || (authbits>1)) {
! 1138: warnf(config->global,
! 1139: "Using --anyauth or --proxy-anyauth with upload from stdin"
! 1140: " involves a big risk of it not working. Use a temporary"
! 1141: " file or a fixed auth type instead!\n");
! 1142: }
! 1143:
! 1144: DEBUGASSERT(per->infdopen == FALSE);
! 1145: DEBUGASSERT(per->infd == STDIN_FILENO);
! 1146:
! 1147: set_binmode(stdin);
! 1148: if(!strcmp(per->uploadfile, ".")) {
! 1149: if(curlx_nonblock((curl_socket_t)per->infd, TRUE) < 0)
! 1150: warnf(config->global,
! 1151: "fcntl failed on fd=%d: %s\n", per->infd, strerror(errno));
! 1152: }
! 1153: }
! 1154:
! 1155: if(per->uploadfile && config->resume_from_current)
! 1156: config->resume_from = -1; /* -1 will then force get-it-yourself */
! 1157:
! 1158: if(output_expected(per->this_url, per->uploadfile) && outs->stream &&
! 1159: isatty(fileno(outs->stream)))
! 1160: /* we send the output to a tty, therefore we switch off the progress
! 1161: meter */
! 1162: per->noprogress = global->noprogress = global->isatty = TRUE;
! 1163: else {
! 1164: /* progress meter is per download, so restore config
! 1165: values */
! 1166: per->noprogress = global->noprogress = orig_noprogress;
! 1167: global->isatty = orig_isatty;
! 1168: }
! 1169:
! 1170: if(urlnum > 1 && !global->mute) {
! 1171: per->separator_err =
! 1172: aprintf("\n[%lu/%lu]: %s --> %s",
! 1173: state->li + 1, urlnum, per->this_url,
! 1174: per->outfile ? per->outfile : "<stdout>");
! 1175: if(separator)
! 1176: per->separator = aprintf("%s%s", CURLseparator, per->this_url);
! 1177: }
! 1178: if(httpgetfields) {
! 1179: char *urlbuffer;
! 1180: /* Find out whether the url contains a file name */
! 1181: const char *pc = strstr(per->this_url, "://");
! 1182: char sep = '?';
! 1183: if(pc)
! 1184: pc += 3;
! 1185: else
! 1186: pc = per->this_url;
! 1187:
! 1188: pc = strrchr(pc, '/'); /* check for a slash */
! 1189:
! 1190: if(pc) {
! 1191: /* there is a slash present in the URL */
! 1192:
! 1193: if(strchr(pc, '?'))
! 1194: /* Ouch, there's already a question mark in the URL string, we
! 1195: then append the data with an ampersand separator instead! */
! 1196: sep = '&';
! 1197: }
! 1198: /*
! 1199: * Then append ? followed by the get fields to the url.
! 1200: */
! 1201: if(pc)
! 1202: urlbuffer = aprintf("%s%c%s", per->this_url, sep, httpgetfields);
! 1203: else
! 1204: /* Append / before the ? to create a well-formed url
! 1205: if the url contains a hostname only
! 1206: */
! 1207: urlbuffer = aprintf("%s/?%s", per->this_url, httpgetfields);
! 1208:
! 1209: if(!urlbuffer) {
! 1210: result = CURLE_OUT_OF_MEMORY;
! 1211: break;
! 1212: }
! 1213:
! 1214: Curl_safefree(per->this_url); /* free previous URL */
! 1215: per->this_url = urlbuffer; /* use our new URL instead! */
! 1216: }
! 1217:
! 1218: if(!global->errors)
! 1219: global->errors = stderr;
! 1220:
! 1221: if((!per->outfile || !strcmp(per->outfile, "-")) &&
! 1222: !config->use_ascii) {
! 1223: /* We get the output to stdout and we have not got the ASCII/text
! 1224: flag, then set stdout to be binary */
! 1225: set_binmode(stdout);
! 1226: }
! 1227:
! 1228: /* explicitly passed to stdout means okaying binary gunk */
! 1229: config->terminal_binary_ok =
! 1230: (per->outfile && !strcmp(per->outfile, "-"));
! 1231:
! 1232: /* Avoid having this setopt added to the --libcurl source output. */
! 1233: result = curl_easy_setopt(curl, CURLOPT_SHARE, share);
! 1234: if(result)
! 1235: break;
! 1236:
! 1237: if(!config->tcp_nodelay)
! 1238: my_setopt(curl, CURLOPT_TCP_NODELAY, 0L);
! 1239:
! 1240: if(config->tcp_fastopen)
! 1241: my_setopt(curl, CURLOPT_TCP_FASTOPEN, 1L);
! 1242:
! 1243: /* where to store */
! 1244: my_setopt(curl, CURLOPT_WRITEDATA, per);
! 1245: my_setopt(curl, CURLOPT_INTERLEAVEDATA, per);
! 1246:
! 1247: if(metalink || !config->use_metalink)
! 1248: /* what call to write */
! 1249: my_setopt(curl, CURLOPT_WRITEFUNCTION, tool_write_cb);
! 1250: #ifdef USE_METALINK
! 1251: else
! 1252: /* Set Metalink specific write callback function to parse
! 1253: XML data progressively. */
! 1254: my_setopt(curl, CURLOPT_WRITEFUNCTION, metalink_write_cb);
! 1255: #endif /* USE_METALINK */
! 1256:
! 1257: /* for uploads */
! 1258: input->config = config;
! 1259: /* Note that if CURLOPT_READFUNCTION is fread (the default), then
! 1260: * lib/telnet.c will Curl_poll() on the input file descriptor
! 1261: * rather then calling the READFUNCTION at regular intervals.
! 1262: * The circumstances in which it is preferable to enable this
! 1263: * behaviour, by omitting to set the READFUNCTION & READDATA options,
! 1264: * have not been determined.
! 1265: */
! 1266: my_setopt(curl, CURLOPT_READDATA, input);
! 1267: /* what call to read */
! 1268: my_setopt(curl, CURLOPT_READFUNCTION, tool_read_cb);
! 1269:
! 1270: /* in 7.18.0, the CURLOPT_SEEKFUNCTION/DATA pair is taking over what
! 1271: CURLOPT_IOCTLFUNCTION/DATA pair previously provided for seeking */
! 1272: my_setopt(curl, CURLOPT_SEEKDATA, input);
! 1273: my_setopt(curl, CURLOPT_SEEKFUNCTION, tool_seek_cb);
! 1274:
! 1275: if(config->recvpersecond &&
! 1276: (config->recvpersecond < BUFFER_SIZE))
! 1277: /* use a smaller sized buffer for better sleeps */
! 1278: my_setopt(curl, CURLOPT_BUFFERSIZE, (long)config->recvpersecond);
! 1279: else
! 1280: my_setopt(curl, CURLOPT_BUFFERSIZE, (long)BUFFER_SIZE);
! 1281:
! 1282: my_setopt_str(curl, CURLOPT_URL, per->this_url);
! 1283: my_setopt(curl, CURLOPT_NOPROGRESS, global->noprogress?1L:0L);
! 1284: if(config->no_body)
! 1285: my_setopt(curl, CURLOPT_NOBODY, 1L);
! 1286:
! 1287: if(config->oauth_bearer)
! 1288: my_setopt_str(curl, CURLOPT_XOAUTH2_BEARER, config->oauth_bearer);
! 1289:
! 1290: {
! 1291: my_setopt_str(curl, CURLOPT_PROXY, config->proxy);
! 1292: /* new in libcurl 7.5 */
! 1293: if(config->proxy)
! 1294: my_setopt_enum(curl, CURLOPT_PROXYTYPE, config->proxyver);
! 1295:
! 1296: my_setopt_str(curl, CURLOPT_PROXYUSERPWD, config->proxyuserpwd);
! 1297:
! 1298: /* new in libcurl 7.3 */
! 1299: my_setopt(curl, CURLOPT_HTTPPROXYTUNNEL, config->proxytunnel?1L:0L);
! 1300:
! 1301: /* new in libcurl 7.52.0 */
! 1302: if(config->preproxy)
! 1303: my_setopt_str(curl, CURLOPT_PRE_PROXY, config->preproxy);
! 1304:
! 1305: /* new in libcurl 7.10.6 */
! 1306: if(config->proxyanyauth)
! 1307: my_setopt_bitmask(curl, CURLOPT_PROXYAUTH,
! 1308: (long)CURLAUTH_ANY);
! 1309: else if(config->proxynegotiate)
! 1310: my_setopt_bitmask(curl, CURLOPT_PROXYAUTH,
! 1311: (long)CURLAUTH_GSSNEGOTIATE);
! 1312: else if(config->proxyntlm)
! 1313: my_setopt_bitmask(curl, CURLOPT_PROXYAUTH,
! 1314: (long)CURLAUTH_NTLM);
! 1315: else if(config->proxydigest)
! 1316: my_setopt_bitmask(curl, CURLOPT_PROXYAUTH,
! 1317: (long)CURLAUTH_DIGEST);
! 1318: else if(config->proxybasic)
! 1319: my_setopt_bitmask(curl, CURLOPT_PROXYAUTH,
! 1320: (long)CURLAUTH_BASIC);
! 1321:
! 1322: /* new in libcurl 7.19.4 */
! 1323: my_setopt_str(curl, CURLOPT_NOPROXY, config->noproxy);
! 1324:
! 1325: my_setopt(curl, CURLOPT_SUPPRESS_CONNECT_HEADERS,
! 1326: config->suppress_connect_headers?1L:0L);
! 1327: }
! 1328:
! 1329: my_setopt(curl, CURLOPT_FAILONERROR, config->failonerror?1L:0L);
! 1330: my_setopt(curl, CURLOPT_REQUEST_TARGET, config->request_target);
! 1331: my_setopt(curl, CURLOPT_UPLOAD, per->uploadfile?1L:0L);
! 1332: my_setopt(curl, CURLOPT_DIRLISTONLY, config->dirlistonly?1L:0L);
! 1333: my_setopt(curl, CURLOPT_APPEND, config->ftp_append?1L:0L);
! 1334:
! 1335: if(config->netrc_opt)
! 1336: my_setopt_enum(curl, CURLOPT_NETRC, (long)CURL_NETRC_OPTIONAL);
! 1337: else if(config->netrc || config->netrc_file)
! 1338: my_setopt_enum(curl, CURLOPT_NETRC, (long)CURL_NETRC_REQUIRED);
! 1339: else
! 1340: my_setopt_enum(curl, CURLOPT_NETRC, (long)CURL_NETRC_IGNORED);
! 1341:
! 1342: if(config->netrc_file)
! 1343: my_setopt_str(curl, CURLOPT_NETRC_FILE, config->netrc_file);
! 1344:
! 1345: my_setopt(curl, CURLOPT_TRANSFERTEXT, config->use_ascii?1L:0L);
! 1346: if(config->login_options)
! 1347: my_setopt_str(curl, CURLOPT_LOGIN_OPTIONS, config->login_options);
! 1348: my_setopt_str(curl, CURLOPT_USERPWD, config->userpwd);
! 1349: my_setopt_str(curl, CURLOPT_RANGE, config->range);
! 1350: my_setopt(curl, CURLOPT_ERRORBUFFER, per->errorbuffer);
! 1351: my_setopt(curl, CURLOPT_TIMEOUT_MS, (long)(config->timeout * 1000));
! 1352:
! 1353: switch(config->httpreq) {
! 1354: case HTTPREQ_SIMPLEPOST:
! 1355: my_setopt_str(curl, CURLOPT_POSTFIELDS,
! 1356: config->postfields);
! 1357: my_setopt(curl, CURLOPT_POSTFIELDSIZE_LARGE,
! 1358: config->postfieldsize);
! 1359: break;
! 1360: case HTTPREQ_MIMEPOST:
! 1361: /* free previous remainders */
! 1362: curl_mime_free(config->mimepost);
! 1363: config->mimepost = NULL;
! 1364: result = tool2curlmime(curl, config->mimeroot, &config->mimepost);
! 1365: if(result)
! 1366: break;
! 1367: my_setopt_mimepost(curl, CURLOPT_MIMEPOST, config->mimepost);
! 1368: break;
! 1369: default:
! 1370: break;
! 1371: }
! 1372: if(result)
! 1373: break;
! 1374:
! 1375: /* new in libcurl 7.10.6 (default is Basic) */
! 1376: if(config->authtype)
! 1377: my_setopt_bitmask(curl, CURLOPT_HTTPAUTH, (long)config->authtype);
! 1378:
! 1379: my_setopt_slist(curl, CURLOPT_HTTPHEADER, config->headers);
! 1380:
! 1381: if(built_in_protos & (CURLPROTO_HTTP | CURLPROTO_RTSP)) {
! 1382: my_setopt_str(curl, CURLOPT_REFERER, config->referer);
! 1383: my_setopt_str(curl, CURLOPT_USERAGENT, config->useragent);
! 1384: }
! 1385:
! 1386: if(built_in_protos & CURLPROTO_HTTP) {
! 1387:
! 1388: long postRedir = 0;
! 1389:
! 1390: my_setopt(curl, CURLOPT_FOLLOWLOCATION,
! 1391: config->followlocation?1L:0L);
! 1392: my_setopt(curl, CURLOPT_UNRESTRICTED_AUTH,
! 1393: config->unrestricted_auth?1L:0L);
! 1394:
! 1395: my_setopt(curl, CURLOPT_AUTOREFERER, config->autoreferer?1L:0L);
! 1396:
! 1397: /* new in libcurl 7.36.0 */
! 1398: if(config->proxyheaders) {
! 1399: my_setopt_slist(curl, CURLOPT_PROXYHEADER, config->proxyheaders);
! 1400: my_setopt(curl, CURLOPT_HEADEROPT, CURLHEADER_SEPARATE);
! 1401: }
! 1402:
! 1403: /* new in libcurl 7.5 */
! 1404: my_setopt(curl, CURLOPT_MAXREDIRS, config->maxredirs);
! 1405:
! 1406: if(config->httpversion)
! 1407: my_setopt_enum(curl, CURLOPT_HTTP_VERSION, config->httpversion);
! 1408: else if(curlinfo->features & CURL_VERSION_HTTP2) {
! 1409: my_setopt_enum(curl, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_2TLS);
! 1410: }
! 1411:
! 1412: /* curl 7.19.1 (the 301 version existed in 7.18.2),
! 1413: 303 was added in 7.26.0 */
! 1414: if(config->post301)
! 1415: postRedir |= CURL_REDIR_POST_301;
! 1416: if(config->post302)
! 1417: postRedir |= CURL_REDIR_POST_302;
! 1418: if(config->post303)
! 1419: postRedir |= CURL_REDIR_POST_303;
! 1420: my_setopt(curl, CURLOPT_POSTREDIR, postRedir);
! 1421:
! 1422: /* new in libcurl 7.21.6 */
! 1423: if(config->encoding)
! 1424: my_setopt_str(curl, CURLOPT_ACCEPT_ENCODING, "");
! 1425:
! 1426: /* new in libcurl 7.21.6 */
! 1427: if(config->tr_encoding)
! 1428: my_setopt(curl, CURLOPT_TRANSFER_ENCODING, 1L);
! 1429: /* new in libcurl 7.64.0 */
! 1430: my_setopt(curl, CURLOPT_HTTP09_ALLOWED,
! 1431: config->http09_allowed ? 1L : 0L);
! 1432:
! 1433: } /* (built_in_protos & CURLPROTO_HTTP) */
! 1434:
! 1435: my_setopt_str(curl, CURLOPT_FTPPORT, config->ftpport);
! 1436: my_setopt(curl, CURLOPT_LOW_SPEED_LIMIT,
! 1437: config->low_speed_limit);
! 1438: my_setopt(curl, CURLOPT_LOW_SPEED_TIME, config->low_speed_time);
! 1439: my_setopt(curl, CURLOPT_MAX_SEND_SPEED_LARGE,
! 1440: config->sendpersecond);
! 1441: my_setopt(curl, CURLOPT_MAX_RECV_SPEED_LARGE,
! 1442: config->recvpersecond);
! 1443:
! 1444: if(config->use_resume)
! 1445: my_setopt(curl, CURLOPT_RESUME_FROM_LARGE, config->resume_from);
! 1446: else
! 1447: my_setopt(curl, CURLOPT_RESUME_FROM_LARGE, CURL_OFF_T_C(0));
! 1448:
! 1449: my_setopt_str(curl, CURLOPT_KEYPASSWD, config->key_passwd);
! 1450: my_setopt_str(curl, CURLOPT_PROXY_KEYPASSWD, config->proxy_key_passwd);
! 1451:
! 1452: if(built_in_protos & (CURLPROTO_SCP|CURLPROTO_SFTP)) {
! 1453:
! 1454: /* SSH and SSL private key uses same command-line option */
! 1455: /* new in libcurl 7.16.1 */
! 1456: my_setopt_str(curl, CURLOPT_SSH_PRIVATE_KEYFILE, config->key);
! 1457: /* new in libcurl 7.16.1 */
! 1458: my_setopt_str(curl, CURLOPT_SSH_PUBLIC_KEYFILE, config->pubkey);
! 1459:
! 1460: /* new in libcurl 7.17.1: SSH host key md5 checking allows us
! 1461: to fail if we are not talking to who we think we should */
! 1462: my_setopt_str(curl, CURLOPT_SSH_HOST_PUBLIC_KEY_MD5,
! 1463: config->hostpubmd5);
! 1464:
! 1465: /* new in libcurl 7.56.0 */
! 1466: if(config->ssh_compression)
! 1467: my_setopt(curl, CURLOPT_SSH_COMPRESSION, 1L);
! 1468: }
! 1469:
! 1470: if(config->cacert)
! 1471: my_setopt_str(curl, CURLOPT_CAINFO, config->cacert);
! 1472: if(config->proxy_cacert)
! 1473: my_setopt_str(curl, CURLOPT_PROXY_CAINFO, config->proxy_cacert);
! 1474:
! 1475: if(config->capath) {
! 1476: result = res_setopt_str(curl, CURLOPT_CAPATH, config->capath);
! 1477: if(result == CURLE_NOT_BUILT_IN) {
! 1478: warnf(config->global, "ignoring %s, not supported by libcurl\n",
! 1479: capath_from_env?
! 1480: "SSL_CERT_DIR environment variable":"--capath");
! 1481: }
! 1482: else if(result)
! 1483: break;
! 1484: }
! 1485: /* For the time being if --proxy-capath is not set then we use the
! 1486: --capath value for it, if any. See #1257 */
! 1487: if((config->proxy_capath || config->capath) &&
! 1488: !tool_setopt_skip(CURLOPT_PROXY_CAPATH)) {
! 1489: result = res_setopt_str(curl, CURLOPT_PROXY_CAPATH,
! 1490: (config->proxy_capath ?
! 1491: config->proxy_capath :
! 1492: config->capath));
! 1493: if(result == CURLE_NOT_BUILT_IN) {
! 1494: if(config->proxy_capath) {
! 1495: warnf(config->global,
! 1496: "ignoring --proxy-capath, not supported by libcurl\n");
! 1497: }
! 1498: }
! 1499: else if(result)
! 1500: break;
! 1501: }
! 1502:
! 1503: if(config->crlfile)
! 1504: my_setopt_str(curl, CURLOPT_CRLFILE, config->crlfile);
! 1505: if(config->proxy_crlfile)
! 1506: my_setopt_str(curl, CURLOPT_PROXY_CRLFILE, config->proxy_crlfile);
! 1507: else if(config->crlfile) /* CURLOPT_PROXY_CRLFILE default is crlfile */
! 1508: my_setopt_str(curl, CURLOPT_PROXY_CRLFILE, config->crlfile);
! 1509:
! 1510: if(config->pinnedpubkey)
! 1511: my_setopt_str(curl, CURLOPT_PINNEDPUBLICKEY, config->pinnedpubkey);
! 1512:
! 1513: if(curlinfo->features & CURL_VERSION_SSL) {
! 1514: /* Check if config->cert is a PKCS#11 URI and set the
! 1515: * config->cert_type if necessary */
! 1516: if(config->cert) {
! 1517: if(!config->cert_type) {
! 1518: if(is_pkcs11_uri(config->cert)) {
! 1519: config->cert_type = strdup("ENG");
! 1520: }
! 1521: }
! 1522: }
! 1523:
! 1524: /* Check if config->key is a PKCS#11 URI and set the
! 1525: * config->key_type if necessary */
! 1526: if(config->key) {
! 1527: if(!config->key_type) {
! 1528: if(is_pkcs11_uri(config->key)) {
! 1529: config->key_type = strdup("ENG");
! 1530: }
! 1531: }
! 1532: }
! 1533:
! 1534: /* Check if config->proxy_cert is a PKCS#11 URI and set the
! 1535: * config->proxy_type if necessary */
! 1536: if(config->proxy_cert) {
! 1537: if(!config->proxy_cert_type) {
! 1538: if(is_pkcs11_uri(config->proxy_cert)) {
! 1539: config->proxy_cert_type = strdup("ENG");
! 1540: }
! 1541: }
! 1542: }
! 1543:
! 1544: /* Check if config->proxy_key is a PKCS#11 URI and set the
! 1545: * config->proxy_key_type if necessary */
! 1546: if(config->proxy_key) {
! 1547: if(!config->proxy_key_type) {
! 1548: if(is_pkcs11_uri(config->proxy_key)) {
! 1549: config->proxy_key_type = strdup("ENG");
! 1550: }
! 1551: }
! 1552: }
! 1553:
! 1554: my_setopt_str(curl, CURLOPT_SSLCERT, config->cert);
! 1555: my_setopt_str(curl, CURLOPT_PROXY_SSLCERT, config->proxy_cert);
! 1556: my_setopt_str(curl, CURLOPT_SSLCERTTYPE, config->cert_type);
! 1557: my_setopt_str(curl, CURLOPT_PROXY_SSLCERTTYPE,
! 1558: config->proxy_cert_type);
! 1559: my_setopt_str(curl, CURLOPT_SSLKEY, config->key);
! 1560: my_setopt_str(curl, CURLOPT_PROXY_SSLKEY, config->proxy_key);
! 1561: my_setopt_str(curl, CURLOPT_SSLKEYTYPE, config->key_type);
! 1562: my_setopt_str(curl, CURLOPT_PROXY_SSLKEYTYPE,
! 1563: config->proxy_key_type);
! 1564:
! 1565: if(config->insecure_ok) {
! 1566: my_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0L);
! 1567: my_setopt(curl, CURLOPT_SSL_VERIFYHOST, 0L);
! 1568: }
! 1569: else {
! 1570: my_setopt(curl, CURLOPT_SSL_VERIFYPEER, 1L);
! 1571: /* libcurl default is strict verifyhost -> 2L */
! 1572: /* my_setopt(curl, CURLOPT_SSL_VERIFYHOST, 2L); */
! 1573: }
! 1574: if(config->proxy_insecure_ok) {
! 1575: my_setopt(curl, CURLOPT_PROXY_SSL_VERIFYPEER, 0L);
! 1576: my_setopt(curl, CURLOPT_PROXY_SSL_VERIFYHOST, 0L);
! 1577: }
! 1578: else {
! 1579: my_setopt(curl, CURLOPT_PROXY_SSL_VERIFYPEER, 1L);
! 1580: }
! 1581:
! 1582: if(config->verifystatus)
! 1583: my_setopt(curl, CURLOPT_SSL_VERIFYSTATUS, 1L);
! 1584:
! 1585: if(config->falsestart)
! 1586: my_setopt(curl, CURLOPT_SSL_FALSESTART, 1L);
! 1587:
! 1588: my_setopt_enum(curl, CURLOPT_SSLVERSION,
! 1589: config->ssl_version | config->ssl_version_max);
! 1590: my_setopt_enum(curl, CURLOPT_PROXY_SSLVERSION,
! 1591: config->proxy_ssl_version);
! 1592: }
! 1593: if(config->path_as_is)
! 1594: my_setopt(curl, CURLOPT_PATH_AS_IS, 1L);
! 1595:
! 1596: if(built_in_protos & (CURLPROTO_SCP|CURLPROTO_SFTP)) {
! 1597: if(!config->insecure_ok) {
! 1598: char *home;
! 1599: char *file;
! 1600: result = CURLE_FAILED_INIT;
! 1601: home = homedir();
! 1602: if(home) {
! 1603: file = aprintf("%s/.ssh/known_hosts", home);
! 1604: if(file) {
! 1605: /* new in curl 7.19.6 */
! 1606: result = res_setopt_str(curl, CURLOPT_SSH_KNOWNHOSTS, file);
! 1607: curl_free(file);
! 1608: if(result == CURLE_UNKNOWN_OPTION)
! 1609: /* libssh2 version older than 1.1.1 */
! 1610: result = CURLE_OK;
! 1611: }
! 1612: Curl_safefree(home);
! 1613: }
! 1614: else {
! 1615: errorf(global, "Failed to figure out user's home dir!");
! 1616: }
! 1617: if(result)
! 1618: break;
! 1619: }
! 1620: }
! 1621:
! 1622: if(config->no_body || config->remote_time) {
! 1623: /* no body or use remote time */
! 1624: my_setopt(curl, CURLOPT_FILETIME, 1L);
! 1625: }
! 1626:
! 1627: my_setopt(curl, CURLOPT_CRLF, config->crlf?1L:0L);
! 1628: my_setopt_slist(curl, CURLOPT_QUOTE, config->quote);
! 1629: my_setopt_slist(curl, CURLOPT_POSTQUOTE, config->postquote);
! 1630: my_setopt_slist(curl, CURLOPT_PREQUOTE, config->prequote);
! 1631:
! 1632: if(config->cookie)
! 1633: my_setopt_str(curl, CURLOPT_COOKIE, config->cookie);
! 1634:
! 1635: if(config->cookiefile)
! 1636: my_setopt_str(curl, CURLOPT_COOKIEFILE, config->cookiefile);
! 1637:
! 1638: /* new in libcurl 7.9 */
! 1639: if(config->cookiejar)
! 1640: my_setopt_str(curl, CURLOPT_COOKIEJAR, config->cookiejar);
! 1641:
! 1642: /* new in libcurl 7.9.7 */
! 1643: my_setopt(curl, CURLOPT_COOKIESESSION, config->cookiesession?1L:0L);
! 1644:
! 1645: my_setopt_enum(curl, CURLOPT_TIMECONDITION, (long)config->timecond);
! 1646: my_setopt(curl, CURLOPT_TIMEVALUE_LARGE, config->condtime);
! 1647: my_setopt_str(curl, CURLOPT_CUSTOMREQUEST, config->customrequest);
! 1648: customrequest_helper(config, config->httpreq, config->customrequest);
! 1649: my_setopt(curl, CURLOPT_STDERR, global->errors);
! 1650:
! 1651: /* three new ones in libcurl 7.3: */
! 1652: my_setopt_str(curl, CURLOPT_INTERFACE, config->iface);
! 1653: my_setopt_str(curl, CURLOPT_KRBLEVEL, config->krblevel);
! 1654: progressbarinit(&per->progressbar, config);
! 1655:
! 1656: if((global->progressmode == CURL_PROGRESS_BAR) &&
! 1657: !global->noprogress && !global->mute) {
! 1658: /* we want the alternative style, then we have to implement it
! 1659: ourselves! */
! 1660: my_setopt(curl, CURLOPT_XFERINFOFUNCTION, tool_progress_cb);
! 1661: my_setopt(curl, CURLOPT_XFERINFODATA, per);
! 1662: }
! 1663: else if(per->uploadfile && !strcmp(per->uploadfile, ".")) {
! 1664: /* when reading from stdin in non-blocking mode, we use the progress
! 1665: function to unpause a busy read */
! 1666: my_setopt(curl, CURLOPT_NOPROGRESS, 0L);
! 1667: my_setopt(curl, CURLOPT_XFERINFOFUNCTION, tool_readbusy_cb);
! 1668: my_setopt(curl, CURLOPT_XFERINFODATA, per);
! 1669: }
! 1670:
! 1671: /* new in libcurl 7.24.0: */
! 1672: if(config->dns_servers)
! 1673: my_setopt_str(curl, CURLOPT_DNS_SERVERS, config->dns_servers);
! 1674:
! 1675: /* new in libcurl 7.33.0: */
! 1676: if(config->dns_interface)
! 1677: my_setopt_str(curl, CURLOPT_DNS_INTERFACE, config->dns_interface);
! 1678: if(config->dns_ipv4_addr)
! 1679: my_setopt_str(curl, CURLOPT_DNS_LOCAL_IP4, config->dns_ipv4_addr);
! 1680: if(config->dns_ipv6_addr)
! 1681: my_setopt_str(curl, CURLOPT_DNS_LOCAL_IP6, config->dns_ipv6_addr);
! 1682:
! 1683: /* new in libcurl 7.6.2: */
! 1684: my_setopt_slist(curl, CURLOPT_TELNETOPTIONS, config->telnet_options);
! 1685:
! 1686: /* new in libcurl 7.7: */
! 1687: my_setopt_str(curl, CURLOPT_RANDOM_FILE, config->random_file);
! 1688: my_setopt_str(curl, CURLOPT_EGDSOCKET, config->egd_file);
! 1689: my_setopt(curl, CURLOPT_CONNECTTIMEOUT_MS,
! 1690: (long)(config->connecttimeout * 1000));
! 1691:
! 1692: if(config->doh_url)
! 1693: my_setopt_str(curl, CURLOPT_DOH_URL, config->doh_url);
! 1694:
! 1695: if(config->cipher_list)
! 1696: my_setopt_str(curl, CURLOPT_SSL_CIPHER_LIST, config->cipher_list);
! 1697:
! 1698: if(config->proxy_cipher_list)
! 1699: my_setopt_str(curl, CURLOPT_PROXY_SSL_CIPHER_LIST,
! 1700: config->proxy_cipher_list);
! 1701:
! 1702: if(config->cipher13_list)
! 1703: my_setopt_str(curl, CURLOPT_TLS13_CIPHERS, config->cipher13_list);
! 1704:
! 1705: if(config->proxy_cipher13_list)
! 1706: my_setopt_str(curl, CURLOPT_PROXY_TLS13_CIPHERS,
! 1707: config->proxy_cipher13_list);
! 1708:
! 1709: /* new in libcurl 7.9.2: */
! 1710: if(config->disable_epsv)
! 1711: /* disable it */
! 1712: my_setopt(curl, CURLOPT_FTP_USE_EPSV, 0L);
! 1713:
! 1714: /* new in libcurl 7.10.5 */
! 1715: if(config->disable_eprt)
! 1716: /* disable it */
! 1717: my_setopt(curl, CURLOPT_FTP_USE_EPRT, 0L);
! 1718:
! 1719: if(global->tracetype != TRACE_NONE) {
! 1720: my_setopt(curl, CURLOPT_DEBUGFUNCTION, tool_debug_cb);
! 1721: my_setopt(curl, CURLOPT_DEBUGDATA, config);
! 1722: my_setopt(curl, CURLOPT_VERBOSE, 1L);
! 1723: }
! 1724:
! 1725: /* new in curl 7.9.3 */
! 1726: if(config->engine) {
! 1727: result = res_setopt_str(curl, CURLOPT_SSLENGINE, config->engine);
! 1728: if(result)
! 1729: break;
! 1730: }
! 1731:
! 1732: /* new in curl 7.10.7, extended in 7.19.4. Modified to use
! 1733: CREATE_DIR_RETRY in 7.49.0 */
! 1734: my_setopt(curl, CURLOPT_FTP_CREATE_MISSING_DIRS,
! 1735: (long)(config->ftp_create_dirs?
! 1736: CURLFTP_CREATE_DIR_RETRY:
! 1737: CURLFTP_CREATE_DIR_NONE));
! 1738:
! 1739: /* new in curl 7.10.8 */
! 1740: if(config->max_filesize)
! 1741: my_setopt(curl, CURLOPT_MAXFILESIZE_LARGE,
! 1742: config->max_filesize);
! 1743:
! 1744: if(4 == config->ip_version)
! 1745: my_setopt(curl, CURLOPT_IPRESOLVE, CURL_IPRESOLVE_V4);
! 1746: else if(6 == config->ip_version)
! 1747: my_setopt(curl, CURLOPT_IPRESOLVE, CURL_IPRESOLVE_V6);
! 1748: else
! 1749: my_setopt(curl, CURLOPT_IPRESOLVE, CURL_IPRESOLVE_WHATEVER);
! 1750:
! 1751: /* new in curl 7.15.5 */
! 1752: if(config->ftp_ssl_reqd)
! 1753: my_setopt_enum(curl, CURLOPT_USE_SSL, (long)CURLUSESSL_ALL);
! 1754:
! 1755: /* new in curl 7.11.0 */
! 1756: else if(config->ftp_ssl)
! 1757: my_setopt_enum(curl, CURLOPT_USE_SSL, (long)CURLUSESSL_TRY);
! 1758:
! 1759: /* new in curl 7.16.0 */
! 1760: else if(config->ftp_ssl_control)
! 1761: my_setopt_enum(curl, CURLOPT_USE_SSL, (long)CURLUSESSL_CONTROL);
! 1762:
! 1763: /* new in curl 7.16.1 */
! 1764: if(config->ftp_ssl_ccc)
! 1765: my_setopt_enum(curl, CURLOPT_FTP_SSL_CCC,
! 1766: (long)config->ftp_ssl_ccc_mode);
! 1767:
! 1768: /* new in curl 7.19.4 */
! 1769: if(config->socks5_gssapi_nec)
! 1770: my_setopt_str(curl, CURLOPT_SOCKS5_GSSAPI_NEC,
! 1771: config->socks5_gssapi_nec);
! 1772:
! 1773: /* new in curl 7.55.0 */
! 1774: if(config->socks5_auth)
! 1775: my_setopt_bitmask(curl, CURLOPT_SOCKS5_AUTH,
! 1776: (long)config->socks5_auth);
! 1777:
! 1778: /* new in curl 7.43.0 */
! 1779: if(config->proxy_service_name)
! 1780: my_setopt_str(curl, CURLOPT_PROXY_SERVICE_NAME,
! 1781: config->proxy_service_name);
! 1782:
! 1783: /* new in curl 7.43.0 */
! 1784: if(config->service_name)
! 1785: my_setopt_str(curl, CURLOPT_SERVICE_NAME,
! 1786: config->service_name);
! 1787:
! 1788: /* curl 7.13.0 */
! 1789: my_setopt_str(curl, CURLOPT_FTP_ACCOUNT, config->ftp_account);
! 1790: my_setopt(curl, CURLOPT_IGNORE_CONTENT_LENGTH, config->ignorecl?1L:0L);
! 1791:
! 1792: /* curl 7.14.2 */
! 1793: my_setopt(curl, CURLOPT_FTP_SKIP_PASV_IP, config->ftp_skip_ip?1L:0L);
! 1794:
! 1795: /* curl 7.15.1 */
! 1796: my_setopt(curl, CURLOPT_FTP_FILEMETHOD, (long)config->ftp_filemethod);
! 1797:
! 1798: /* curl 7.15.2 */
! 1799: if(config->localport) {
! 1800: my_setopt(curl, CURLOPT_LOCALPORT, config->localport);
! 1801: my_setopt_str(curl, CURLOPT_LOCALPORTRANGE, config->localportrange);
! 1802: }
! 1803:
! 1804: /* curl 7.15.5 */
! 1805: my_setopt_str(curl, CURLOPT_FTP_ALTERNATIVE_TO_USER,
! 1806: config->ftp_alternative_to_user);
! 1807:
! 1808: /* curl 7.16.0 */
! 1809: if(config->disable_sessionid)
! 1810: /* disable it */
! 1811: my_setopt(curl, CURLOPT_SSL_SESSIONID_CACHE, 0L);
! 1812:
! 1813: /* curl 7.16.2 */
! 1814: if(config->raw) {
! 1815: my_setopt(curl, CURLOPT_HTTP_CONTENT_DECODING, 0L);
! 1816: my_setopt(curl, CURLOPT_HTTP_TRANSFER_DECODING, 0L);
! 1817: }
! 1818:
! 1819: /* curl 7.17.1 */
! 1820: if(!config->nokeepalive) {
! 1821: my_setopt(curl, CURLOPT_TCP_KEEPALIVE, 1L);
! 1822: if(config->alivetime != 0) {
! 1823: my_setopt(curl, CURLOPT_TCP_KEEPIDLE, config->alivetime);
! 1824: my_setopt(curl, CURLOPT_TCP_KEEPINTVL, config->alivetime);
! 1825: }
! 1826: }
! 1827: else
! 1828: my_setopt(curl, CURLOPT_TCP_KEEPALIVE, 0L);
! 1829:
! 1830: /* curl 7.20.0 */
! 1831: if(config->tftp_blksize)
! 1832: my_setopt(curl, CURLOPT_TFTP_BLKSIZE, config->tftp_blksize);
! 1833:
! 1834: if(config->mail_from)
! 1835: my_setopt_str(curl, CURLOPT_MAIL_FROM, config->mail_from);
! 1836:
! 1837: if(config->mail_rcpt)
! 1838: my_setopt_slist(curl, CURLOPT_MAIL_RCPT, config->mail_rcpt);
! 1839:
! 1840: /* curl 7.69.x */
! 1841: my_setopt(curl, CURLOPT_MAIL_RCPT_ALLLOWFAILS,
! 1842: config->mail_rcpt_allowfails ? 1L : 0L);
! 1843:
! 1844: /* curl 7.20.x */
! 1845: if(config->ftp_pret)
! 1846: my_setopt(curl, CURLOPT_FTP_USE_PRET, 1L);
! 1847:
! 1848: if(config->proto_present)
! 1849: my_setopt_flags(curl, CURLOPT_PROTOCOLS, config->proto);
! 1850: if(config->proto_redir_present)
! 1851: my_setopt_flags(curl, CURLOPT_REDIR_PROTOCOLS, config->proto_redir);
! 1852:
! 1853: if(config->content_disposition
! 1854: && (urlnode->flags & GETOUT_USEREMOTE))
! 1855: hdrcbdata->honor_cd_filename = TRUE;
! 1856: else
! 1857: hdrcbdata->honor_cd_filename = FALSE;
! 1858:
! 1859: hdrcbdata->outs = outs;
! 1860: hdrcbdata->heads = heads;
! 1861: hdrcbdata->etag_save = etag_save;
! 1862: hdrcbdata->global = global;
! 1863: hdrcbdata->config = config;
! 1864:
! 1865: my_setopt(curl, CURLOPT_HEADERFUNCTION, tool_header_cb);
! 1866: my_setopt(curl, CURLOPT_HEADERDATA, per);
! 1867:
! 1868: if(config->resolve)
! 1869: /* new in 7.21.3 */
! 1870: my_setopt_slist(curl, CURLOPT_RESOLVE, config->resolve);
! 1871:
! 1872: if(config->connect_to)
! 1873: /* new in 7.49.0 */
! 1874: my_setopt_slist(curl, CURLOPT_CONNECT_TO, config->connect_to);
! 1875:
! 1876: /* new in 7.21.4 */
! 1877: if(curlinfo->features & CURL_VERSION_TLSAUTH_SRP) {
! 1878: if(config->tls_username)
! 1879: my_setopt_str(curl, CURLOPT_TLSAUTH_USERNAME,
! 1880: config->tls_username);
! 1881: if(config->tls_password)
! 1882: my_setopt_str(curl, CURLOPT_TLSAUTH_PASSWORD,
! 1883: config->tls_password);
! 1884: if(config->tls_authtype)
! 1885: my_setopt_str(curl, CURLOPT_TLSAUTH_TYPE,
! 1886: config->tls_authtype);
! 1887: if(config->proxy_tls_username)
! 1888: my_setopt_str(curl, CURLOPT_PROXY_TLSAUTH_USERNAME,
! 1889: config->proxy_tls_username);
! 1890: if(config->proxy_tls_password)
! 1891: my_setopt_str(curl, CURLOPT_PROXY_TLSAUTH_PASSWORD,
! 1892: config->proxy_tls_password);
! 1893: if(config->proxy_tls_authtype)
! 1894: my_setopt_str(curl, CURLOPT_PROXY_TLSAUTH_TYPE,
! 1895: config->proxy_tls_authtype);
! 1896: }
! 1897:
! 1898: /* new in 7.22.0 */
! 1899: if(config->gssapi_delegation)
! 1900: my_setopt_str(curl, CURLOPT_GSSAPI_DELEGATION,
! 1901: config->gssapi_delegation);
! 1902:
! 1903: /* new in 7.25.0, 7.44.0 and 7.70.0 */
! 1904: {
! 1905: long mask = (config->ssl_allow_beast ? CURLSSLOPT_ALLOW_BEAST : 0) |
! 1906: (config->ssl_revoke_best_effort ?
! 1907: CURLSSLOPT_REVOKE_BEST_EFFORT : 0) |
! 1908: (config->ssl_no_revoke ? CURLSSLOPT_NO_REVOKE : 0);
! 1909: if(mask)
! 1910: my_setopt_bitmask(curl, CURLOPT_SSL_OPTIONS, mask);
! 1911: }
! 1912:
! 1913: if(config->proxy_ssl_allow_beast)
! 1914: my_setopt(curl, CURLOPT_PROXY_SSL_OPTIONS,
! 1915: (long)CURLSSLOPT_ALLOW_BEAST);
! 1916:
! 1917: if(config->mail_auth)
! 1918: my_setopt_str(curl, CURLOPT_MAIL_AUTH, config->mail_auth);
! 1919:
! 1920: /* new in 7.66.0 */
! 1921: if(config->sasl_authzid)
! 1922: my_setopt_str(curl, CURLOPT_SASL_AUTHZID, config->sasl_authzid);
! 1923:
! 1924: /* new in 7.31.0 */
! 1925: if(config->sasl_ir)
! 1926: my_setopt(curl, CURLOPT_SASL_IR, 1L);
! 1927:
! 1928: if(config->nonpn) {
! 1929: my_setopt(curl, CURLOPT_SSL_ENABLE_NPN, 0L);
! 1930: }
! 1931:
! 1932: if(config->noalpn) {
! 1933: my_setopt(curl, CURLOPT_SSL_ENABLE_ALPN, 0L);
! 1934: }
! 1935:
! 1936: /* new in 7.40.0, abstract support added in 7.53.0 */
! 1937: if(config->unix_socket_path) {
! 1938: if(config->abstract_unix_socket) {
! 1939: my_setopt_str(curl, CURLOPT_ABSTRACT_UNIX_SOCKET,
! 1940: config->unix_socket_path);
! 1941: }
! 1942: else {
! 1943: my_setopt_str(curl, CURLOPT_UNIX_SOCKET_PATH,
! 1944: config->unix_socket_path);
! 1945: }
! 1946: }
! 1947:
! 1948: /* new in 7.45.0 */
! 1949: if(config->proto_default)
! 1950: my_setopt_str(curl, CURLOPT_DEFAULT_PROTOCOL, config->proto_default);
! 1951:
! 1952: /* new in 7.47.0 */
! 1953: if(config->expect100timeout > 0)
! 1954: my_setopt_str(curl, CURLOPT_EXPECT_100_TIMEOUT_MS,
! 1955: (long)(config->expect100timeout*1000));
! 1956:
! 1957: /* new in 7.48.0 */
! 1958: if(config->tftp_no_options)
! 1959: my_setopt(curl, CURLOPT_TFTP_NO_OPTIONS, 1L);
! 1960:
! 1961: /* new in 7.59.0 */
! 1962: if(config->happy_eyeballs_timeout_ms != CURL_HET_DEFAULT)
! 1963: my_setopt(curl, CURLOPT_HAPPY_EYEBALLS_TIMEOUT_MS,
! 1964: config->happy_eyeballs_timeout_ms);
! 1965:
! 1966: /* new in 7.60.0 */
! 1967: if(config->haproxy_protocol)
! 1968: my_setopt(curl, CURLOPT_HAPROXYPROTOCOL, 1L);
! 1969:
! 1970: if(config->disallow_username_in_url)
! 1971: my_setopt(curl, CURLOPT_DISALLOW_USERNAME_IN_URL, 1L);
! 1972:
! 1973: if(config->altsvc)
! 1974: my_setopt_str(curl, CURLOPT_ALTSVC, config->altsvc);
! 1975:
! 1976: #ifdef USE_METALINK
! 1977: if(!metalink && config->use_metalink) {
! 1978: outs->metalink_parser = metalink_parser_context_new();
! 1979: if(outs->metalink_parser == NULL) {
! 1980: result = CURLE_OUT_OF_MEMORY;
! 1981: break;
! 1982: }
! 1983: fprintf(config->global->errors,
! 1984: "Metalink: parsing (%s) metalink/XML...\n", per->this_url);
! 1985: }
! 1986: else if(metalink)
! 1987: fprintf(config->global->errors,
! 1988: "Metalink: fetching (%s) from (%s)...\n",
! 1989: mlfile->filename, per->this_url);
! 1990: #endif /* USE_METALINK */
! 1991:
! 1992: per->metalink = metalink;
! 1993: /* initialize retry vars for loop below */
! 1994: per->retry_sleep_default = (config->retry_delay) ?
! 1995: config->retry_delay*1000L : RETRY_SLEEP_DEFAULT; /* ms */
! 1996: per->retry_numretries = config->req_retry;
! 1997: per->retry_sleep = per->retry_sleep_default; /* ms */
! 1998: per->retrystart = tvnow();
! 1999:
! 2000: state->li++;
! 2001: /* Here's looping around each globbed URL */
! 2002: if(state->li >= urlnum) {
! 2003: state->li = 0;
! 2004: state->urlnum = 0; /* forced reglob of URLs */
! 2005: glob_cleanup(state->urls);
! 2006: state->urls = NULL;
! 2007: state->up++;
! 2008: Curl_safefree(state->uploadfile); /* clear it to get the next */
! 2009: }
! 2010: }
! 2011: else {
! 2012: /* Free this URL node data without destroying the
! 2013: the node itself nor modifying next pointer. */
! 2014: Curl_safefree(urlnode->outfile);
! 2015: Curl_safefree(urlnode->infile);
! 2016: urlnode->flags = 0;
! 2017: glob_cleanup(state->urls);
! 2018: state->urls = NULL;
! 2019: state->urlnum = 0;
! 2020:
! 2021: Curl_safefree(state->outfiles);
! 2022: Curl_safefree(state->uploadfile);
! 2023: if(state->inglob) {
! 2024: /* Free list of globbed upload files */
! 2025: glob_cleanup(state->inglob);
! 2026: state->inglob = NULL;
! 2027: }
! 2028: config->state.urlnode = urlnode->next;
! 2029: state->up = 0;
! 2030: continue;
! 2031: }
! 2032: }
! 2033: break;
! 2034: }
! 2035:
! 2036: if(!*added || result) {
! 2037: *added = FALSE;
! 2038: single_transfer_cleanup(config);
! 2039: }
! 2040: return result;
! 2041: }
! 2042:
! 2043: static long all_added; /* number of easy handles currently added */
! 2044:
! 2045: /*
! 2046: * add_parallel_transfers() sets 'morep' to TRUE if there are more transfers
! 2047: * to add even after this call returns. sets 'addedp' to TRUE if one or more
! 2048: * transfers were added.
! 2049: */
! 2050: static CURLcode add_parallel_transfers(struct GlobalConfig *global,
! 2051: CURLM *multi,
! 2052: CURLSH *share,
! 2053: bool *morep,
! 2054: bool *addedp)
! 2055: {
! 2056: struct per_transfer *per;
! 2057: CURLcode result = CURLE_OK;
! 2058: CURLMcode mcode;
! 2059: *addedp = FALSE;
! 2060: *morep = FALSE;
! 2061: result = create_transfer(global, share, addedp);
! 2062: if(result)
! 2063: return result;
! 2064: for(per = transfers; per && (all_added < global->parallel_max);
! 2065: per = per->next) {
! 2066: bool getadded = FALSE;
! 2067: if(per->added)
! 2068: /* already added */
! 2069: continue;
! 2070:
! 2071: result = pre_transfer(global, per);
! 2072: if(result)
! 2073: break;
! 2074:
! 2075: /* parallel connect means that we don't set PIPEWAIT since pipewait
! 2076: will make libcurl prefer multiplexing */
! 2077: (void)curl_easy_setopt(per->curl, CURLOPT_PIPEWAIT,
! 2078: global->parallel_connect ? 0L : 1L);
! 2079: (void)curl_easy_setopt(per->curl, CURLOPT_PRIVATE, per);
! 2080: (void)curl_easy_setopt(per->curl, CURLOPT_XFERINFOFUNCTION, xferinfo_cb);
! 2081: (void)curl_easy_setopt(per->curl, CURLOPT_XFERINFODATA, per);
! 2082:
! 2083: mcode = curl_multi_add_handle(multi, per->curl);
! 2084: if(mcode)
! 2085: return CURLE_OUT_OF_MEMORY;
! 2086:
! 2087: result = create_transfer(global, share, &getadded);
! 2088: if(result)
! 2089: return result;
! 2090: per->added = TRUE;
! 2091: all_added++;
! 2092: *addedp = TRUE;
! 2093: }
! 2094: *morep = per ? TRUE : FALSE;
! 2095: return CURLE_OK;
! 2096: }
! 2097:
! 2098: static CURLcode parallel_transfers(struct GlobalConfig *global,
! 2099: CURLSH *share)
! 2100: {
! 2101: CURLM *multi;
! 2102: CURLMcode mcode = CURLM_OK;
! 2103: CURLcode result = CURLE_OK;
! 2104: int still_running = 1;
! 2105: struct timeval start = tvnow();
! 2106: bool more_transfers;
! 2107: bool added_transfers;
! 2108:
! 2109: multi = curl_multi_init();
! 2110: if(!multi)
! 2111: return CURLE_OUT_OF_MEMORY;
! 2112:
! 2113: result = add_parallel_transfers(global, multi, share,
! 2114: &more_transfers, &added_transfers);
! 2115: if(result) {
! 2116: curl_multi_cleanup(multi);
! 2117: return result;
! 2118: }
! 2119:
! 2120: while(!mcode && (still_running || more_transfers)) {
! 2121: mcode = curl_multi_poll(multi, NULL, 0, 1000, NULL);
! 2122: if(!mcode)
! 2123: mcode = curl_multi_perform(multi, &still_running);
! 2124:
! 2125: progress_meter(global, &start, FALSE);
! 2126:
! 2127: if(!mcode) {
! 2128: int rc;
! 2129: CURLMsg *msg;
! 2130: bool removed = FALSE;
! 2131: do {
! 2132: msg = curl_multi_info_read(multi, &rc);
! 2133: if(msg) {
! 2134: bool retry;
! 2135: struct per_transfer *ended;
! 2136: CURL *easy = msg->easy_handle;
! 2137: result = msg->data.result;
! 2138: curl_easy_getinfo(easy, CURLINFO_PRIVATE, (void *)&ended);
! 2139: curl_multi_remove_handle(multi, easy);
! 2140:
! 2141: result = post_per_transfer(global, ended, result, &retry);
! 2142: if(retry)
! 2143: continue;
! 2144: progress_finalize(ended); /* before it goes away */
! 2145: all_added--; /* one fewer added */
! 2146: removed = TRUE;
! 2147: (void)del_per_transfer(ended);
! 2148: }
! 2149: } while(msg);
! 2150: if(removed) {
! 2151: /* one or more transfers completed, add more! */
! 2152: (void)add_parallel_transfers(global, multi, share,
! 2153: &more_transfers,
! 2154: &added_transfers);
! 2155: if(added_transfers)
! 2156: /* we added new ones, make sure the loop doesn't exit yet */
! 2157: still_running = 1;
! 2158: }
! 2159: }
! 2160: }
! 2161:
! 2162: (void)progress_meter(global, &start, TRUE);
! 2163:
! 2164: /* Make sure to return some kind of error if there was a multi problem */
! 2165: if(mcode) {
! 2166: result = (mcode == CURLM_OUT_OF_MEMORY) ? CURLE_OUT_OF_MEMORY :
! 2167: /* The other multi errors should never happen, so return
! 2168: something suitably generic */
! 2169: CURLE_BAD_FUNCTION_ARGUMENT;
! 2170: }
! 2171:
! 2172: curl_multi_cleanup(multi);
! 2173:
! 2174: return result;
! 2175: }
! 2176:
! 2177: static CURLcode serial_transfers(struct GlobalConfig *global,
! 2178: CURLSH *share)
! 2179: {
! 2180: CURLcode returncode = CURLE_OK;
! 2181: CURLcode result = CURLE_OK;
! 2182: struct per_transfer *per;
! 2183: bool added = FALSE;
! 2184:
! 2185: result = create_transfer(global, share, &added);
! 2186: if(result || !added)
! 2187: return result;
! 2188: for(per = transfers; per;) {
! 2189: bool retry;
! 2190: bool bailout = FALSE;
! 2191: result = pre_transfer(global, per);
! 2192: if(result)
! 2193: break;
! 2194:
! 2195: #ifndef CURL_DISABLE_LIBCURL_OPTION
! 2196: if(global->libcurl) {
! 2197: result = easysrc_perform();
! 2198: if(result)
! 2199: break;
! 2200: }
! 2201: #endif
! 2202: #ifdef CURLDEBUG
! 2203: if(global->test_event_based)
! 2204: result = curl_easy_perform_ev(per->curl);
! 2205: else
! 2206: #endif
! 2207: result = curl_easy_perform(per->curl);
! 2208:
! 2209: /* store the result of the actual transfer */
! 2210: returncode = result;
! 2211:
! 2212: result = post_per_transfer(global, per, result, &retry);
! 2213: if(retry)
! 2214: continue;
! 2215:
! 2216: /* Bail out upon critical errors or --fail-early */
! 2217: if(result || is_fatal_error(returncode) ||
! 2218: (returncode && global->fail_early))
! 2219: bailout = TRUE;
! 2220: else {
! 2221: /* setup the next one just before we delete this */
! 2222: result = create_transfer(global, share, &added);
! 2223: if(result)
! 2224: bailout = TRUE;
! 2225: }
! 2226:
! 2227: /* Release metalink related resources here */
! 2228: delete_metalinkfile(per->mlfile);
! 2229:
! 2230: per = del_per_transfer(per);
! 2231:
! 2232: if(bailout)
! 2233: break;
! 2234: }
! 2235: if(returncode)
! 2236: /* returncode errors have priority */
! 2237: result = returncode;
! 2238:
! 2239: if(result)
! 2240: single_transfer_cleanup(global->current);
! 2241:
! 2242: return result;
! 2243: }
! 2244:
! 2245: /* setup a transfer for the given config */
! 2246: static CURLcode transfer_per_config(struct GlobalConfig *global,
! 2247: struct OperationConfig *config,
! 2248: CURLSH *share,
! 2249: bool *added)
! 2250: {
! 2251: CURLcode result = CURLE_OK;
! 2252: bool capath_from_env;
! 2253: *added = FALSE;
! 2254:
! 2255: /* Check we have a url */
! 2256: if(!config->url_list || !config->url_list->url) {
! 2257: helpf(global->errors, "no URL specified!\n");
! 2258: return CURLE_FAILED_INIT;
! 2259: }
! 2260:
! 2261: /* On WIN32 we can't set the path to curl-ca-bundle.crt
! 2262: * at compile time. So we look here for the file in two ways:
! 2263: * 1: look at the environment variable CURL_CA_BUNDLE for a path
! 2264: * 2: if #1 isn't found, use the windows API function SearchPath()
! 2265: * to find it along the app's path (includes app's dir and CWD)
! 2266: *
! 2267: * We support the environment variable thing for non-Windows platforms
! 2268: * too. Just for the sake of it.
! 2269: */
! 2270: capath_from_env = false;
! 2271: if(!config->cacert &&
! 2272: !config->capath &&
! 2273: !config->insecure_ok) {
! 2274: CURL *curltls = curl_easy_init();
! 2275: struct curl_tlssessioninfo *tls_backend_info = NULL;
! 2276:
! 2277: /* With the addition of CAINFO support for Schannel, this search could find
! 2278: * a certificate bundle that was previously ignored. To maintain backward
! 2279: * compatibility, only perform this search if not using Schannel.
! 2280: */
! 2281: result = curl_easy_getinfo(curltls, CURLINFO_TLS_SSL_PTR,
! 2282: &tls_backend_info);
! 2283: if(result)
! 2284: return result;
! 2285:
! 2286: /* Set the CA cert locations specified in the environment. For Windows if
! 2287: * no environment-specified filename is found then check for CA bundle
! 2288: * default filename curl-ca-bundle.crt in the user's PATH.
! 2289: *
! 2290: * If Schannel is the selected SSL backend then these locations are
! 2291: * ignored. We allow setting CA location for schannel only when explicitly
! 2292: * specified by the user via CURLOPT_CAINFO / --cacert.
! 2293: */
! 2294: if(tls_backend_info->backend != CURLSSLBACKEND_SCHANNEL) {
! 2295: char *env;
! 2296: env = curlx_getenv("CURL_CA_BUNDLE");
! 2297: if(env) {
! 2298: config->cacert = strdup(env);
! 2299: if(!config->cacert) {
! 2300: curl_free(env);
! 2301: errorf(global, "out of memory\n");
! 2302: return CURLE_OUT_OF_MEMORY;
! 2303: }
! 2304: }
! 2305: else {
! 2306: env = curlx_getenv("SSL_CERT_DIR");
! 2307: if(env) {
! 2308: config->capath = strdup(env);
! 2309: if(!config->capath) {
! 2310: curl_free(env);
! 2311: helpf(global->errors, "out of memory\n");
! 2312: return CURLE_OUT_OF_MEMORY;
! 2313: }
! 2314: capath_from_env = true;
! 2315: }
! 2316: else {
! 2317: env = curlx_getenv("SSL_CERT_FILE");
! 2318: if(env) {
! 2319: config->cacert = strdup(env);
! 2320: if(!config->cacert) {
! 2321: curl_free(env);
! 2322: errorf(global, "out of memory\n");
! 2323: return CURLE_OUT_OF_MEMORY;
! 2324: }
! 2325: }
! 2326: }
! 2327: }
! 2328:
! 2329: if(env)
! 2330: curl_free(env);
! 2331: #ifdef WIN32
! 2332: else {
! 2333: result = FindWin32CACert(config, tls_backend_info->backend,
! 2334: "curl-ca-bundle.crt");
! 2335: }
! 2336: #endif
! 2337: }
! 2338: curl_easy_cleanup(curltls);
! 2339: }
! 2340:
! 2341: if(!result)
! 2342: result = single_transfer(global, config, share, capath_from_env, added);
! 2343:
! 2344: return result;
! 2345: }
! 2346:
! 2347: /*
! 2348: * 'create_transfer' gets the details and sets up a new transfer if 'added'
! 2349: * returns TRUE.
! 2350: */
! 2351: static CURLcode create_transfer(struct GlobalConfig *global,
! 2352: CURLSH *share,
! 2353: bool *added)
! 2354: {
! 2355: CURLcode result = CURLE_OK;
! 2356: *added = FALSE;
! 2357: while(global->current) {
! 2358: result = transfer_per_config(global, global->current, share, added);
! 2359: if(!result && !*added) {
! 2360: /* when one set is drained, continue to next */
! 2361: global->current = global->current->next;
! 2362: continue;
! 2363: }
! 2364: break;
! 2365: }
! 2366: return result;
! 2367: }
! 2368:
! 2369: static CURLcode run_all_transfers(struct GlobalConfig *global,
! 2370: CURLSH *share,
! 2371: CURLcode result)
! 2372: {
! 2373: /* Save the values of noprogress and isatty to restore them later on */
! 2374: bool orig_noprogress = global->noprogress;
! 2375: bool orig_isatty = global->isatty;
! 2376: struct per_transfer *per;
! 2377:
! 2378: /* Time to actually do the transfers */
! 2379: if(!result) {
! 2380: if(global->parallel)
! 2381: result = parallel_transfers(global, share);
! 2382: else
! 2383: result = serial_transfers(global, share);
! 2384: }
! 2385:
! 2386: /* cleanup if there are any left */
! 2387: for(per = transfers; per;) {
! 2388: bool retry;
! 2389: CURLcode result2 = post_per_transfer(global, per, result, &retry);
! 2390: if(!result)
! 2391: /* don't overwrite the original error */
! 2392: result = result2;
! 2393:
! 2394: /* Free list of given URLs */
! 2395: clean_getout(per->config);
! 2396:
! 2397: /* Release metalink related resources here */
! 2398: clean_metalink(per->config);
! 2399: per = del_per_transfer(per);
! 2400: }
! 2401:
! 2402: /* Reset the global config variables */
! 2403: global->noprogress = orig_noprogress;
! 2404: global->isatty = orig_isatty;
! 2405:
! 2406:
! 2407: return result;
! 2408: }
! 2409:
! 2410: CURLcode operate(struct GlobalConfig *global, int argc, argv_item_t argv[])
! 2411: {
! 2412: CURLcode result = CURLE_OK;
! 2413:
! 2414: /* Setup proper locale from environment */
! 2415: #ifdef HAVE_SETLOCALE
! 2416: setlocale(LC_ALL, "");
! 2417: #endif
! 2418:
! 2419: /* Parse .curlrc if necessary */
! 2420: if((argc == 1) ||
! 2421: (!curl_strequal(argv[1], "-q") &&
! 2422: !curl_strequal(argv[1], "--disable"))) {
! 2423: parseconfig(NULL, global); /* ignore possible failure */
! 2424:
! 2425: /* If we had no arguments then make sure a url was specified in .curlrc */
! 2426: if((argc < 2) && (!global->first->url_list)) {
! 2427: helpf(global->errors, NULL);
! 2428: result = CURLE_FAILED_INIT;
! 2429: }
! 2430: }
! 2431:
! 2432: if(!result) {
! 2433: /* Parse the command line arguments */
! 2434: ParameterError res = parse_args(global, argc, argv);
! 2435: if(res) {
! 2436: result = CURLE_OK;
! 2437:
! 2438: /* Check if we were asked for the help */
! 2439: if(res == PARAM_HELP_REQUESTED)
! 2440: tool_help();
! 2441: /* Check if we were asked for the manual */
! 2442: else if(res == PARAM_MANUAL_REQUESTED)
! 2443: hugehelp();
! 2444: /* Check if we were asked for the version information */
! 2445: else if(res == PARAM_VERSION_INFO_REQUESTED)
! 2446: tool_version_info();
! 2447: /* Check if we were asked to list the SSL engines */
! 2448: else if(res == PARAM_ENGINES_REQUESTED)
! 2449: tool_list_engines();
! 2450: else if(res == PARAM_LIBCURL_UNSUPPORTED_PROTOCOL)
! 2451: result = CURLE_UNSUPPORTED_PROTOCOL;
! 2452: else
! 2453: result = CURLE_FAILED_INIT;
! 2454: }
! 2455: else {
! 2456: #ifndef CURL_DISABLE_LIBCURL_OPTION
! 2457: if(global->libcurl) {
! 2458: /* Initialise the libcurl source output */
! 2459: result = easysrc_init();
! 2460: }
! 2461: #endif
! 2462:
! 2463: /* Perform the main operations */
! 2464: if(!result) {
! 2465: size_t count = 0;
! 2466: struct OperationConfig *operation = global->first;
! 2467: CURLSH *share = curl_share_init();
! 2468: if(!share) {
! 2469: #ifndef CURL_DISABLE_LIBCURL_OPTION
! 2470: if(global->libcurl) {
! 2471: /* Cleanup the libcurl source output */
! 2472: easysrc_cleanup();
! 2473: }
! 2474: #endif
! 2475: return CURLE_OUT_OF_MEMORY;
! 2476: }
! 2477:
! 2478: curl_share_setopt(share, CURLSHOPT_SHARE, CURL_LOCK_DATA_COOKIE);
! 2479: curl_share_setopt(share, CURLSHOPT_SHARE, CURL_LOCK_DATA_DNS);
! 2480: curl_share_setopt(share, CURLSHOPT_SHARE, CURL_LOCK_DATA_SSL_SESSION);
! 2481: curl_share_setopt(share, CURLSHOPT_SHARE, CURL_LOCK_DATA_CONNECT);
! 2482: curl_share_setopt(share, CURLSHOPT_SHARE, CURL_LOCK_DATA_PSL);
! 2483:
! 2484: /* Get the required arguments for each operation */
! 2485: do {
! 2486: result = get_args(operation, count++);
! 2487:
! 2488: operation = operation->next;
! 2489: } while(!result && operation);
! 2490:
! 2491: /* Set the current operation pointer */
! 2492: global->current = global->first;
! 2493:
! 2494: /* now run! */
! 2495: result = run_all_transfers(global, share, result);
! 2496:
! 2497: curl_share_cleanup(share);
! 2498: #ifndef CURL_DISABLE_LIBCURL_OPTION
! 2499: if(global->libcurl) {
! 2500: /* Cleanup the libcurl source output */
! 2501: easysrc_cleanup();
! 2502:
! 2503: /* Dump the libcurl code if previously enabled */
! 2504: dumpeasysrc(global);
! 2505: }
! 2506: #endif
! 2507: }
! 2508: else
! 2509: errorf(global, "out of memory\n");
! 2510: }
! 2511: }
! 2512:
! 2513: return result;
! 2514: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>