Annotation of embedaddon/curl/src/tool_getparam.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: #include "strcase.h"
! 25:
! 26: #define ENABLE_CURLX_PRINTF
! 27: /* use our own printf() functions */
! 28: #include "curlx.h"
! 29:
! 30: #include "tool_binmode.h"
! 31: #include "tool_cfgable.h"
! 32: #include "tool_cb_prg.h"
! 33: #include "tool_convert.h"
! 34: #include "tool_filetime.h"
! 35: #include "tool_formparse.h"
! 36: #include "tool_getparam.h"
! 37: #include "tool_helpers.h"
! 38: #include "tool_libinfo.h"
! 39: #include "tool_metalink.h"
! 40: #include "tool_msgs.h"
! 41: #include "tool_paramhlp.h"
! 42: #include "tool_parsecfg.h"
! 43: #include "tool_main.h"
! 44:
! 45: #include "memdebug.h" /* keep this as LAST include */
! 46:
! 47: #ifdef MSDOS
! 48: # define USE_WATT32
! 49: #endif
! 50:
! 51: #define GetStr(str,val) do { \
! 52: if(*(str)) { \
! 53: free(*(str)); \
! 54: *(str) = NULL; \
! 55: } \
! 56: if((val)) { \
! 57: *(str) = strdup((val)); \
! 58: if(!(*(str))) \
! 59: return PARAM_NO_MEM; \
! 60: } \
! 61: } while(0)
! 62:
! 63: struct LongShort {
! 64: const char *letter; /* short name option */
! 65: const char *lname; /* long name option */
! 66: enum {
! 67: ARG_NONE, /* stand-alone but not a boolean */
! 68: ARG_BOOL, /* accepts a --no-[name] prefix */
! 69: ARG_STRING, /* requires an argument */
! 70: ARG_FILENAME /* requires an argument, usually a file name */
! 71: } desc;
! 72: };
! 73:
! 74: static const struct LongShort aliases[]= {
! 75: /* 'letter' strings with more than one character have *no* short option to
! 76: mention. */
! 77: {"*@", "url", ARG_STRING},
! 78: {"*4", "dns-ipv4-addr", ARG_STRING},
! 79: {"*6", "dns-ipv6-addr", ARG_STRING},
! 80: {"*a", "random-file", ARG_FILENAME},
! 81: {"*b", "egd-file", ARG_STRING},
! 82: {"*B", "oauth2-bearer", ARG_STRING},
! 83: {"*c", "connect-timeout", ARG_STRING},
! 84: {"*C", "doh-url" , ARG_STRING},
! 85: {"*d", "ciphers", ARG_STRING},
! 86: {"*D", "dns-interface", ARG_STRING},
! 87: {"*e", "disable-epsv", ARG_BOOL},
! 88: {"*f", "disallow-username-in-url", ARG_BOOL},
! 89: {"*E", "epsv", ARG_BOOL},
! 90: /* 'epsv' made like this to make --no-epsv and --epsv to work
! 91: although --disable-epsv is the documented option */
! 92: {"*F", "dns-servers", ARG_STRING},
! 93: {"*g", "trace", ARG_FILENAME},
! 94: {"*G", "npn", ARG_BOOL},
! 95: {"*h", "trace-ascii", ARG_FILENAME},
! 96: {"*H", "alpn", ARG_BOOL},
! 97: {"*i", "limit-rate", ARG_STRING},
! 98: {"*j", "compressed", ARG_BOOL},
! 99: {"*J", "tr-encoding", ARG_BOOL},
! 100: {"*k", "digest", ARG_BOOL},
! 101: {"*l", "negotiate", ARG_BOOL},
! 102: {"*m", "ntlm", ARG_BOOL},
! 103: {"*M", "ntlm-wb", ARG_BOOL},
! 104: {"*n", "basic", ARG_BOOL},
! 105: {"*o", "anyauth", ARG_BOOL},
! 106: #ifdef USE_WATT32
! 107: {"*p", "wdebug", ARG_BOOL},
! 108: #endif
! 109: {"*q", "ftp-create-dirs", ARG_BOOL},
! 110: {"*r", "create-dirs", ARG_BOOL},
! 111: {"*s", "max-redirs", ARG_STRING},
! 112: {"*t", "proxy-ntlm", ARG_BOOL},
! 113: {"*u", "crlf", ARG_BOOL},
! 114: {"*v", "stderr", ARG_FILENAME},
! 115: {"*w", "interface", ARG_STRING},
! 116: {"*x", "krb", ARG_STRING},
! 117: {"*x", "krb4", ARG_STRING},
! 118: /* 'krb4' is the previous name */
! 119: {"*X", "haproxy-protocol", ARG_BOOL},
! 120: {"*y", "max-filesize", ARG_STRING},
! 121: {"*z", "disable-eprt", ARG_BOOL},
! 122: {"*Z", "eprt", ARG_BOOL},
! 123: /* 'eprt' made like this to make --no-eprt and --eprt to work
! 124: although --disable-eprt is the documented option */
! 125: {"*~", "xattr", ARG_BOOL},
! 126: {"$a", "ftp-ssl", ARG_BOOL},
! 127: /* 'ftp-ssl' deprecated name since 7.20.0 */
! 128: {"$a", "ssl", ARG_BOOL},
! 129: /* 'ssl' new option name in 7.20.0, previously this was ftp-ssl */
! 130: {"$b", "ftp-pasv", ARG_BOOL},
! 131: {"$c", "socks5", ARG_STRING},
! 132: {"$d", "tcp-nodelay", ARG_BOOL},
! 133: {"$e", "proxy-digest", ARG_BOOL},
! 134: {"$f", "proxy-basic", ARG_BOOL},
! 135: {"$g", "retry", ARG_STRING},
! 136: {"$V", "retry-connrefused", ARG_BOOL},
! 137: {"$h", "retry-delay", ARG_STRING},
! 138: {"$i", "retry-max-time", ARG_STRING},
! 139: {"$k", "proxy-negotiate", ARG_BOOL},
! 140: {"$m", "ftp-account", ARG_STRING},
! 141: {"$n", "proxy-anyauth", ARG_BOOL},
! 142: {"$o", "trace-time", ARG_BOOL},
! 143: {"$p", "ignore-content-length", ARG_BOOL},
! 144: {"$q", "ftp-skip-pasv-ip", ARG_BOOL},
! 145: {"$r", "ftp-method", ARG_STRING},
! 146: {"$s", "local-port", ARG_STRING},
! 147: {"$t", "socks4", ARG_STRING},
! 148: {"$T", "socks4a", ARG_STRING},
! 149: {"$u", "ftp-alternative-to-user", ARG_STRING},
! 150: {"$v", "ftp-ssl-reqd", ARG_BOOL},
! 151: /* 'ftp-ssl-reqd' deprecated name since 7.20.0 */
! 152: {"$v", "ssl-reqd", ARG_BOOL},
! 153: /* 'ssl-reqd' new in 7.20.0, previously this was ftp-ssl-reqd */
! 154: {"$w", "sessionid", ARG_BOOL},
! 155: /* 'sessionid' listed as --no-sessionid in the help */
! 156: {"$x", "ftp-ssl-control", ARG_BOOL},
! 157: {"$y", "ftp-ssl-ccc", ARG_BOOL},
! 158: {"$j", "ftp-ssl-ccc-mode", ARG_STRING},
! 159: {"$z", "libcurl", ARG_STRING},
! 160: {"$#", "raw", ARG_BOOL},
! 161: {"$0", "post301", ARG_BOOL},
! 162: {"$1", "keepalive", ARG_BOOL},
! 163: /* 'keepalive' listed as --no-keepalive in the help */
! 164: {"$2", "socks5-hostname", ARG_STRING},
! 165: {"$3", "keepalive-time", ARG_STRING},
! 166: {"$4", "post302", ARG_BOOL},
! 167: {"$5", "noproxy", ARG_STRING},
! 168: {"$7", "socks5-gssapi-nec", ARG_BOOL},
! 169: {"$8", "proxy1.0", ARG_STRING},
! 170: {"$9", "tftp-blksize", ARG_STRING},
! 171: {"$A", "mail-from", ARG_STRING},
! 172: {"$B", "mail-rcpt", ARG_STRING},
! 173: {"$C", "ftp-pret", ARG_BOOL},
! 174: {"$D", "proto", ARG_STRING},
! 175: {"$E", "proto-redir", ARG_STRING},
! 176: {"$F", "resolve", ARG_STRING},
! 177: {"$G", "delegation", ARG_STRING},
! 178: {"$H", "mail-auth", ARG_STRING},
! 179: {"$I", "post303", ARG_BOOL},
! 180: {"$J", "metalink", ARG_BOOL},
! 181: {"$6", "sasl-authzid", ARG_STRING},
! 182: {"$K", "sasl-ir", ARG_BOOL },
! 183: {"$L", "test-event", ARG_BOOL},
! 184: {"$M", "unix-socket", ARG_FILENAME},
! 185: {"$N", "path-as-is", ARG_BOOL},
! 186: {"$O", "socks5-gssapi-service", ARG_STRING},
! 187: /* 'socks5-gssapi-service' merged with'proxy-service-name' and
! 188: deprecated since 7.49.0 */
! 189: {"$O", "proxy-service-name", ARG_STRING},
! 190: {"$P", "service-name", ARG_STRING},
! 191: {"$Q", "proto-default", ARG_STRING},
! 192: {"$R", "expect100-timeout", ARG_STRING},
! 193: {"$S", "tftp-no-options", ARG_BOOL},
! 194: {"$U", "connect-to", ARG_STRING},
! 195: {"$W", "abstract-unix-socket", ARG_FILENAME},
! 196: {"$X", "tls-max", ARG_STRING},
! 197: {"$Y", "suppress-connect-headers", ARG_BOOL},
! 198: {"$Z", "compressed-ssh", ARG_BOOL},
! 199: {"$~", "happy-eyeballs-timeout-ms", ARG_STRING},
! 200: {"0", "http1.0", ARG_NONE},
! 201: {"01", "http1.1", ARG_NONE},
! 202: {"02", "http2", ARG_NONE},
! 203: {"03", "http2-prior-knowledge", ARG_NONE},
! 204: {"04", "http3", ARG_NONE},
! 205: {"09", "http0.9", ARG_BOOL},
! 206: {"1", "tlsv1", ARG_NONE},
! 207: {"10", "tlsv1.0", ARG_NONE},
! 208: {"11", "tlsv1.1", ARG_NONE},
! 209: {"12", "tlsv1.2", ARG_NONE},
! 210: {"13", "tlsv1.3", ARG_NONE},
! 211: {"1A", "tls13-ciphers", ARG_STRING},
! 212: {"1B", "proxy-tls13-ciphers", ARG_STRING},
! 213: {"2", "sslv2", ARG_NONE},
! 214: {"3", "sslv3", ARG_NONE},
! 215: {"4", "ipv4", ARG_NONE},
! 216: {"6", "ipv6", ARG_NONE},
! 217: {"a", "append", ARG_BOOL},
! 218: {"A", "user-agent", ARG_STRING},
! 219: {"b", "cookie", ARG_STRING},
! 220: {"ba", "alt-svc", ARG_STRING},
! 221: {"B", "use-ascii", ARG_BOOL},
! 222: {"c", "cookie-jar", ARG_STRING},
! 223: {"C", "continue-at", ARG_STRING},
! 224: {"d", "data", ARG_STRING},
! 225: {"dr", "data-raw", ARG_STRING},
! 226: {"da", "data-ascii", ARG_STRING},
! 227: {"db", "data-binary", ARG_STRING},
! 228: {"de", "data-urlencode", ARG_STRING},
! 229: {"D", "dump-header", ARG_FILENAME},
! 230: {"e", "referer", ARG_STRING},
! 231: {"E", "cert", ARG_FILENAME},
! 232: {"Ea", "cacert", ARG_FILENAME},
! 233: {"Eb", "cert-type", ARG_STRING},
! 234: {"Ec", "key", ARG_FILENAME},
! 235: {"Ed", "key-type", ARG_STRING},
! 236: {"Ee", "pass", ARG_STRING},
! 237: {"Ef", "engine", ARG_STRING},
! 238: {"Eg", "capath", ARG_FILENAME},
! 239: {"Eh", "pubkey", ARG_STRING},
! 240: {"Ei", "hostpubmd5", ARG_STRING},
! 241: {"Ej", "crlfile", ARG_FILENAME},
! 242: {"Ek", "tlsuser", ARG_STRING},
! 243: {"El", "tlspassword", ARG_STRING},
! 244: {"Em", "tlsauthtype", ARG_STRING},
! 245: {"En", "ssl-allow-beast", ARG_BOOL},
! 246: /* Eo */
! 247: {"Ep", "pinnedpubkey", ARG_STRING},
! 248: {"EP", "proxy-pinnedpubkey", ARG_STRING},
! 249: {"Eq", "cert-status", ARG_BOOL},
! 250: {"Er", "false-start", ARG_BOOL},
! 251: {"Es", "ssl-no-revoke", ARG_BOOL},
! 252: {"ES", "ssl-revoke-best-effort", ARG_BOOL},
! 253: {"Et", "tcp-fastopen", ARG_BOOL},
! 254: {"Eu", "proxy-tlsuser", ARG_STRING},
! 255: {"Ev", "proxy-tlspassword", ARG_STRING},
! 256: {"Ew", "proxy-tlsauthtype", ARG_STRING},
! 257: {"Ex", "proxy-cert", ARG_FILENAME},
! 258: {"Ey", "proxy-cert-type", ARG_STRING},
! 259: {"Ez", "proxy-key", ARG_FILENAME},
! 260: {"E0", "proxy-key-type", ARG_STRING},
! 261: {"E1", "proxy-pass", ARG_STRING},
! 262: {"E2", "proxy-ciphers", ARG_STRING},
! 263: {"E3", "proxy-crlfile", ARG_FILENAME},
! 264: {"E4", "proxy-ssl-allow-beast", ARG_BOOL},
! 265: {"E5", "login-options", ARG_STRING},
! 266: {"E6", "proxy-cacert", ARG_FILENAME},
! 267: {"E7", "proxy-capath", ARG_FILENAME},
! 268: {"E8", "proxy-insecure", ARG_BOOL},
! 269: {"E9", "proxy-tlsv1", ARG_NONE},
! 270: {"EA", "socks5-basic", ARG_BOOL},
! 271: {"EB", "socks5-gssapi", ARG_BOOL},
! 272: {"EC", "etag-save", ARG_FILENAME},
! 273: {"ED", "etag-compare", ARG_FILENAME},
! 274: {"f", "fail", ARG_BOOL},
! 275: {"fa", "fail-early", ARG_BOOL},
! 276: {"fb", "styled-output", ARG_BOOL},
! 277: {"fc", "mail-rcpt-allowfails", ARG_BOOL},
! 278: {"F", "form", ARG_STRING},
! 279: {"Fs", "form-string", ARG_STRING},
! 280: {"g", "globoff", ARG_BOOL},
! 281: {"G", "get", ARG_NONE},
! 282: {"Ga", "request-target", ARG_STRING},
! 283: {"h", "help", ARG_BOOL},
! 284: {"H", "header", ARG_STRING},
! 285: {"Hp", "proxy-header", ARG_STRING},
! 286: {"i", "include", ARG_BOOL},
! 287: {"I", "head", ARG_BOOL},
! 288: {"j", "junk-session-cookies", ARG_BOOL},
! 289: {"J", "remote-header-name", ARG_BOOL},
! 290: {"k", "insecure", ARG_BOOL},
! 291: {"K", "config", ARG_FILENAME},
! 292: {"l", "list-only", ARG_BOOL},
! 293: {"L", "location", ARG_BOOL},
! 294: {"Lt", "location-trusted", ARG_BOOL},
! 295: {"m", "max-time", ARG_STRING},
! 296: {"M", "manual", ARG_BOOL},
! 297: {"n", "netrc", ARG_BOOL},
! 298: {"no", "netrc-optional", ARG_BOOL},
! 299: {"ne", "netrc-file", ARG_FILENAME},
! 300: {"N", "buffer", ARG_BOOL},
! 301: /* 'buffer' listed as --no-buffer in the help */
! 302: {"o", "output", ARG_FILENAME},
! 303: {"O", "remote-name", ARG_NONE},
! 304: {"Oa", "remote-name-all", ARG_BOOL},
! 305: {"p", "proxytunnel", ARG_BOOL},
! 306: {"P", "ftp-port", ARG_STRING},
! 307: {"q", "disable", ARG_BOOL},
! 308: {"Q", "quote", ARG_STRING},
! 309: {"r", "range", ARG_STRING},
! 310: {"R", "remote-time", ARG_BOOL},
! 311: {"s", "silent", ARG_BOOL},
! 312: {"S", "show-error", ARG_BOOL},
! 313: {"t", "telnet-option", ARG_STRING},
! 314: {"T", "upload-file", ARG_FILENAME},
! 315: {"u", "user", ARG_STRING},
! 316: {"U", "proxy-user", ARG_STRING},
! 317: {"v", "verbose", ARG_BOOL},
! 318: {"V", "version", ARG_BOOL},
! 319: {"w", "write-out", ARG_STRING},
! 320: {"x", "proxy", ARG_STRING},
! 321: {"xa", "preproxy", ARG_STRING},
! 322: {"X", "request", ARG_STRING},
! 323: {"Y", "speed-limit", ARG_STRING},
! 324: {"y", "speed-time", ARG_STRING},
! 325: {"z", "time-cond", ARG_STRING},
! 326: {"Z", "parallel", ARG_BOOL},
! 327: {"Zb", "parallel-max", ARG_STRING},
! 328: {"Zc", "parallel-immediate", ARG_BOOL},
! 329: {"#", "progress-bar", ARG_BOOL},
! 330: {"#m", "progress-meter", ARG_BOOL},
! 331: {":", "next", ARG_NONE},
! 332: };
! 333:
! 334: /* Split the argument of -E to 'certname' and 'passphrase' separated by colon.
! 335: * We allow ':' and '\' to be escaped by '\' so that we can use certificate
! 336: * nicknames containing ':'. See <https://sourceforge.net/p/curl/bugs/1196/>
! 337: * for details. */
! 338: #ifndef UNITTESTS
! 339: static
! 340: #endif
! 341: void parse_cert_parameter(const char *cert_parameter,
! 342: char **certname,
! 343: char **passphrase)
! 344: {
! 345: size_t param_length = strlen(cert_parameter);
! 346: size_t span;
! 347: const char *param_place = NULL;
! 348: char *certname_place = NULL;
! 349: *certname = NULL;
! 350: *passphrase = NULL;
! 351:
! 352: /* most trivial assumption: cert_parameter is empty */
! 353: if(param_length == 0)
! 354: return;
! 355:
! 356: /* next less trivial: cert_parameter starts 'pkcs11:' and thus
! 357: * looks like a RFC7512 PKCS#11 URI which can be used as-is.
! 358: * Also if cert_parameter contains no colon nor backslash, this
! 359: * means no passphrase was given and no characters escaped */
! 360: if(curl_strnequal(cert_parameter, "pkcs11:", 7) ||
! 361: !strpbrk(cert_parameter, ":\\")) {
! 362: *certname = strdup(cert_parameter);
! 363: return;
! 364: }
! 365: /* deal with escaped chars; find unescaped colon if it exists */
! 366: certname_place = malloc(param_length + 1);
! 367: if(!certname_place)
! 368: return;
! 369:
! 370: *certname = certname_place;
! 371: param_place = cert_parameter;
! 372: while(*param_place) {
! 373: span = strcspn(param_place, ":\\");
! 374: strncpy(certname_place, param_place, span);
! 375: param_place += span;
! 376: certname_place += span;
! 377: /* we just ate all the non-special chars. now we're on either a special
! 378: * char or the end of the string. */
! 379: switch(*param_place) {
! 380: case '\0':
! 381: break;
! 382: case '\\':
! 383: param_place++;
! 384: switch(*param_place) {
! 385: case '\0':
! 386: *certname_place++ = '\\';
! 387: break;
! 388: case '\\':
! 389: *certname_place++ = '\\';
! 390: param_place++;
! 391: break;
! 392: case ':':
! 393: *certname_place++ = ':';
! 394: param_place++;
! 395: break;
! 396: default:
! 397: *certname_place++ = '\\';
! 398: *certname_place++ = *param_place;
! 399: param_place++;
! 400: break;
! 401: }
! 402: break;
! 403: case ':':
! 404: /* Since we live in a world of weirdness and confusion, the win32
! 405: dudes can use : when using drive letters and thus c:\file:password
! 406: needs to work. In order not to break compatibility, we still use : as
! 407: separator, but we try to detect when it is used for a file name! On
! 408: windows. */
! 409: #ifdef WIN32
! 410: if(param_place &&
! 411: (param_place == &cert_parameter[1]) &&
! 412: (cert_parameter[2] == '\\' || cert_parameter[2] == '/') &&
! 413: (ISALPHA(cert_parameter[0])) ) {
! 414: /* colon in the second column, followed by a backslash, and the
! 415: first character is an alphabetic letter:
! 416:
! 417: this is a drive letter colon */
! 418: *certname_place++ = ':';
! 419: param_place++;
! 420: break;
! 421: }
! 422: #endif
! 423: /* escaped colons and Windows drive letter colons were handled
! 424: * above; if we're still here, this is a separating colon */
! 425: param_place++;
! 426: if(*param_place) {
! 427: *passphrase = strdup(param_place);
! 428: }
! 429: goto done;
! 430: }
! 431: }
! 432: done:
! 433: *certname_place = '\0';
! 434: }
! 435:
! 436: static void
! 437: GetFileAndPassword(char *nextarg, char **file, char **password)
! 438: {
! 439: char *certname, *passphrase;
! 440: parse_cert_parameter(nextarg, &certname, &passphrase);
! 441: Curl_safefree(*file);
! 442: *file = certname;
! 443: if(passphrase) {
! 444: Curl_safefree(*password);
! 445: *password = passphrase;
! 446: }
! 447: cleanarg(nextarg);
! 448: }
! 449:
! 450: /* Get a size parameter for '--limit-rate' or '--max-filesize'.
! 451: * We support a 'G', 'M' or 'K' suffix too.
! 452: */
! 453: static ParameterError GetSizeParameter(struct GlobalConfig *global,
! 454: const char *arg,
! 455: const char *which,
! 456: curl_off_t *value_out)
! 457: {
! 458: char *unit;
! 459: curl_off_t value;
! 460:
! 461: if(curlx_strtoofft(arg, &unit, 0, &value)) {
! 462: warnf(global, "invalid number specified for %s\n", which);
! 463: return PARAM_BAD_USE;
! 464: }
! 465:
! 466: if(!*unit)
! 467: unit = (char *)"b";
! 468: else if(strlen(unit) > 1)
! 469: unit = (char *)"w"; /* unsupported */
! 470:
! 471: switch(*unit) {
! 472: case 'G':
! 473: case 'g':
! 474: if(value > (CURL_OFF_T_MAX / (1024*1024*1024)))
! 475: return PARAM_NUMBER_TOO_LARGE;
! 476: value *= 1024*1024*1024;
! 477: break;
! 478: case 'M':
! 479: case 'm':
! 480: if(value > (CURL_OFF_T_MAX / (1024*1024)))
! 481: return PARAM_NUMBER_TOO_LARGE;
! 482: value *= 1024*1024;
! 483: break;
! 484: case 'K':
! 485: case 'k':
! 486: if(value > (CURL_OFF_T_MAX / 1024))
! 487: return PARAM_NUMBER_TOO_LARGE;
! 488: value *= 1024;
! 489: break;
! 490: case 'b':
! 491: case 'B':
! 492: /* for plain bytes, leave as-is */
! 493: break;
! 494: default:
! 495: warnf(global, "unsupported %s unit. Use G, M, K or B!\n", which);
! 496: return PARAM_BAD_USE;
! 497: }
! 498: *value_out = value;
! 499: return PARAM_OK;
! 500: }
! 501:
! 502: ParameterError getparameter(const char *flag, /* f or -long-flag */
! 503: char *nextarg, /* NULL if unset */
! 504: bool *usedarg, /* set to TRUE if the arg
! 505: has been used */
! 506: struct GlobalConfig *global,
! 507: struct OperationConfig *config)
! 508: {
! 509: char letter;
! 510: char subletter = '\0'; /* subletters can only occur on long options */
! 511: int rc;
! 512: const char *parse = NULL;
! 513: unsigned int j;
! 514: time_t now;
! 515: int hit = -1;
! 516: bool longopt = FALSE;
! 517: bool singleopt = FALSE; /* when true means '-o foo' used '-ofoo' */
! 518: ParameterError err;
! 519: bool toggle = TRUE; /* how to switch boolean options, on or off. Controlled
! 520: by using --OPTION or --no-OPTION */
! 521:
! 522: *usedarg = FALSE; /* default is that we don't use the arg */
! 523:
! 524: if(('-' != flag[0]) || ('-' == flag[1])) {
! 525: /* this should be a long name */
! 526: const char *word = ('-' == flag[0]) ? flag + 2 : flag;
! 527: size_t fnam = strlen(word);
! 528: int numhits = 0;
! 529: bool noflagged = FALSE;
! 530:
! 531: if(!strncmp(word, "no-", 3)) {
! 532: /* disable this option but ignore the "no-" part when looking for it */
! 533: word += 3;
! 534: toggle = FALSE;
! 535: noflagged = TRUE;
! 536: }
! 537:
! 538: for(j = 0; j < sizeof(aliases)/sizeof(aliases[0]); j++) {
! 539: if(curl_strnequal(aliases[j].lname, word, fnam)) {
! 540: longopt = TRUE;
! 541: numhits++;
! 542: if(curl_strequal(aliases[j].lname, word)) {
! 543: parse = aliases[j].letter;
! 544: hit = j;
! 545: numhits = 1; /* a single unique hit */
! 546: break;
! 547: }
! 548: parse = aliases[j].letter;
! 549: hit = j;
! 550: }
! 551: }
! 552: if(numhits > 1) {
! 553: /* this is at least the second match! */
! 554: return PARAM_OPTION_AMBIGUOUS;
! 555: }
! 556: if(hit < 0) {
! 557: return PARAM_OPTION_UNKNOWN;
! 558: }
! 559: if(noflagged && (aliases[hit].desc != ARG_BOOL))
! 560: /* --no- prefixed an option that isn't boolean! */
! 561: return PARAM_NO_NOT_BOOLEAN;
! 562: }
! 563: else {
! 564: flag++; /* prefixed with one dash, pass it */
! 565: hit = -1;
! 566: parse = flag;
! 567: }
! 568:
! 569: do {
! 570: /* we can loop here if we have multiple single-letters */
! 571:
! 572: if(!longopt) {
! 573: letter = (char)*parse;
! 574: subletter = '\0';
! 575: }
! 576: else {
! 577: letter = parse[0];
! 578: subletter = parse[1];
! 579: }
! 580:
! 581: if(hit < 0) {
! 582: for(j = 0; j < sizeof(aliases)/sizeof(aliases[0]); j++) {
! 583: if(letter == aliases[j].letter[0]) {
! 584: hit = j;
! 585: break;
! 586: }
! 587: }
! 588: if(hit < 0) {
! 589: return PARAM_OPTION_UNKNOWN;
! 590: }
! 591: }
! 592:
! 593: if(aliases[hit].desc >= ARG_STRING) {
! 594: /* this option requires an extra parameter */
! 595: if(!longopt && parse[1]) {
! 596: nextarg = (char *)&parse[1]; /* this is the actual extra parameter */
! 597: singleopt = TRUE; /* don't loop anymore after this */
! 598: }
! 599: else if(!nextarg)
! 600: return PARAM_REQUIRES_PARAMETER;
! 601: else
! 602: *usedarg = TRUE; /* mark it as used */
! 603:
! 604: if((aliases[hit].desc == ARG_FILENAME) &&
! 605: (nextarg[0] == '-') && nextarg[1]) {
! 606: /* if the file name looks like a command line option */
! 607: warnf(global, "The file name argument '%s' looks like a flag.\n",
! 608: nextarg);
! 609: }
! 610: }
! 611: else if((aliases[hit].desc == ARG_NONE) && !toggle)
! 612: return PARAM_NO_PREFIX;
! 613:
! 614: switch(letter) {
! 615: case '*': /* options without a short option */
! 616: switch(subletter) {
! 617: case '4': /* --dns-ipv4-addr */
! 618: /* addr in dot notation */
! 619: GetStr(&config->dns_ipv4_addr, nextarg);
! 620: break;
! 621: case '6': /* --dns-ipv6-addr */
! 622: /* addr in dot notation */
! 623: GetStr(&config->dns_ipv6_addr, nextarg);
! 624: break;
! 625: case 'a': /* random-file */
! 626: GetStr(&config->random_file, nextarg);
! 627: break;
! 628: case 'b': /* egd-file */
! 629: GetStr(&config->egd_file, nextarg);
! 630: break;
! 631: case 'B': /* OAuth 2.0 bearer token */
! 632: GetStr(&config->oauth_bearer, nextarg);
! 633: config->authtype |= CURLAUTH_BEARER;
! 634: break;
! 635: case 'c': /* connect-timeout */
! 636: err = str2udouble(&config->connecttimeout, nextarg,
! 637: LONG_MAX/1000);
! 638: if(err)
! 639: return err;
! 640: break;
! 641: case 'C': /* doh-url */
! 642: GetStr(&config->doh_url, nextarg);
! 643: break;
! 644: case 'd': /* ciphers */
! 645: GetStr(&config->cipher_list, nextarg);
! 646: break;
! 647: case 'D': /* --dns-interface */
! 648: /* interface name */
! 649: GetStr(&config->dns_interface, nextarg);
! 650: break;
! 651: case 'e': /* --disable-epsv */
! 652: config->disable_epsv = toggle;
! 653: break;
! 654: case 'f': /* --disallow-username-in-url */
! 655: config->disallow_username_in_url = toggle;
! 656: break;
! 657: case 'E': /* --epsv */
! 658: config->disable_epsv = (!toggle)?TRUE:FALSE;
! 659: break;
! 660: case 'F': /* --dns-servers */
! 661: /* IP addrs of DNS servers */
! 662: GetStr(&config->dns_servers, nextarg);
! 663: break;
! 664: case 'g': /* --trace */
! 665: GetStr(&global->trace_dump, nextarg);
! 666: if(global->tracetype && (global->tracetype != TRACE_BIN))
! 667: warnf(global, "--trace overrides an earlier trace/verbose option\n");
! 668: global->tracetype = TRACE_BIN;
! 669: break;
! 670: case 'G': /* --npn */
! 671: config->nonpn = (!toggle)?TRUE:FALSE;
! 672: break;
! 673: case 'h': /* --trace-ascii */
! 674: GetStr(&global->trace_dump, nextarg);
! 675: if(global->tracetype && (global->tracetype != TRACE_ASCII))
! 676: warnf(global,
! 677: "--trace-ascii overrides an earlier trace/verbose option\n");
! 678: global->tracetype = TRACE_ASCII;
! 679: break;
! 680: case 'H': /* --alpn */
! 681: config->noalpn = (!toggle)?TRUE:FALSE;
! 682: break;
! 683: case 'i': /* --limit-rate */
! 684: {
! 685: curl_off_t value;
! 686: ParameterError pe = GetSizeParameter(global, nextarg, "rate", &value);
! 687:
! 688: if(pe != PARAM_OK)
! 689: return pe;
! 690: config->recvpersecond = value;
! 691: config->sendpersecond = value;
! 692: }
! 693: break;
! 694:
! 695: case 'j': /* --compressed */
! 696: if(toggle &&
! 697: !(curlinfo->features & (CURL_VERSION_LIBZ | CURL_VERSION_BROTLI)))
! 698: return PARAM_LIBCURL_DOESNT_SUPPORT;
! 699: config->encoding = toggle;
! 700: break;
! 701:
! 702: case 'J': /* --tr-encoding */
! 703: config->tr_encoding = toggle;
! 704: break;
! 705:
! 706: case 'k': /* --digest */
! 707: if(toggle)
! 708: config->authtype |= CURLAUTH_DIGEST;
! 709: else
! 710: config->authtype &= ~CURLAUTH_DIGEST;
! 711: break;
! 712:
! 713: case 'l': /* --negotiate */
! 714: if(toggle) {
! 715: if(curlinfo->features & CURL_VERSION_SPNEGO)
! 716: config->authtype |= CURLAUTH_NEGOTIATE;
! 717: else
! 718: return PARAM_LIBCURL_DOESNT_SUPPORT;
! 719: }
! 720: else
! 721: config->authtype &= ~CURLAUTH_NEGOTIATE;
! 722: break;
! 723:
! 724: case 'm': /* --ntlm */
! 725: if(toggle) {
! 726: if(curlinfo->features & CURL_VERSION_NTLM)
! 727: config->authtype |= CURLAUTH_NTLM;
! 728: else
! 729: return PARAM_LIBCURL_DOESNT_SUPPORT;
! 730: }
! 731: else
! 732: config->authtype &= ~CURLAUTH_NTLM;
! 733: break;
! 734:
! 735: case 'M': /* --ntlm-wb */
! 736: if(toggle) {
! 737: if(curlinfo->features & CURL_VERSION_NTLM_WB)
! 738: config->authtype |= CURLAUTH_NTLM_WB;
! 739: else
! 740: return PARAM_LIBCURL_DOESNT_SUPPORT;
! 741: }
! 742: else
! 743: config->authtype &= ~CURLAUTH_NTLM_WB;
! 744: break;
! 745:
! 746: case 'n': /* --basic for completeness */
! 747: if(toggle)
! 748: config->authtype |= CURLAUTH_BASIC;
! 749: else
! 750: config->authtype &= ~CURLAUTH_BASIC;
! 751: break;
! 752:
! 753: case 'o': /* --anyauth, let libcurl pick it */
! 754: if(toggle)
! 755: config->authtype = CURLAUTH_ANY;
! 756: /* --no-anyauth simply doesn't touch it */
! 757: break;
! 758:
! 759: #ifdef USE_WATT32
! 760: case 'p': /* --wdebug */
! 761: dbug_init();
! 762: break;
! 763: #endif
! 764: case 'q': /* --ftp-create-dirs */
! 765: config->ftp_create_dirs = toggle;
! 766: break;
! 767:
! 768: case 'r': /* --create-dirs */
! 769: config->create_dirs = toggle;
! 770: break;
! 771:
! 772: case 's': /* --max-redirs */
! 773: /* specified max no of redirects (http(s)), this accepts -1 as a
! 774: special condition */
! 775: err = str2num(&config->maxredirs, nextarg);
! 776: if(err)
! 777: return err;
! 778: if(config->maxredirs < -1)
! 779: return PARAM_BAD_NUMERIC;
! 780: break;
! 781:
! 782: case 't': /* --proxy-ntlm */
! 783: if(curlinfo->features & CURL_VERSION_NTLM)
! 784: config->proxyntlm = toggle;
! 785: else
! 786: return PARAM_LIBCURL_DOESNT_SUPPORT;
! 787: break;
! 788:
! 789: case 'u': /* --crlf */
! 790: /* LF -> CRLF conversion? */
! 791: config->crlf = toggle;
! 792: break;
! 793:
! 794: case 'v': /* --stderr */
! 795: if(strcmp(nextarg, "-")) {
! 796: FILE *newfile = fopen(nextarg, FOPEN_WRITETEXT);
! 797: if(!newfile)
! 798: warnf(global, "Failed to open %s!\n", nextarg);
! 799: else {
! 800: if(global->errors_fopened)
! 801: fclose(global->errors);
! 802: global->errors = newfile;
! 803: global->errors_fopened = TRUE;
! 804: }
! 805: }
! 806: else
! 807: global->errors = stdout;
! 808: break;
! 809: case 'w': /* --interface */
! 810: /* interface */
! 811: GetStr(&config->iface, nextarg);
! 812: break;
! 813: case 'x': /* --krb */
! 814: /* kerberos level string */
! 815: if(curlinfo->features & CURL_VERSION_KERBEROS4)
! 816: GetStr(&config->krblevel, nextarg);
! 817: else
! 818: return PARAM_LIBCURL_DOESNT_SUPPORT;
! 819: break;
! 820: case 'X': /* --haproxy-protocol */
! 821: config->haproxy_protocol = toggle;
! 822: break;
! 823: case 'y': /* --max-filesize */
! 824: {
! 825: curl_off_t value;
! 826: ParameterError pe =
! 827: GetSizeParameter(global, nextarg, "max-filesize", &value);
! 828:
! 829: if(pe != PARAM_OK)
! 830: return pe;
! 831: config->max_filesize = value;
! 832: }
! 833: break;
! 834: case 'z': /* --disable-eprt */
! 835: config->disable_eprt = toggle;
! 836: break;
! 837: case 'Z': /* --eprt */
! 838: config->disable_eprt = (!toggle)?TRUE:FALSE;
! 839: break;
! 840: case '~': /* --xattr */
! 841: config->xattr = toggle;
! 842: break;
! 843: case '@': /* the URL! */
! 844: {
! 845: struct getout *url;
! 846:
! 847: if(!config->url_get)
! 848: config->url_get = config->url_list;
! 849:
! 850: if(config->url_get) {
! 851: /* there's a node here, if it already is filled-in continue to find
! 852: an "empty" node */
! 853: while(config->url_get && (config->url_get->flags & GETOUT_URL))
! 854: config->url_get = config->url_get->next;
! 855: }
! 856:
! 857: /* now there might or might not be an available node to fill in! */
! 858:
! 859: if(config->url_get)
! 860: /* existing node */
! 861: url = config->url_get;
! 862: else
! 863: /* there was no free node, create one! */
! 864: config->url_get = url = new_getout(config);
! 865:
! 866: if(!url)
! 867: return PARAM_NO_MEM;
! 868:
! 869: /* fill in the URL */
! 870: GetStr(&url->url, nextarg);
! 871: url->flags |= GETOUT_URL;
! 872: }
! 873: }
! 874: break;
! 875: case '$': /* more options without a short option */
! 876: switch(subletter) {
! 877: case 'a': /* --ssl */
! 878: if(toggle && !(curlinfo->features & CURL_VERSION_SSL))
! 879: return PARAM_LIBCURL_DOESNT_SUPPORT;
! 880: config->ftp_ssl = toggle;
! 881: break;
! 882: case 'b': /* --ftp-pasv */
! 883: Curl_safefree(config->ftpport);
! 884: break;
! 885: case 'c': /* --socks5 specifies a socks5 proxy to use, and resolves
! 886: the name locally and passes on the resolved address */
! 887: GetStr(&config->proxy, nextarg);
! 888: config->proxyver = CURLPROXY_SOCKS5;
! 889: break;
! 890: case 't': /* --socks4 specifies a socks4 proxy to use */
! 891: GetStr(&config->proxy, nextarg);
! 892: config->proxyver = CURLPROXY_SOCKS4;
! 893: break;
! 894: case 'T': /* --socks4a specifies a socks4a proxy to use */
! 895: GetStr(&config->proxy, nextarg);
! 896: config->proxyver = CURLPROXY_SOCKS4A;
! 897: break;
! 898: case '2': /* --socks5-hostname specifies a socks5 proxy and enables name
! 899: resolving with the proxy */
! 900: GetStr(&config->proxy, nextarg);
! 901: config->proxyver = CURLPROXY_SOCKS5_HOSTNAME;
! 902: break;
! 903: case 'd': /* --tcp-nodelay option */
! 904: config->tcp_nodelay = toggle;
! 905: break;
! 906: case 'e': /* --proxy-digest */
! 907: config->proxydigest = toggle;
! 908: break;
! 909: case 'f': /* --proxy-basic */
! 910: config->proxybasic = toggle;
! 911: break;
! 912: case 'g': /* --retry */
! 913: err = str2unum(&config->req_retry, nextarg);
! 914: if(err)
! 915: return err;
! 916: break;
! 917: case 'V': /* --retry-connrefused */
! 918: config->retry_connrefused = toggle;
! 919: break;
! 920: case 'h': /* --retry-delay */
! 921: err = str2unummax(&config->retry_delay, nextarg, LONG_MAX/1000);
! 922: if(err)
! 923: return err;
! 924: break;
! 925: case 'i': /* --retry-max-time */
! 926: err = str2unummax(&config->retry_maxtime, nextarg, LONG_MAX/1000);
! 927: if(err)
! 928: return err;
! 929: break;
! 930:
! 931: case 'k': /* --proxy-negotiate */
! 932: if(curlinfo->features & CURL_VERSION_SPNEGO)
! 933: config->proxynegotiate = toggle;
! 934: else
! 935: return PARAM_LIBCURL_DOESNT_SUPPORT;
! 936: break;
! 937:
! 938: case 'm': /* --ftp-account */
! 939: GetStr(&config->ftp_account, nextarg);
! 940: break;
! 941: case 'n': /* --proxy-anyauth */
! 942: config->proxyanyauth = toggle;
! 943: break;
! 944: case 'o': /* --trace-time */
! 945: global->tracetime = toggle;
! 946: break;
! 947: case 'p': /* --ignore-content-length */
! 948: config->ignorecl = toggle;
! 949: break;
! 950: case 'q': /* --ftp-skip-pasv-ip */
! 951: config->ftp_skip_ip = toggle;
! 952: break;
! 953: case 'r': /* --ftp-method (undocumented at this point) */
! 954: config->ftp_filemethod = ftpfilemethod(config, nextarg);
! 955: break;
! 956: case 's': { /* --local-port */
! 957: char lrange[7]; /* 16bit base 10 is 5 digits, but we allow 6 so that
! 958: this catches overflows, not just truncates */
! 959: char *p = nextarg;
! 960: while(ISDIGIT(*p))
! 961: p++;
! 962: if(*p) {
! 963: /* if there's anything more than a plain decimal number */
! 964: rc = sscanf(p, " - %6s", lrange);
! 965: *p = 0; /* zero terminate to make str2unum() work below */
! 966: }
! 967: else
! 968: rc = 0;
! 969:
! 970: err = str2unum(&config->localport, nextarg);
! 971: if(err || (config->localport > 65535))
! 972: return PARAM_BAD_USE;
! 973: if(!rc)
! 974: config->localportrange = 1; /* default number of ports to try */
! 975: else {
! 976: err = str2unum(&config->localportrange, lrange);
! 977: if(err || (config->localportrange > 65535))
! 978: return PARAM_BAD_USE;
! 979: config->localportrange -= (config->localport-1);
! 980: if(config->localportrange < 1)
! 981: return PARAM_BAD_USE;
! 982: }
! 983: break;
! 984: }
! 985: case 'u': /* --ftp-alternative-to-user */
! 986: GetStr(&config->ftp_alternative_to_user, nextarg);
! 987: break;
! 988: case 'v': /* --ssl-reqd */
! 989: if(toggle && !(curlinfo->features & CURL_VERSION_SSL))
! 990: return PARAM_LIBCURL_DOESNT_SUPPORT;
! 991: config->ftp_ssl_reqd = toggle;
! 992: break;
! 993: case 'w': /* --no-sessionid */
! 994: config->disable_sessionid = (!toggle)?TRUE:FALSE;
! 995: break;
! 996: case 'x': /* --ftp-ssl-control */
! 997: if(toggle && !(curlinfo->features & CURL_VERSION_SSL))
! 998: return PARAM_LIBCURL_DOESNT_SUPPORT;
! 999: config->ftp_ssl_control = toggle;
! 1000: break;
! 1001: case 'y': /* --ftp-ssl-ccc */
! 1002: config->ftp_ssl_ccc = toggle;
! 1003: if(!config->ftp_ssl_ccc_mode)
! 1004: config->ftp_ssl_ccc_mode = CURLFTPSSL_CCC_PASSIVE;
! 1005: break;
! 1006: case 'j': /* --ftp-ssl-ccc-mode */
! 1007: config->ftp_ssl_ccc = TRUE;
! 1008: config->ftp_ssl_ccc_mode = ftpcccmethod(config, nextarg);
! 1009: break;
! 1010: case 'z': /* --libcurl */
! 1011: #ifdef CURL_DISABLE_LIBCURL_OPTION
! 1012: warnf(global,
! 1013: "--libcurl option was disabled at build-time!\n");
! 1014: return PARAM_OPTION_UNKNOWN;
! 1015: #else
! 1016: GetStr(&global->libcurl, nextarg);
! 1017: break;
! 1018: #endif
! 1019: case '#': /* --raw */
! 1020: config->raw = toggle;
! 1021: break;
! 1022: case '0': /* --post301 */
! 1023: config->post301 = toggle;
! 1024: break;
! 1025: case '1': /* --no-keepalive */
! 1026: config->nokeepalive = (!toggle)?TRUE:FALSE;
! 1027: break;
! 1028: case '3': /* --keepalive-time */
! 1029: err = str2unum(&config->alivetime, nextarg);
! 1030: if(err)
! 1031: return err;
! 1032: break;
! 1033: case '4': /* --post302 */
! 1034: config->post302 = toggle;
! 1035: break;
! 1036: case 'I': /* --post303 */
! 1037: config->post303 = toggle;
! 1038: break;
! 1039: case '5': /* --noproxy */
! 1040: /* This specifies the noproxy list */
! 1041: GetStr(&config->noproxy, nextarg);
! 1042: break;
! 1043: case '7': /* --socks5-gssapi-nec*/
! 1044: config->socks5_gssapi_nec = toggle;
! 1045: break;
! 1046: case '8': /* --proxy1.0 */
! 1047: /* http 1.0 proxy */
! 1048: GetStr(&config->proxy, nextarg);
! 1049: config->proxyver = CURLPROXY_HTTP_1_0;
! 1050: break;
! 1051: case '9': /* --tftp-blksize */
! 1052: err = str2unum(&config->tftp_blksize, nextarg);
! 1053: if(err)
! 1054: return err;
! 1055: break;
! 1056: case 'A': /* --mail-from */
! 1057: GetStr(&config->mail_from, nextarg);
! 1058: break;
! 1059: case 'B': /* --mail-rcpt */
! 1060: /* append receiver to a list */
! 1061: err = add2list(&config->mail_rcpt, nextarg);
! 1062: if(err)
! 1063: return err;
! 1064: break;
! 1065: case 'C': /* --ftp-pret */
! 1066: config->ftp_pret = toggle;
! 1067: break;
! 1068: case 'D': /* --proto */
! 1069: config->proto_present = TRUE;
! 1070: if(proto2num(config, &config->proto, nextarg))
! 1071: return PARAM_BAD_USE;
! 1072: break;
! 1073: case 'E': /* --proto-redir */
! 1074: config->proto_redir_present = TRUE;
! 1075: if(proto2num(config, &config->proto_redir, nextarg))
! 1076: return PARAM_BAD_USE;
! 1077: break;
! 1078: case 'F': /* --resolve */
! 1079: err = add2list(&config->resolve, nextarg);
! 1080: if(err)
! 1081: return err;
! 1082: break;
! 1083: case 'G': /* --delegation LEVEL */
! 1084: config->gssapi_delegation = delegation(config, nextarg);
! 1085: break;
! 1086: case 'H': /* --mail-auth */
! 1087: GetStr(&config->mail_auth, nextarg);
! 1088: break;
! 1089: case 'J': /* --metalink */
! 1090: {
! 1091: #ifdef USE_METALINK
! 1092: int mlmaj, mlmin, mlpatch;
! 1093: metalink_get_version(&mlmaj, &mlmin, &mlpatch);
! 1094: if((mlmaj*10000)+(mlmin*100) + mlpatch < CURL_REQ_LIBMETALINK_VERS) {
! 1095: warnf(global,
! 1096: "--metalink option cannot be used because the version of "
! 1097: "the linked libmetalink library is too old. "
! 1098: "Required: %d.%d.%d, found %d.%d.%d\n",
! 1099: CURL_REQ_LIBMETALINK_MAJOR,
! 1100: CURL_REQ_LIBMETALINK_MINOR,
! 1101: CURL_REQ_LIBMETALINK_PATCH,
! 1102: mlmaj, mlmin, mlpatch);
! 1103: return PARAM_BAD_USE;
! 1104: }
! 1105: else
! 1106: config->use_metalink = toggle;
! 1107: #else
! 1108: warnf(global, "--metalink option is ignored because the binary is "
! 1109: "built without the Metalink support.\n");
! 1110: #endif
! 1111: break;
! 1112: }
! 1113: case '6': /* --sasl-authzid */
! 1114: GetStr(&config->sasl_authzid, nextarg);
! 1115: break;
! 1116: case 'K': /* --sasl-ir */
! 1117: config->sasl_ir = toggle;
! 1118: break;
! 1119: case 'L': /* --test-event */
! 1120: #ifdef CURLDEBUG
! 1121: global->test_event_based = toggle;
! 1122: #else
! 1123: warnf(global, "--test-event is ignored unless a debug build!\n");
! 1124: #endif
! 1125: break;
! 1126: case 'M': /* --unix-socket */
! 1127: config->abstract_unix_socket = FALSE;
! 1128: GetStr(&config->unix_socket_path, nextarg);
! 1129: break;
! 1130: case 'N': /* --path-as-is */
! 1131: config->path_as_is = toggle;
! 1132: break;
! 1133: case 'O': /* --proxy-service-name */
! 1134: GetStr(&config->proxy_service_name, nextarg);
! 1135: break;
! 1136: case 'P': /* --service-name */
! 1137: GetStr(&config->service_name, nextarg);
! 1138: break;
! 1139: case 'Q': /* --proto-default */
! 1140: GetStr(&config->proto_default, nextarg);
! 1141: err = check_protocol(config->proto_default);
! 1142: if(err)
! 1143: return err;
! 1144: break;
! 1145: case 'R': /* --expect100-timeout */
! 1146: err = str2udouble(&config->expect100timeout, nextarg, LONG_MAX/1000);
! 1147: if(err)
! 1148: return err;
! 1149: break;
! 1150: case 'S': /* --tftp-no-options */
! 1151: config->tftp_no_options = toggle;
! 1152: break;
! 1153: case 'U': /* --connect-to */
! 1154: err = add2list(&config->connect_to, nextarg);
! 1155: if(err)
! 1156: return err;
! 1157: break;
! 1158: case 'W': /* --abstract-unix-socket */
! 1159: config->abstract_unix_socket = TRUE;
! 1160: GetStr(&config->unix_socket_path, nextarg);
! 1161: break;
! 1162: case 'X': /* --tls-max */
! 1163: err = str2tls_max(&config->ssl_version_max, nextarg);
! 1164: if(err)
! 1165: return err;
! 1166: break;
! 1167: case 'Y': /* --suppress-connect-headers */
! 1168: config->suppress_connect_headers = toggle;
! 1169: break;
! 1170: case 'Z': /* --compressed-ssh */
! 1171: config->ssh_compression = toggle;
! 1172: break;
! 1173: case '~': /* --happy-eyeballs-timeout-ms */
! 1174: err = str2unum(&config->happy_eyeballs_timeout_ms, nextarg);
! 1175: if(err)
! 1176: return err;
! 1177: /* 0 is a valid value for this timeout */
! 1178: break;
! 1179: }
! 1180: break;
! 1181: case '#':
! 1182: switch(subletter) {
! 1183: case 'm': /* --progress-meter */
! 1184: global->noprogress = !toggle;
! 1185: break;
! 1186: default: /* --progress-bar */
! 1187: global->progressmode =
! 1188: toggle ? CURL_PROGRESS_BAR : CURL_PROGRESS_STATS;
! 1189: break;
! 1190: }
! 1191: break;
! 1192: case ':': /* --next */
! 1193: return PARAM_NEXT_OPERATION;
! 1194: case '0': /* --http* options */
! 1195: switch(subletter) {
! 1196: case '\0':
! 1197: /* HTTP version 1.0 */
! 1198: config->httpversion = CURL_HTTP_VERSION_1_0;
! 1199: break;
! 1200: case '1':
! 1201: /* HTTP version 1.1 */
! 1202: config->httpversion = CURL_HTTP_VERSION_1_1;
! 1203: break;
! 1204: case '2':
! 1205: /* HTTP version 2.0 */
! 1206: config->httpversion = CURL_HTTP_VERSION_2_0;
! 1207: break;
! 1208: case '3': /* --http2-prior-knowledge */
! 1209: /* HTTP version 2.0 over clean TCP*/
! 1210: config->httpversion = CURL_HTTP_VERSION_2_PRIOR_KNOWLEDGE;
! 1211: break;
! 1212: case '4': /* --http3 */
! 1213: /* HTTP version 3 go over QUIC - at once */
! 1214: if(curlinfo->features & CURL_VERSION_HTTP3)
! 1215: config->httpversion = CURL_HTTP_VERSION_3;
! 1216: else
! 1217: return PARAM_LIBCURL_DOESNT_SUPPORT;
! 1218: break;
! 1219: case '9':
! 1220: /* Allow HTTP/0.9 responses! */
! 1221: config->http09_allowed = toggle;
! 1222: break;
! 1223: }
! 1224: break;
! 1225: case '1': /* --tlsv1* options */
! 1226: switch(subletter) {
! 1227: case '\0':
! 1228: /* TLS version 1.x */
! 1229: config->ssl_version = CURL_SSLVERSION_TLSv1;
! 1230: break;
! 1231: case '0':
! 1232: /* TLS version 1.0 */
! 1233: config->ssl_version = CURL_SSLVERSION_TLSv1_0;
! 1234: break;
! 1235: case '1':
! 1236: /* TLS version 1.1 */
! 1237: config->ssl_version = CURL_SSLVERSION_TLSv1_1;
! 1238: break;
! 1239: case '2':
! 1240: /* TLS version 1.2 */
! 1241: config->ssl_version = CURL_SSLVERSION_TLSv1_2;
! 1242: break;
! 1243: case '3':
! 1244: /* TLS version 1.3 */
! 1245: config->ssl_version = CURL_SSLVERSION_TLSv1_3;
! 1246: break;
! 1247: case 'A': /* --tls13-ciphers */
! 1248: GetStr(&config->cipher13_list, nextarg);
! 1249: break;
! 1250: case 'B': /* --proxy-tls13-ciphers */
! 1251: GetStr(&config->proxy_cipher13_list, nextarg);
! 1252: break;
! 1253: }
! 1254: break;
! 1255: case '2':
! 1256: /* SSL version 2 */
! 1257: config->ssl_version = CURL_SSLVERSION_SSLv2;
! 1258: break;
! 1259: case '3':
! 1260: /* SSL version 3 */
! 1261: config->ssl_version = CURL_SSLVERSION_SSLv3;
! 1262: break;
! 1263: case '4':
! 1264: /* IPv4 */
! 1265: config->ip_version = 4;
! 1266: break;
! 1267: case '6':
! 1268: /* IPv6 */
! 1269: config->ip_version = 6;
! 1270: break;
! 1271: case 'a':
! 1272: /* This makes the FTP sessions use APPE instead of STOR */
! 1273: config->ftp_append = toggle;
! 1274: break;
! 1275: case 'A':
! 1276: /* This specifies the User-Agent name */
! 1277: GetStr(&config->useragent, nextarg);
! 1278: break;
! 1279: case 'b':
! 1280: switch(subletter) {
! 1281: case 'a': /* --alt-svc */
! 1282: if(curlinfo->features & CURL_VERSION_ALTSVC)
! 1283: GetStr(&config->altsvc, nextarg);
! 1284: else
! 1285: return PARAM_LIBCURL_DOESNT_SUPPORT;
! 1286: break;
! 1287: default: /* --cookie string coming up: */
! 1288: if(nextarg[0] == '@') {
! 1289: nextarg++;
! 1290: }
! 1291: else if(strchr(nextarg, '=')) {
! 1292: /* A cookie string must have a =-letter */
! 1293: GetStr(&config->cookie, nextarg);
! 1294: break;
! 1295: }
! 1296: /* We have a cookie file to read from! */
! 1297: GetStr(&config->cookiefile, nextarg);
! 1298: }
! 1299: break;
! 1300: case 'B':
! 1301: /* use ASCII/text when transferring */
! 1302: config->use_ascii = toggle;
! 1303: break;
! 1304: case 'c':
! 1305: /* get the file name to dump all cookies in */
! 1306: GetStr(&config->cookiejar, nextarg);
! 1307: break;
! 1308: case 'C':
! 1309: /* This makes us continue an ftp transfer at given position */
! 1310: if(strcmp(nextarg, "-")) {
! 1311: err = str2offset(&config->resume_from, nextarg);
! 1312: if(err)
! 1313: return err;
! 1314: config->resume_from_current = FALSE;
! 1315: }
! 1316: else {
! 1317: config->resume_from_current = TRUE;
! 1318: config->resume_from = 0;
! 1319: }
! 1320: config->use_resume = TRUE;
! 1321: break;
! 1322: case 'd':
! 1323: /* postfield data */
! 1324: {
! 1325: char *postdata = NULL;
! 1326: FILE *file;
! 1327: size_t size = 0;
! 1328: bool raw_mode = (subletter == 'r');
! 1329:
! 1330: if(subletter == 'e') { /* --data-urlencode*/
! 1331: /* [name]=[content], we encode the content part only
! 1332: * [name]@[file name]
! 1333: *
! 1334: * Case 2: we first load the file using that name and then encode
! 1335: * the content.
! 1336: */
! 1337: const char *p = strchr(nextarg, '=');
! 1338: size_t nlen;
! 1339: char is_file;
! 1340: if(!p)
! 1341: /* there was no '=' letter, check for a '@' instead */
! 1342: p = strchr(nextarg, '@');
! 1343: if(p) {
! 1344: nlen = p - nextarg; /* length of the name part */
! 1345: is_file = *p++; /* pass the separator */
! 1346: }
! 1347: else {
! 1348: /* neither @ nor =, so no name and it isn't a file */
! 1349: nlen = is_file = 0;
! 1350: p = nextarg;
! 1351: }
! 1352: if('@' == is_file) {
! 1353: /* a '@' letter, it means that a file name or - (stdin) follows */
! 1354: if(!strcmp("-", p)) {
! 1355: file = stdin;
! 1356: set_binmode(stdin);
! 1357: }
! 1358: else {
! 1359: file = fopen(p, "rb");
! 1360: if(!file)
! 1361: warnf(global,
! 1362: "Couldn't read data from file \"%s\", this makes "
! 1363: "an empty POST.\n", nextarg);
! 1364: }
! 1365:
! 1366: err = file2memory(&postdata, &size, file);
! 1367:
! 1368: if(file && (file != stdin))
! 1369: fclose(file);
! 1370: if(err)
! 1371: return err;
! 1372: }
! 1373: else {
! 1374: GetStr(&postdata, p);
! 1375: if(postdata)
! 1376: size = strlen(postdata);
! 1377: }
! 1378:
! 1379: if(!postdata) {
! 1380: /* no data from the file, point to a zero byte string to make this
! 1381: get sent as a POST anyway */
! 1382: postdata = strdup("");
! 1383: if(!postdata)
! 1384: return PARAM_NO_MEM;
! 1385: size = 0;
! 1386: }
! 1387: else {
! 1388: char *enc = curl_easy_escape(NULL, postdata, (int)size);
! 1389: Curl_safefree(postdata); /* no matter if it worked or not */
! 1390: if(enc) {
! 1391: /* now make a string with the name from above and append the
! 1392: encoded string */
! 1393: size_t outlen = nlen + strlen(enc) + 2;
! 1394: char *n = malloc(outlen);
! 1395: if(!n) {
! 1396: curl_free(enc);
! 1397: return PARAM_NO_MEM;
! 1398: }
! 1399: if(nlen > 0) { /* only append '=' if we have a name */
! 1400: msnprintf(n, outlen, "%.*s=%s", nlen, nextarg, enc);
! 1401: size = outlen-1;
! 1402: }
! 1403: else {
! 1404: strcpy(n, enc);
! 1405: size = outlen-2; /* since no '=' was inserted */
! 1406: }
! 1407: curl_free(enc);
! 1408: postdata = n;
! 1409: }
! 1410: else
! 1411: return PARAM_NO_MEM;
! 1412: }
! 1413: }
! 1414: else if('@' == *nextarg && !raw_mode) {
! 1415: /* the data begins with a '@' letter, it means that a file name
! 1416: or - (stdin) follows */
! 1417: nextarg++; /* pass the @ */
! 1418:
! 1419: if(!strcmp("-", nextarg)) {
! 1420: file = stdin;
! 1421: if(subletter == 'b') /* forced data-binary */
! 1422: set_binmode(stdin);
! 1423: }
! 1424: else {
! 1425: file = fopen(nextarg, "rb");
! 1426: if(!file)
! 1427: warnf(global, "Couldn't read data from file \"%s\", this makes "
! 1428: "an empty POST.\n", nextarg);
! 1429: }
! 1430:
! 1431: if(subletter == 'b')
! 1432: /* forced binary */
! 1433: err = file2memory(&postdata, &size, file);
! 1434: else {
! 1435: err = file2string(&postdata, file);
! 1436: if(postdata)
! 1437: size = strlen(postdata);
! 1438: }
! 1439:
! 1440: if(file && (file != stdin))
! 1441: fclose(file);
! 1442: if(err)
! 1443: return err;
! 1444:
! 1445: if(!postdata) {
! 1446: /* no data from the file, point to a zero byte string to make this
! 1447: get sent as a POST anyway */
! 1448: postdata = strdup("");
! 1449: if(!postdata)
! 1450: return PARAM_NO_MEM;
! 1451: }
! 1452: }
! 1453: else {
! 1454: GetStr(&postdata, nextarg);
! 1455: if(postdata)
! 1456: size = strlen(postdata);
! 1457: }
! 1458:
! 1459: #ifdef CURL_DOES_CONVERSIONS
! 1460: if(subletter != 'b') {
! 1461: /* NOT forced binary, convert to ASCII */
! 1462: if(convert_to_network(postdata, strlen(postdata))) {
! 1463: Curl_safefree(postdata);
! 1464: return PARAM_NO_MEM;
! 1465: }
! 1466: }
! 1467: #endif
! 1468:
! 1469: if(config->postfields) {
! 1470: /* we already have a string, we append this one with a separating
! 1471: &-letter */
! 1472: char *oldpost = config->postfields;
! 1473: curl_off_t oldlen = config->postfieldsize;
! 1474: curl_off_t newlen = oldlen + curlx_uztoso(size) + 2;
! 1475: config->postfields = malloc((size_t)newlen);
! 1476: if(!config->postfields) {
! 1477: Curl_safefree(oldpost);
! 1478: Curl_safefree(postdata);
! 1479: return PARAM_NO_MEM;
! 1480: }
! 1481: memcpy(config->postfields, oldpost, (size_t)oldlen);
! 1482: /* use byte value 0x26 for '&' to accommodate non-ASCII platforms */
! 1483: config->postfields[oldlen] = '\x26';
! 1484: memcpy(&config->postfields[oldlen + 1], postdata, size);
! 1485: config->postfields[oldlen + 1 + size] = '\0';
! 1486: Curl_safefree(oldpost);
! 1487: Curl_safefree(postdata);
! 1488: config->postfieldsize += size + 1;
! 1489: }
! 1490: else {
! 1491: config->postfields = postdata;
! 1492: config->postfieldsize = curlx_uztoso(size);
! 1493: }
! 1494: }
! 1495: /*
! 1496: We can't set the request type here, as this data might be used in
! 1497: a simple GET if -G is used. Already or soon.
! 1498:
! 1499: if(SetHTTPrequest(HTTPREQ_SIMPLEPOST, &config->httpreq)) {
! 1500: Curl_safefree(postdata);
! 1501: return PARAM_BAD_USE;
! 1502: }
! 1503: */
! 1504: break;
! 1505: case 'D':
! 1506: /* dump-header to given file name */
! 1507: GetStr(&config->headerfile, nextarg);
! 1508: break;
! 1509: case 'e':
! 1510: {
! 1511: char *ptr = strstr(nextarg, ";auto");
! 1512: if(ptr) {
! 1513: /* Automatic referer requested, this may be combined with a
! 1514: set initial one */
! 1515: config->autoreferer = TRUE;
! 1516: *ptr = 0; /* zero terminate here */
! 1517: }
! 1518: else
! 1519: config->autoreferer = FALSE;
! 1520: GetStr(&config->referer, nextarg);
! 1521: }
! 1522: break;
! 1523: case 'E':
! 1524: switch(subletter) {
! 1525: case '\0': /* certificate file */
! 1526: GetFileAndPassword(nextarg, &config->cert, &config->key_passwd);
! 1527: break;
! 1528: case 'a': /* CA info PEM file */
! 1529: GetStr(&config->cacert, nextarg);
! 1530: break;
! 1531: case 'b': /* cert file type */
! 1532: GetStr(&config->cert_type, nextarg);
! 1533: break;
! 1534: case 'c': /* private key file */
! 1535: GetStr(&config->key, nextarg);
! 1536: break;
! 1537: case 'd': /* private key file type */
! 1538: GetStr(&config->key_type, nextarg);
! 1539: break;
! 1540: case 'e': /* private key passphrase */
! 1541: GetStr(&config->key_passwd, nextarg);
! 1542: cleanarg(nextarg);
! 1543: break;
! 1544: case 'f': /* crypto engine */
! 1545: GetStr(&config->engine, nextarg);
! 1546: if(config->engine && curl_strequal(config->engine, "list"))
! 1547: return PARAM_ENGINES_REQUESTED;
! 1548: break;
! 1549: case 'g': /* CA cert directory */
! 1550: GetStr(&config->capath, nextarg);
! 1551: break;
! 1552: case 'h': /* --pubkey public key file */
! 1553: GetStr(&config->pubkey, nextarg);
! 1554: break;
! 1555: case 'i': /* --hostpubmd5 md5 of the host public key */
! 1556: GetStr(&config->hostpubmd5, nextarg);
! 1557: if(!config->hostpubmd5 || strlen(config->hostpubmd5) != 32)
! 1558: return PARAM_BAD_USE;
! 1559: break;
! 1560: case 'j': /* CRL file */
! 1561: GetStr(&config->crlfile, nextarg);
! 1562: break;
! 1563: case 'k': /* TLS username */
! 1564: if(curlinfo->features & CURL_VERSION_TLSAUTH_SRP)
! 1565: GetStr(&config->tls_username, nextarg);
! 1566: else
! 1567: return PARAM_LIBCURL_DOESNT_SUPPORT;
! 1568: break;
! 1569: case 'l': /* TLS password */
! 1570: if(curlinfo->features & CURL_VERSION_TLSAUTH_SRP)
! 1571: GetStr(&config->tls_password, nextarg);
! 1572: else
! 1573: return PARAM_LIBCURL_DOESNT_SUPPORT;
! 1574: break;
! 1575: case 'm': /* TLS authentication type */
! 1576: if(curlinfo->features & CURL_VERSION_TLSAUTH_SRP) {
! 1577: GetStr(&config->tls_authtype, nextarg);
! 1578: if(!curl_strequal(config->tls_authtype, "SRP"))
! 1579: return PARAM_LIBCURL_DOESNT_SUPPORT; /* only support TLS-SRP */
! 1580: }
! 1581: else
! 1582: return PARAM_LIBCURL_DOESNT_SUPPORT;
! 1583: break;
! 1584: case 'n': /* no empty SSL fragments, --ssl-allow-beast */
! 1585: if(curlinfo->features & CURL_VERSION_SSL)
! 1586: config->ssl_allow_beast = toggle;
! 1587: break;
! 1588:
! 1589: case 'p': /* Pinned public key DER file */
! 1590: GetStr(&config->pinnedpubkey, nextarg);
! 1591: break;
! 1592:
! 1593: case 'P': /* proxy pinned public key */
! 1594: GetStr(&config->proxy_pinnedpubkey, nextarg);
! 1595: break;
! 1596:
! 1597: case 'q': /* --cert-status */
! 1598: config->verifystatus = TRUE;
! 1599: break;
! 1600:
! 1601: case 'r': /* --false-start */
! 1602: config->falsestart = TRUE;
! 1603: break;
! 1604:
! 1605: case 's': /* --ssl-no-revoke */
! 1606: if(curlinfo->features & CURL_VERSION_SSL)
! 1607: config->ssl_no_revoke = TRUE;
! 1608: break;
! 1609:
! 1610: case 'S': /* --ssl-revoke-best-effort */
! 1611: if(curlinfo->features & CURL_VERSION_SSL)
! 1612: config->ssl_revoke_best_effort = TRUE;
! 1613: break;
! 1614:
! 1615: case 't': /* --tcp-fastopen */
! 1616: config->tcp_fastopen = TRUE;
! 1617: break;
! 1618:
! 1619: case 'u': /* TLS username for proxy */
! 1620: if(curlinfo->features & CURL_VERSION_TLSAUTH_SRP)
! 1621: GetStr(&config->proxy_tls_username, nextarg);
! 1622: else
! 1623: return PARAM_LIBCURL_DOESNT_SUPPORT;
! 1624: break;
! 1625:
! 1626: case 'v': /* TLS password for proxy */
! 1627: if(curlinfo->features & CURL_VERSION_TLSAUTH_SRP)
! 1628: GetStr(&config->proxy_tls_password, nextarg);
! 1629: else
! 1630: return PARAM_LIBCURL_DOESNT_SUPPORT;
! 1631: break;
! 1632:
! 1633: case 'w': /* TLS authentication type for proxy */
! 1634: if(curlinfo->features & CURL_VERSION_TLSAUTH_SRP) {
! 1635: GetStr(&config->proxy_tls_authtype, nextarg);
! 1636: if(!curl_strequal(config->proxy_tls_authtype, "SRP"))
! 1637: return PARAM_LIBCURL_DOESNT_SUPPORT; /* only support TLS-SRP */
! 1638: }
! 1639: else
! 1640: return PARAM_LIBCURL_DOESNT_SUPPORT;
! 1641: break;
! 1642:
! 1643: case 'x': /* certificate file for proxy */
! 1644: GetFileAndPassword(nextarg, &config->proxy_cert,
! 1645: &config->proxy_key_passwd);
! 1646: break;
! 1647:
! 1648: case 'y': /* cert file type for proxy */
! 1649: GetStr(&config->proxy_cert_type, nextarg);
! 1650: break;
! 1651:
! 1652: case 'z': /* private key file for proxy */
! 1653: GetStr(&config->proxy_key, nextarg);
! 1654: break;
! 1655:
! 1656: case '0': /* private key file type for proxy */
! 1657: GetStr(&config->proxy_key_type, nextarg);
! 1658: break;
! 1659:
! 1660: case '1': /* private key passphrase for proxy */
! 1661: GetStr(&config->proxy_key_passwd, nextarg);
! 1662: cleanarg(nextarg);
! 1663: break;
! 1664:
! 1665: case '2': /* ciphers for proxy */
! 1666: GetStr(&config->proxy_cipher_list, nextarg);
! 1667: break;
! 1668:
! 1669: case '3': /* CRL file for proxy */
! 1670: GetStr(&config->proxy_crlfile, nextarg);
! 1671: break;
! 1672:
! 1673: case '4': /* no empty SSL fragments for proxy */
! 1674: if(curlinfo->features & CURL_VERSION_SSL)
! 1675: config->proxy_ssl_allow_beast = toggle;
! 1676: break;
! 1677:
! 1678: case '5': /* --login-options */
! 1679: GetStr(&config->login_options, nextarg);
! 1680: break;
! 1681:
! 1682: case '6': /* CA info PEM file for proxy */
! 1683: GetStr(&config->proxy_cacert, nextarg);
! 1684: break;
! 1685:
! 1686: case '7': /* CA cert directory for proxy */
! 1687: GetStr(&config->proxy_capath, nextarg);
! 1688: break;
! 1689:
! 1690: case '8': /* allow insecure SSL connects for proxy */
! 1691: config->proxy_insecure_ok = toggle;
! 1692: break;
! 1693:
! 1694: case '9': /* --proxy-tlsv1 */
! 1695: /* TLS version 1 for proxy */
! 1696: config->proxy_ssl_version = CURL_SSLVERSION_TLSv1;
! 1697: break;
! 1698:
! 1699: case 'A':
! 1700: /* --socks5-basic */
! 1701: if(toggle)
! 1702: config->socks5_auth |= CURLAUTH_BASIC;
! 1703: else
! 1704: config->socks5_auth &= ~CURLAUTH_BASIC;
! 1705: break;
! 1706:
! 1707: case 'B':
! 1708: /* --socks5-gssapi */
! 1709: if(toggle)
! 1710: config->socks5_auth |= CURLAUTH_GSSAPI;
! 1711: else
! 1712: config->socks5_auth &= ~CURLAUTH_GSSAPI;
! 1713: break;
! 1714:
! 1715: case 'C':
! 1716: GetStr(&config->etag_save_file, nextarg);
! 1717: break;
! 1718:
! 1719: case 'D':
! 1720: GetStr(&config->etag_compare_file, nextarg);
! 1721: break;
! 1722:
! 1723: default: /* unknown flag */
! 1724: return PARAM_OPTION_UNKNOWN;
! 1725: }
! 1726: break;
! 1727: case 'f':
! 1728: switch(subletter) {
! 1729: case 'a': /* --fail-early */
! 1730: global->fail_early = toggle;
! 1731: break;
! 1732: case 'b': /* --styled-output */
! 1733: global->styled_output = toggle;
! 1734: break;
! 1735: case 'c': /* --mail-rcpt-allowfails */
! 1736: config->mail_rcpt_allowfails = toggle;
! 1737: break;
! 1738: default: /* --fail (hard on errors) */
! 1739: config->failonerror = toggle;
! 1740: }
! 1741: break;
! 1742: case 'F':
! 1743: /* "form data" simulation, this is a little advanced so lets do our best
! 1744: to sort this out slowly and carefully */
! 1745: if(formparse(config,
! 1746: nextarg,
! 1747: &config->mimeroot,
! 1748: &config->mimecurrent,
! 1749: (subletter == 's')?TRUE:FALSE)) /* 's' is literal string */
! 1750: return PARAM_BAD_USE;
! 1751: if(SetHTTPrequest(config, HTTPREQ_MIMEPOST, &config->httpreq))
! 1752: return PARAM_BAD_USE;
! 1753: break;
! 1754:
! 1755: case 'g': /* g disables URLglobbing */
! 1756: config->globoff = toggle;
! 1757: break;
! 1758:
! 1759: case 'G': /* HTTP GET */
! 1760: if(subletter == 'a') { /* --request-target */
! 1761: GetStr(&config->request_target, nextarg);
! 1762: }
! 1763: else
! 1764: config->use_httpget = TRUE;
! 1765: break;
! 1766:
! 1767: case 'h': /* h for help */
! 1768: if(toggle) {
! 1769: return PARAM_HELP_REQUESTED;
! 1770: }
! 1771: /* we now actually support --no-help too! */
! 1772: break;
! 1773: case 'H':
! 1774: /* A custom header to append to a list */
! 1775: if(nextarg[0] == '@') {
! 1776: /* read many headers from a file or stdin */
! 1777: char *string;
! 1778: size_t len;
! 1779: bool use_stdin = !strcmp(&nextarg[1], "-");
! 1780: FILE *file = use_stdin?stdin:fopen(&nextarg[1], FOPEN_READTEXT);
! 1781: if(!file)
! 1782: warnf(global, "Failed to open %s!\n", &nextarg[1]);
! 1783: else {
! 1784: err = file2memory(&string, &len, file);
! 1785: if(!err && string) {
! 1786: /* Allow strtok() here since this isn't used threaded */
! 1787: /* !checksrc! disable BANNEDFUNC 2 */
! 1788: char *h = strtok(string, "\r\n");
! 1789: while(h) {
! 1790: if(subletter == 'p') /* --proxy-header */
! 1791: err = add2list(&config->proxyheaders, h);
! 1792: else
! 1793: err = add2list(&config->headers, h);
! 1794: if(err)
! 1795: break;
! 1796: h = strtok(NULL, "\r\n");
! 1797: }
! 1798: free(string);
! 1799: }
! 1800: if(!use_stdin)
! 1801: fclose(file);
! 1802: if(err)
! 1803: return err;
! 1804: }
! 1805: }
! 1806: else {
! 1807: if(subletter == 'p') /* --proxy-header */
! 1808: err = add2list(&config->proxyheaders, nextarg);
! 1809: else
! 1810: err = add2list(&config->headers, nextarg);
! 1811: if(err)
! 1812: return err;
! 1813: }
! 1814: break;
! 1815: case 'i':
! 1816: config->show_headers = toggle; /* show the headers as well in the
! 1817: general output stream */
! 1818: break;
! 1819: case 'j':
! 1820: config->cookiesession = toggle;
! 1821: break;
! 1822: case 'I': /* --head */
! 1823: config->no_body = toggle;
! 1824: config->show_headers = toggle;
! 1825: if(SetHTTPrequest(config,
! 1826: (config->no_body)?HTTPREQ_HEAD:HTTPREQ_GET,
! 1827: &config->httpreq))
! 1828: return PARAM_BAD_USE;
! 1829: break;
! 1830: case 'J': /* --remote-header-name */
! 1831: if(config->show_headers) {
! 1832: warnf(global,
! 1833: "--include and --remote-header-name cannot be combined.\n");
! 1834: return PARAM_BAD_USE;
! 1835: }
! 1836: config->content_disposition = toggle;
! 1837: break;
! 1838: case 'k': /* allow insecure SSL connects */
! 1839: config->insecure_ok = toggle;
! 1840: break;
! 1841: case 'K': /* parse config file */
! 1842: if(parseconfig(nextarg, global))
! 1843: warnf(global, "error trying read config from the '%s' file\n",
! 1844: nextarg);
! 1845: break;
! 1846: case 'l':
! 1847: config->dirlistonly = toggle; /* only list the names of the FTP dir */
! 1848: break;
! 1849: case 'L':
! 1850: config->followlocation = toggle; /* Follow Location: HTTP headers */
! 1851: switch(subletter) {
! 1852: case 't':
! 1853: /* Continue to send authentication (user+password) when following
! 1854: * locations, even when hostname changed */
! 1855: config->unrestricted_auth = toggle;
! 1856: break;
! 1857: }
! 1858: break;
! 1859: case 'm':
! 1860: /* specified max time */
! 1861: err = str2udouble(&config->timeout, nextarg, LONG_MAX/1000);
! 1862: if(err)
! 1863: return err;
! 1864: break;
! 1865: case 'M': /* M for manual, huge help */
! 1866: if(toggle) { /* --no-manual shows no manual... */
! 1867: #ifdef USE_MANUAL
! 1868: return PARAM_MANUAL_REQUESTED;
! 1869: #else
! 1870: warnf(global,
! 1871: "built-in manual was disabled at build-time!\n");
! 1872: return PARAM_OPTION_UNKNOWN;
! 1873: #endif
! 1874: }
! 1875: break;
! 1876: case 'n':
! 1877: switch(subletter) {
! 1878: case 'o': /* use .netrc or URL */
! 1879: config->netrc_opt = toggle;
! 1880: break;
! 1881: case 'e': /* netrc-file */
! 1882: GetStr(&config->netrc_file, nextarg);
! 1883: break;
! 1884: default:
! 1885: /* pick info from .netrc, if this is used for http, curl will
! 1886: automatically enfore user+password with the request */
! 1887: config->netrc = toggle;
! 1888: break;
! 1889: }
! 1890: break;
! 1891: case 'N':
! 1892: /* disable the output I/O buffering. note that the option is called
! 1893: --buffer but is mostly used in the negative form: --no-buffer */
! 1894: if(longopt)
! 1895: config->nobuffer = (!toggle)?TRUE:FALSE;
! 1896: else
! 1897: config->nobuffer = toggle;
! 1898: break;
! 1899: case 'O': /* --remote-name */
! 1900: if(subletter == 'a') { /* --remote-name-all */
! 1901: config->default_node_flags = toggle?GETOUT_USEREMOTE:0;
! 1902: break;
! 1903: }
! 1904: /* FALLTHROUGH */
! 1905: case 'o': /* --output */
! 1906: /* output file */
! 1907: {
! 1908: struct getout *url;
! 1909: if(!config->url_out)
! 1910: config->url_out = config->url_list;
! 1911: if(config->url_out) {
! 1912: /* there's a node here, if it already is filled-in continue to find
! 1913: an "empty" node */
! 1914: while(config->url_out && (config->url_out->flags & GETOUT_OUTFILE))
! 1915: config->url_out = config->url_out->next;
! 1916: }
! 1917:
! 1918: /* now there might or might not be an available node to fill in! */
! 1919:
! 1920: if(config->url_out)
! 1921: /* existing node */
! 1922: url = config->url_out;
! 1923: else
! 1924: /* there was no free node, create one! */
! 1925: config->url_out = url = new_getout(config);
! 1926:
! 1927: if(!url)
! 1928: return PARAM_NO_MEM;
! 1929:
! 1930: /* fill in the outfile */
! 1931: if('o' == letter) {
! 1932: GetStr(&url->outfile, nextarg);
! 1933: url->flags &= ~GETOUT_USEREMOTE; /* switch off */
! 1934: }
! 1935: else {
! 1936: url->outfile = NULL; /* leave it */
! 1937: if(toggle)
! 1938: url->flags |= GETOUT_USEREMOTE; /* switch on */
! 1939: else
! 1940: url->flags &= ~GETOUT_USEREMOTE; /* switch off */
! 1941: }
! 1942: url->flags |= GETOUT_OUTFILE;
! 1943: }
! 1944: break;
! 1945: case 'P':
! 1946: /* This makes the FTP sessions use PORT instead of PASV */
! 1947: /* use <eth0> or <192.168.10.10> style addresses. Anything except
! 1948: this will make us try to get the "default" address.
! 1949: NOTE: this is a changed behaviour since the released 4.1!
! 1950: */
! 1951: GetStr(&config->ftpport, nextarg);
! 1952: break;
! 1953: case 'p':
! 1954: /* proxy tunnel for non-http protocols */
! 1955: config->proxytunnel = toggle;
! 1956: break;
! 1957:
! 1958: case 'q': /* if used first, already taken care of, we do it like
! 1959: this so we don't cause an error! */
! 1960: break;
! 1961: case 'Q':
! 1962: /* QUOTE command to send to FTP server */
! 1963: switch(nextarg[0]) {
! 1964: case '-':
! 1965: /* prefixed with a dash makes it a POST TRANSFER one */
! 1966: nextarg++;
! 1967: err = add2list(&config->postquote, nextarg);
! 1968: break;
! 1969: case '+':
! 1970: /* prefixed with a plus makes it a just-before-transfer one */
! 1971: nextarg++;
! 1972: err = add2list(&config->prequote, nextarg);
! 1973: break;
! 1974: default:
! 1975: err = add2list(&config->quote, nextarg);
! 1976: break;
! 1977: }
! 1978: if(err)
! 1979: return err;
! 1980: break;
! 1981: case 'r':
! 1982: /* Specifying a range WITHOUT A DASH will create an illegal HTTP range
! 1983: (and won't actually be range by definition). The man page previously
! 1984: claimed that to be a good way, why this code is added to work-around
! 1985: it. */
! 1986: if(ISDIGIT(*nextarg) && !strchr(nextarg, '-')) {
! 1987: char buffer[32];
! 1988: curl_off_t off;
! 1989: if(curlx_strtoofft(nextarg, NULL, 10, &off)) {
! 1990: warnf(global, "unsupported range point\n");
! 1991: return PARAM_BAD_USE;
! 1992: }
! 1993: warnf(global,
! 1994: "A specified range MUST include at least one dash (-). "
! 1995: "Appending one for you!\n");
! 1996: msnprintf(buffer, sizeof(buffer), "%" CURL_FORMAT_CURL_OFF_T "-", off);
! 1997: Curl_safefree(config->range);
! 1998: config->range = strdup(buffer);
! 1999: if(!config->range)
! 2000: return PARAM_NO_MEM;
! 2001: }
! 2002: {
! 2003: /* byte range requested */
! 2004: const char *tmp_range = nextarg;
! 2005: while(*tmp_range != '\0') {
! 2006: if(!ISDIGIT(*tmp_range) && *tmp_range != '-' && *tmp_range != ',') {
! 2007: warnf(global, "Invalid character is found in given range. "
! 2008: "A specified range MUST have only digits in "
! 2009: "\'start\'-\'stop\'. The server's response to this "
! 2010: "request is uncertain.\n");
! 2011: break;
! 2012: }
! 2013: tmp_range++;
! 2014: }
! 2015: /* byte range requested */
! 2016: GetStr(&config->range, nextarg);
! 2017: }
! 2018: break;
! 2019: case 'R':
! 2020: /* use remote file's time */
! 2021: config->remote_time = toggle;
! 2022: break;
! 2023: case 's':
! 2024: /* don't show progress meter, don't show errors : */
! 2025: if(toggle)
! 2026: global->mute = global->noprogress = TRUE;
! 2027: else
! 2028: global->mute = global->noprogress = FALSE;
! 2029: if(global->showerror < 0)
! 2030: /* if still on the default value, set showerror to the reverse of
! 2031: toggle. This is to allow -S and -s to be used in an independent
! 2032: order but still have the same effect. */
! 2033: global->showerror = (!toggle)?TRUE:FALSE; /* toggle off */
! 2034: break;
! 2035: case 'S':
! 2036: /* show errors */
! 2037: global->showerror = toggle?1:0; /* toggle on if used with -s */
! 2038: break;
! 2039: case 't':
! 2040: /* Telnet options */
! 2041: err = add2list(&config->telnet_options, nextarg);
! 2042: if(err)
! 2043: return err;
! 2044: break;
! 2045: case 'T':
! 2046: /* we are uploading */
! 2047: {
! 2048: struct getout *url;
! 2049: if(!config->url_ul)
! 2050: config->url_ul = config->url_list;
! 2051: if(config->url_ul) {
! 2052: /* there's a node here, if it already is filled-in continue to find
! 2053: an "empty" node */
! 2054: while(config->url_ul && (config->url_ul->flags & GETOUT_UPLOAD))
! 2055: config->url_ul = config->url_ul->next;
! 2056: }
! 2057:
! 2058: /* now there might or might not be an available node to fill in! */
! 2059:
! 2060: if(config->url_ul)
! 2061: /* existing node */
! 2062: url = config->url_ul;
! 2063: else
! 2064: /* there was no free node, create one! */
! 2065: config->url_ul = url = new_getout(config);
! 2066:
! 2067: if(!url)
! 2068: return PARAM_NO_MEM;
! 2069:
! 2070: url->flags |= GETOUT_UPLOAD; /* mark -T used */
! 2071: if(!*nextarg)
! 2072: url->flags |= GETOUT_NOUPLOAD;
! 2073: else {
! 2074: /* "-" equals stdin, but keep the string around for now */
! 2075: GetStr(&url->infile, nextarg);
! 2076: }
! 2077: }
! 2078: break;
! 2079: case 'u':
! 2080: /* user:password */
! 2081: GetStr(&config->userpwd, nextarg);
! 2082: cleanarg(nextarg);
! 2083: break;
! 2084: case 'U':
! 2085: /* Proxy user:password */
! 2086: GetStr(&config->proxyuserpwd, nextarg);
! 2087: cleanarg(nextarg);
! 2088: break;
! 2089: case 'v':
! 2090: if(toggle) {
! 2091: /* the '%' thing here will cause the trace get sent to stderr */
! 2092: Curl_safefree(global->trace_dump);
! 2093: global->trace_dump = strdup("%");
! 2094: if(!global->trace_dump)
! 2095: return PARAM_NO_MEM;
! 2096: if(global->tracetype && (global->tracetype != TRACE_PLAIN))
! 2097: warnf(global,
! 2098: "-v, --verbose overrides an earlier trace/verbose option\n");
! 2099: global->tracetype = TRACE_PLAIN;
! 2100: }
! 2101: else
! 2102: /* verbose is disabled here */
! 2103: global->tracetype = TRACE_NONE;
! 2104: break;
! 2105: case 'V':
! 2106: if(toggle) /* --no-version yields no output! */
! 2107: return PARAM_VERSION_INFO_REQUESTED;
! 2108: break;
! 2109:
! 2110: case 'w':
! 2111: /* get the output string */
! 2112: if('@' == *nextarg) {
! 2113: /* the data begins with a '@' letter, it means that a file name
! 2114: or - (stdin) follows */
! 2115: FILE *file;
! 2116: const char *fname;
! 2117: nextarg++; /* pass the @ */
! 2118: if(!strcmp("-", nextarg)) {
! 2119: fname = "<stdin>";
! 2120: file = stdin;
! 2121: }
! 2122: else {
! 2123: fname = nextarg;
! 2124: file = fopen(nextarg, FOPEN_READTEXT);
! 2125: }
! 2126: Curl_safefree(config->writeout);
! 2127: err = file2string(&config->writeout, file);
! 2128: if(file && (file != stdin))
! 2129: fclose(file);
! 2130: if(err)
! 2131: return err;
! 2132: if(!config->writeout)
! 2133: warnf(global, "Failed to read %s", fname);
! 2134: }
! 2135: else
! 2136: GetStr(&config->writeout, nextarg);
! 2137: break;
! 2138: case 'x':
! 2139: switch(subletter) {
! 2140: case 'a': /* --preproxy */
! 2141: GetStr(&config->preproxy, nextarg);
! 2142: break;
! 2143: default:
! 2144: /* --proxy */
! 2145: GetStr(&config->proxy, nextarg);
! 2146: config->proxyver = CURLPROXY_HTTP;
! 2147: break;
! 2148: }
! 2149: break;
! 2150: case 'X':
! 2151: /* set custom request */
! 2152: GetStr(&config->customrequest, nextarg);
! 2153: break;
! 2154: case 'y':
! 2155: /* low speed time */
! 2156: err = str2unum(&config->low_speed_time, nextarg);
! 2157: if(err)
! 2158: return err;
! 2159: if(!config->low_speed_limit)
! 2160: config->low_speed_limit = 1;
! 2161: break;
! 2162: case 'Y':
! 2163: /* low speed limit */
! 2164: err = str2unum(&config->low_speed_limit, nextarg);
! 2165: if(err)
! 2166: return err;
! 2167: if(!config->low_speed_time)
! 2168: config->low_speed_time = 30;
! 2169: break;
! 2170: case 'Z':
! 2171: switch(subletter) {
! 2172: case '\0': /* --parallel */
! 2173: global->parallel = toggle;
! 2174: break;
! 2175: case 'b': /* --parallel-max */
! 2176: err = str2unum(&global->parallel_max, nextarg);
! 2177: if(err)
! 2178: return err;
! 2179: if((global->parallel_max > MAX_PARALLEL) ||
! 2180: (global->parallel_max < 1))
! 2181: global->parallel_max = PARALLEL_DEFAULT;
! 2182: break;
! 2183: case 'c': /* --parallel-connect */
! 2184: global->parallel_connect = toggle;
! 2185: break;
! 2186: }
! 2187: break;
! 2188: case 'z': /* time condition coming up */
! 2189: switch(*nextarg) {
! 2190: case '+':
! 2191: nextarg++;
! 2192: /* FALLTHROUGH */
! 2193: default:
! 2194: /* If-Modified-Since: (section 14.28 in RFC2068) */
! 2195: config->timecond = CURL_TIMECOND_IFMODSINCE;
! 2196: break;
! 2197: case '-':
! 2198: /* If-Unmodified-Since: (section 14.24 in RFC2068) */
! 2199: config->timecond = CURL_TIMECOND_IFUNMODSINCE;
! 2200: nextarg++;
! 2201: break;
! 2202: case '=':
! 2203: /* Last-Modified: (section 14.29 in RFC2068) */
! 2204: config->timecond = CURL_TIMECOND_LASTMOD;
! 2205: nextarg++;
! 2206: break;
! 2207: }
! 2208: now = time(NULL);
! 2209: config->condtime = (curl_off_t)curl_getdate(nextarg, &now);
! 2210: if(-1 == config->condtime) {
! 2211: /* now let's see if it is a file name to get the time from instead! */
! 2212: curl_off_t filetime = getfiletime(nextarg, config->global->errors);
! 2213: if(filetime >= 0) {
! 2214: /* pull the time out from the file */
! 2215: config->condtime = filetime;
! 2216: }
! 2217: else {
! 2218: /* failed, remove time condition */
! 2219: config->timecond = CURL_TIMECOND_NONE;
! 2220: warnf(global,
! 2221: "Illegal date format for -z, --time-cond (and not "
! 2222: "a file name). Disabling time condition. "
! 2223: "See curl_getdate(3) for valid date syntax.\n");
! 2224: }
! 2225: }
! 2226: break;
! 2227: default: /* unknown flag */
! 2228: return PARAM_OPTION_UNKNOWN;
! 2229: }
! 2230: hit = -1;
! 2231:
! 2232: } while(!longopt && !singleopt && *++parse && !*usedarg);
! 2233:
! 2234: return PARAM_OK;
! 2235: }
! 2236:
! 2237: ParameterError parse_args(struct GlobalConfig *global, int argc,
! 2238: argv_item_t argv[])
! 2239: {
! 2240: int i;
! 2241: bool stillflags;
! 2242: char *orig_opt = NULL;
! 2243: ParameterError result = PARAM_OK;
! 2244: struct OperationConfig *config = global->first;
! 2245:
! 2246: for(i = 1, stillflags = TRUE; i < argc && !result; i++) {
! 2247: orig_opt = argv[i];
! 2248:
! 2249: if(stillflags && ('-' == argv[i][0])) {
! 2250: bool passarg;
! 2251: char *flag = argv[i];
! 2252:
! 2253: if(!strcmp("--", argv[i]))
! 2254: /* This indicates the end of the flags and thus enables the
! 2255: following (URL) argument to start with -. */
! 2256: stillflags = FALSE;
! 2257: else {
! 2258: char *nextarg = (i < (argc - 1)) ? argv[i + 1] : NULL;
! 2259:
! 2260: result = getparameter(flag, nextarg, &passarg, global, config);
! 2261: config = global->last;
! 2262: if(result == PARAM_NEXT_OPERATION) {
! 2263: /* Reset result as PARAM_NEXT_OPERATION is only used here and not
! 2264: returned from this function */
! 2265: result = PARAM_OK;
! 2266:
! 2267: if(config->url_list && config->url_list->url) {
! 2268: /* Allocate the next config */
! 2269: config->next = malloc(sizeof(struct OperationConfig));
! 2270: if(config->next) {
! 2271: /* Initialise the newly created config */
! 2272: config_init(config->next);
! 2273:
! 2274: /* Set the global config pointer */
! 2275: config->next->global = global;
! 2276:
! 2277: /* Update the last config pointer */
! 2278: global->last = config->next;
! 2279:
! 2280: /* Move onto the new config */
! 2281: config->next->prev = config;
! 2282: config = config->next;
! 2283: }
! 2284: else
! 2285: result = PARAM_NO_MEM;
! 2286: }
! 2287: }
! 2288: else if(!result && passarg)
! 2289: i++; /* we're supposed to skip this */
! 2290: }
! 2291: }
! 2292: else {
! 2293: bool used;
! 2294:
! 2295: /* Just add the URL please */
! 2296: result = getparameter((char *)"--url", argv[i], &used, global,
! 2297: config);
! 2298: }
! 2299: }
! 2300:
! 2301: if(result && result != PARAM_HELP_REQUESTED &&
! 2302: result != PARAM_MANUAL_REQUESTED &&
! 2303: result != PARAM_VERSION_INFO_REQUESTED &&
! 2304: result != PARAM_ENGINES_REQUESTED) {
! 2305: const char *reason = param2text(result);
! 2306:
! 2307: if(orig_opt && strcmp(":", orig_opt))
! 2308: helpf(global->errors, "option %s: %s\n", orig_opt, reason);
! 2309: else
! 2310: helpf(global->errors, "%s\n", reason);
! 2311: }
! 2312:
! 2313: return result;
! 2314: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>