1999-12-29 15:20:26 +01:00
|
|
|
/*****************************************************************************
|
|
|
|
* _ _ ____ _
|
|
|
|
* Project ___| | | | _ \| |
|
|
|
|
* / __| | | | |_) | |
|
|
|
|
* | (__| |_| | _ <| |___
|
|
|
|
* \___|\___/|_| \_\_____|
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla Public License
|
|
|
|
* Version 1.0 (the "License"); you may not use this file except in
|
|
|
|
* compliance with the License. You may obtain a copy of the License at
|
|
|
|
* http://www.mozilla.org/MPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an "AS IS"
|
|
|
|
* basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the
|
|
|
|
* License for the specific language governing rights and limitations
|
|
|
|
* under the License.
|
|
|
|
*
|
|
|
|
* The Original Code is Curl.
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is Daniel Stenberg.
|
|
|
|
*
|
|
|
|
* Portions created by the Initial Developer are Copyright (C) 1998.
|
|
|
|
* All Rights Reserved.
|
|
|
|
*
|
|
|
|
* ------------------------------------------------------------
|
|
|
|
* Main author:
|
2000-11-09 13:51:43 +01:00
|
|
|
* - Daniel Stenberg <daniel@haxx.se>
|
1999-12-29 15:20:26 +01:00
|
|
|
*
|
2000-11-09 13:51:43 +01:00
|
|
|
* http://curl.haxx.se
|
1999-12-29 15:20:26 +01:00
|
|
|
*
|
|
|
|
* $Source$
|
|
|
|
* $Revision$
|
|
|
|
* $Date$
|
|
|
|
* $Author$
|
|
|
|
* $State$
|
|
|
|
* $Locker$
|
|
|
|
*
|
|
|
|
* ------------------------------------------------------------
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
|
|
|
|
#include <curl/curl.h>
|
2000-05-22 16:12:12 +02:00
|
|
|
#include <curl/types.h> /* new for v7 */
|
|
|
|
#include <curl/easy.h> /* new for v7 */
|
2000-10-09 23:35:40 +02:00
|
|
|
|
|
|
|
#define _MPRINTF_REPLACE /* we want curl-functions instead of native ones */
|
1999-12-29 15:20:26 +01:00
|
|
|
#include <curl/mprintf.h>
|
2000-05-10 00:42:53 +02:00
|
|
|
|
1999-12-29 15:20:26 +01:00
|
|
|
#include "urlglob.h"
|
|
|
|
#define CURLseparator "--_curl_--"
|
|
|
|
#define MIMEseparator "_curl_"
|
2000-05-10 00:42:53 +02:00
|
|
|
|
|
|
|
/* This define make use of the "Curlseparator" as opposed to the
|
|
|
|
MIMEseparator. We might add support for the latter one in the
|
|
|
|
future, and that's why this is left in the source. */
|
|
|
|
#define CURL_SEPARATORS
|
1999-12-29 15:20:26 +01:00
|
|
|
|
|
|
|
/* This is now designed to have its own local setup.h */
|
|
|
|
#include "setup.h"
|
|
|
|
|
2000-05-22 19:20:29 +02:00
|
|
|
#ifdef WIN32
|
|
|
|
#include <winsock.h>
|
|
|
|
#endif
|
|
|
|
|
1999-12-29 15:20:26 +01:00
|
|
|
#include "version.h"
|
|
|
|
|
|
|
|
#ifdef HAVE_IO_H /* typical win32 habit */
|
|
|
|
#include <io.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
|
|
|
|
2000-06-14 14:52:21 +02:00
|
|
|
#ifdef HAVE_FCNTL_H
|
|
|
|
#include <fcntl.h>
|
|
|
|
#endif
|
|
|
|
|
2000-10-09 13:13:17 +02:00
|
|
|
/* The last #include file should be: */
|
|
|
|
#ifdef MALLOCDEBUG
|
|
|
|
/* this is low-level hard-hacking memory leak tracking shit */
|
|
|
|
#include "../lib/memdebug.h"
|
|
|
|
#endif
|
|
|
|
|
2000-11-17 11:08:39 +01:00
|
|
|
#define DEBUG_CONFIG
|
|
|
|
|
2000-10-09 23:35:40 +02:00
|
|
|
#ifndef __cplusplus /* (rabe) */
|
|
|
|
typedef char bool;
|
|
|
|
#endif /* (rabe) */
|
|
|
|
|
2000-07-25 09:34:04 +02:00
|
|
|
typedef enum {
|
|
|
|
HTTPREQ_UNSPEC,
|
|
|
|
HTTPREQ_GET,
|
|
|
|
HTTPREQ_HEAD,
|
|
|
|
HTTPREQ_POST,
|
|
|
|
HTTPREQ_SIMPLEPOST,
|
|
|
|
HTTPREQ_CUSTOM,
|
|
|
|
HTTPREQ_LAST
|
|
|
|
} HttpReq;
|
|
|
|
|
2000-05-22 16:12:12 +02:00
|
|
|
/* Just a set of bits */
|
|
|
|
#define CONF_DEFAULT 0
|
2000-06-20 11:28:09 +02:00
|
|
|
|
|
|
|
#define CONF_AUTO_REFERER (1<<4) /* the automatic referer-system please! */
|
2000-05-22 16:12:12 +02:00
|
|
|
#define CONF_VERBOSE (1<<5) /* talk a lot */
|
|
|
|
#define CONF_HEADER (1<<8) /* throw the header out too */
|
|
|
|
#define CONF_NOPROGRESS (1<<10) /* shut off the progress meter */
|
|
|
|
#define CONF_NOBODY (1<<11) /* use HEAD to get http document */
|
|
|
|
#define CONF_FAILONERROR (1<<12) /* no output on http error codes >= 300 */
|
|
|
|
#define CONF_UPLOAD (1<<14) /* this is an upload */
|
|
|
|
#define CONF_POST (1<<15) /* HTTP POST method */
|
|
|
|
#define CONF_FTPLISTONLY (1<<16) /* Use NLST when listing ftp dir */
|
|
|
|
#define CONF_FTPAPPEND (1<<20) /* Append instead of overwrite on upload! */
|
|
|
|
#define CONF_NETRC (1<<22) /* read user+password from .netrc */
|
|
|
|
#define CONF_FOLLOWLOCATION (1<<23) /* use Location: Luke! */
|
2000-06-14 14:50:38 +02:00
|
|
|
#define CONF_GETTEXT (1<<24) /* use ASCII/text for transfer */
|
2000-05-22 16:12:12 +02:00
|
|
|
#define CONF_HTTPPOST (1<<25) /* multipart/form-data HTTP POST */
|
|
|
|
#define CONF_PUT (1<<27) /* PUT the input file */
|
|
|
|
#define CONF_MUTE (1<<28) /* force NOPROGRESS */
|
|
|
|
|
2000-03-03 00:06:34 +01:00
|
|
|
#ifndef HAVE_STRDUP
|
|
|
|
/* Ultrix doesn't have strdup(), so make a quick clone: */
|
|
|
|
char *strdup(char *str)
|
|
|
|
{
|
|
|
|
int len;
|
|
|
|
char *newstr;
|
|
|
|
|
|
|
|
len = strlen(str);
|
|
|
|
newstr = (char *) malloc((len+1)*sizeof(char));
|
|
|
|
if (!newstr)
|
|
|
|
return (char *)NULL;
|
|
|
|
|
|
|
|
strcpy(newstr,str);
|
|
|
|
|
|
|
|
return newstr;
|
|
|
|
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
1999-12-29 15:20:26 +01:00
|
|
|
extern void hugehelp(void);
|
|
|
|
|
2000-05-17 23:21:10 +02:00
|
|
|
/***********************************************************************
|
|
|
|
* Start with some silly functions to make win32-systems survive
|
|
|
|
***********************************************************************/
|
|
|
|
#if defined(WIN32) && !defined(__GNUC__) || defined(__MINGW32__)
|
|
|
|
static void win32_cleanup(void)
|
|
|
|
{
|
|
|
|
WSACleanup();
|
|
|
|
}
|
|
|
|
|
2000-05-22 19:20:29 +02:00
|
|
|
static CURLcode win32_init(void)
|
2000-05-17 23:21:10 +02:00
|
|
|
{
|
|
|
|
WORD wVersionRequested;
|
|
|
|
WSADATA wsaData;
|
|
|
|
int err;
|
|
|
|
wVersionRequested = MAKEWORD(1, 1);
|
|
|
|
|
|
|
|
err = WSAStartup(wVersionRequested, &wsaData);
|
|
|
|
|
|
|
|
if (err != 0)
|
|
|
|
/* Tell the user that we couldn't find a useable */
|
|
|
|
/* winsock.dll. */
|
2000-05-22 16:12:12 +02:00
|
|
|
return CURLE_FAILED_INIT;
|
2000-05-17 23:21:10 +02:00
|
|
|
|
|
|
|
/* Confirm that the Windows Sockets DLL supports 1.1.*/
|
|
|
|
/* Note that if the DLL supports versions greater */
|
|
|
|
/* than 1.1 in addition to 1.1, it will still return */
|
|
|
|
/* 1.1 in wVersion since that is the version we */
|
|
|
|
/* requested. */
|
|
|
|
|
|
|
|
if ( LOBYTE( wsaData.wVersion ) != 1 ||
|
|
|
|
HIBYTE( wsaData.wVersion ) != 1 ) {
|
|
|
|
/* Tell the user that we couldn't find a useable */
|
|
|
|
|
|
|
|
/* winsock.dll. */
|
|
|
|
WSACleanup();
|
2000-05-22 16:12:12 +02:00
|
|
|
return CURLE_FAILED_INIT;
|
2000-05-17 23:21:10 +02:00
|
|
|
}
|
2000-05-22 16:12:12 +02:00
|
|
|
return CURLE_OK;
|
2000-05-17 23:21:10 +02:00
|
|
|
}
|
|
|
|
/* The Windows Sockets DLL is acceptable. Proceed. */
|
|
|
|
#else
|
2000-05-22 16:12:12 +02:00
|
|
|
static CURLcode win32_init(void) { return CURLE_OK; }
|
2000-05-17 23:21:10 +02:00
|
|
|
#define win32_cleanup()
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This is the main global constructor for the app. Call this before
|
|
|
|
* _any_ libcurl usage. If this fails, *NO* libcurl functions may be
|
|
|
|
* used, or havoc may be the result.
|
|
|
|
*/
|
2000-05-22 16:12:12 +02:00
|
|
|
CURLcode main_init(void)
|
2000-05-17 23:21:10 +02:00
|
|
|
{
|
|
|
|
return win32_init();
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This is the main global destructor for the app. Call this after
|
|
|
|
* _all_ libcurl usage is done.
|
|
|
|
*/
|
|
|
|
void main_free(void)
|
|
|
|
{
|
|
|
|
win32_cleanup();
|
|
|
|
}
|
|
|
|
|
2000-07-25 09:34:04 +02:00
|
|
|
int SetHTTPrequest(HttpReq req, HttpReq *store)
|
|
|
|
{
|
|
|
|
if((*store == HTTPREQ_UNSPEC) ||
|
|
|
|
(*store == req)) {
|
|
|
|
*store = req;
|
2000-11-17 10:47:18 +01:00
|
|
|
return 0;
|
2000-07-25 09:34:04 +02:00
|
|
|
}
|
|
|
|
fprintf(stderr, "You can only select one HTTP request!\n");
|
2000-11-17 10:47:18 +01:00
|
|
|
return 1;
|
2000-07-25 09:34:04 +02:00
|
|
|
}
|
2000-05-17 23:21:10 +02:00
|
|
|
|
1999-12-29 15:20:26 +01:00
|
|
|
static void helpf(char *fmt, ...)
|
|
|
|
{
|
|
|
|
va_list ap;
|
|
|
|
if(fmt) {
|
|
|
|
va_start(ap, fmt);
|
|
|
|
fputs("curl: ", stderr); /* prefix it */
|
|
|
|
vfprintf(stderr, fmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
}
|
|
|
|
fprintf(stderr, "curl: try 'curl --help' for more information\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void help(void)
|
|
|
|
{
|
|
|
|
printf(CURL_ID "%s\n"
|
|
|
|
"Usage: curl [options...] <url>\n"
|
|
|
|
"Options: (H) means HTTP/HTTPS only, (F) means FTP only\n"
|
|
|
|
" -a/--append Append to target file when uploading (F)\n"
|
|
|
|
" -A/--user-agent <string> User-Agent to send to server (H)\n"
|
|
|
|
" -b/--cookie <name=string/file> Cookie string or file to read cookies from (H)\n"
|
2000-06-14 14:50:38 +02:00
|
|
|
" -B/--use-ascii Use ASCII/text transfer\n"
|
1999-12-29 15:20:26 +01:00
|
|
|
" -c/--continue Resume a previous transfer where we left it\n"
|
|
|
|
" -C/--continue-at <offset> Specify absolute resume offset\n"
|
2000-08-24 19:56:20 +02:00
|
|
|
" -d/--data <data> HTTP POST data (H)\n"
|
|
|
|
" --data-ascii <data> HTTP POST ASCII data (H)\n"
|
|
|
|
" --data-binary <data> HTTP POST binary data (H)\n"
|
1999-12-29 15:20:26 +01:00
|
|
|
" -D/--dump-header <file> Write the headers to this file\n"
|
|
|
|
" -e/--referer Referer page (H)\n"
|
|
|
|
" -E/--cert <cert:passwd> Specifies your certificate file and password (HTTPS)\n"
|
|
|
|
" -f/--fail Fail silently (no output at all) on errors (H)\n"
|
|
|
|
" -F/--form <name=content> Specify HTTP POST data (H)\n"
|
|
|
|
|
|
|
|
" -h/--help This help text\n"
|
|
|
|
" -H/--header <line> Custom header to pass to server. (H)\n"
|
|
|
|
" -i/--include Include the HTTP-header in the output (H)\n"
|
|
|
|
" -I/--head Fetch document info only (HTTP HEAD/FTP SIZE)\n"
|
2000-09-18 23:54:46 +02:00
|
|
|
" --interface <interface> Specify the interface to be used\n"
|
2000-09-21 10:52:08 +02:00
|
|
|
" --krb4 <level> Enable krb4 with specified security level (F)\n"
|
1999-12-29 15:20:26 +01:00
|
|
|
" -K/--config Specify which config file to read\n"
|
|
|
|
" -l/--list-only List only names of an FTP directory (F)\n"
|
|
|
|
" -L/--location Follow Location: hints (H)\n"
|
|
|
|
" -m/--max-time <seconds> Maximum time allowed for the transfer\n"
|
|
|
|
" -M/--manual Display huge help text\n"
|
|
|
|
" -n/--netrc Read .netrc for user name and password\n"
|
2000-03-03 00:06:34 +01:00
|
|
|
" -N/--no-buffer Disables the buffering of the output stream\n"
|
1999-12-29 15:20:26 +01:00
|
|
|
" -o/--output <file> Write output to <file> instead of stdout\n"
|
|
|
|
" -O/--remote-name Write output to a file named as the remote file\n"
|
2000-09-15 08:11:46 +02:00
|
|
|
" -p/--proxytunnel Perform non-HTTP services through a HTTP proxy\n"
|
1999-12-29 15:20:26 +01:00
|
|
|
" -P/--ftpport <address> Use PORT with address instead of PASV when ftping (F)\n"
|
|
|
|
" -q When used as the first parameter disables .curlrc\n"
|
|
|
|
" -Q/--quote <cmd> Send QUOTE command to FTP before file transfer (F)\n"
|
|
|
|
" -r/--range <range> Retrieve a byte range from a HTTP/1.1 or FTP server\n"
|
|
|
|
" -s/--silent Silent mode. Don't output anything\n"
|
|
|
|
" -S/--show-error Show error. With -s, make curl show errors when they occur\n"
|
|
|
|
" -t/--upload Transfer/upload stdin to remote site\n"
|
|
|
|
" -T/--upload-file <file> Transfer/upload <file> to remote site\n"
|
|
|
|
" -u/--user <user:password> Specify user and password to use\n"
|
|
|
|
" -U/--proxy-user <user:password> Specify Proxy authentication\n"
|
|
|
|
" -v/--verbose Makes the operation more talkative\n"
|
|
|
|
" -V/--version Outputs version number then quits\n"
|
2000-02-16 01:05:26 +01:00
|
|
|
" -w/--write-out [format] What to output after completion\n"
|
2000-07-28 09:56:50 +02:00
|
|
|
" -x/--proxy <host[:port]> Use proxy. (Default port is 1080)\n"
|
1999-12-29 15:20:26 +01:00
|
|
|
" -X/--request <command> Specific request command to use\n"
|
2000-01-31 23:21:10 +01:00
|
|
|
" -y/--speed-time Time needed to trig speed-limit abort. Defaults to 30\n"
|
|
|
|
" -Y/--speed-limit Stop transfer if below speed-limit for 'speed-time' secs\n"
|
1999-12-29 15:20:26 +01:00
|
|
|
" -z/--time-cond <time> Includes a time condition to the server (H)\n"
|
|
|
|
" -2/--sslv2 Force usage of SSLv2 (H)\n"
|
|
|
|
" -3/--sslv3 Force usage of SSLv3 (H)\n"
|
|
|
|
" -#/--progress-bar Display transfer progress as a progress bar\n"
|
|
|
|
" --crlf Convert LF to CRLF in upload. Useful for MVS (OS/390)\n"
|
|
|
|
" --stderr <file> Where to redirect stderr. - means stdout.\n",
|
|
|
|
curl_version()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct LongShort {
|
|
|
|
char *letter;
|
|
|
|
char *lname;
|
|
|
|
bool extraparam;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Configurable {
|
|
|
|
char *useragent;
|
|
|
|
char *cookie;
|
|
|
|
bool use_resume;
|
|
|
|
int resume_from;
|
|
|
|
char *postfields;
|
2000-08-24 19:56:20 +02:00
|
|
|
long postfieldsize;
|
1999-12-29 15:20:26 +01:00
|
|
|
char *referer;
|
|
|
|
long timeout;
|
|
|
|
char *outfile;
|
|
|
|
char *headerfile;
|
|
|
|
char remotefile;
|
|
|
|
char *ftpport;
|
2000-09-21 10:52:08 +02:00
|
|
|
char *iface;
|
1999-12-29 15:20:26 +01:00
|
|
|
unsigned short porttouse;
|
|
|
|
char *range;
|
|
|
|
int low_speed_limit;
|
|
|
|
int low_speed_time;
|
|
|
|
bool showerror;
|
|
|
|
char *infile;
|
|
|
|
char *userpwd;
|
|
|
|
char *proxyuserpwd;
|
|
|
|
char *proxy;
|
|
|
|
bool configread;
|
2000-09-15 08:11:46 +02:00
|
|
|
bool proxytunnel;
|
1999-12-29 15:20:26 +01:00
|
|
|
long conf;
|
|
|
|
char *url;
|
|
|
|
char *cert;
|
|
|
|
char *cert_passwd;
|
|
|
|
bool crlf;
|
|
|
|
char *cookiefile;
|
|
|
|
char *customrequest;
|
2000-09-21 10:52:08 +02:00
|
|
|
char *krb4level;
|
1999-12-29 15:20:26 +01:00
|
|
|
bool progressmode;
|
2000-03-03 00:06:34 +01:00
|
|
|
bool nobuffer;
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2000-02-16 01:05:26 +01:00
|
|
|
char *writeout; /* %-styled format string to output */
|
|
|
|
|
1999-12-29 15:20:26 +01:00
|
|
|
FILE *errors; /* if stderr redirect is requested */
|
|
|
|
|
|
|
|
struct curl_slist *quote;
|
2000-01-11 00:36:14 +01:00
|
|
|
struct curl_slist *postquote;
|
1999-12-29 15:20:26 +01:00
|
|
|
|
|
|
|
long ssl_version;
|
|
|
|
TimeCond timecond;
|
|
|
|
time_t condtime;
|
|
|
|
|
2000-06-05 10:26:13 +02:00
|
|
|
struct curl_slist *headers;
|
1999-12-29 15:20:26 +01:00
|
|
|
|
|
|
|
struct HttpPost *httppost;
|
|
|
|
struct HttpPost *last_post;
|
2000-07-25 09:34:04 +02:00
|
|
|
|
|
|
|
HttpReq httpreq;
|
1999-12-29 15:20:26 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
static int parseconfig(char *filename,
|
|
|
|
struct Configurable *config);
|
2000-02-11 00:03:08 +01:00
|
|
|
static char *my_get_line(FILE *fp);
|
|
|
|
static char *my_get_token(const char *line);
|
1999-12-29 15:20:26 +01:00
|
|
|
|
|
|
|
static void GetStr(char **string,
|
|
|
|
char *value)
|
|
|
|
{
|
2000-11-17 15:03:58 +01:00
|
|
|
fprintf(stderr, "called\n");
|
1999-12-29 15:20:26 +01:00
|
|
|
if(*string)
|
|
|
|
free(*string);
|
2000-05-22 16:12:12 +02:00
|
|
|
if(value && *value)
|
|
|
|
*string = strdup(value);
|
|
|
|
else
|
|
|
|
*string = NULL;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static char *file2string(FILE *file)
|
|
|
|
{
|
|
|
|
char buffer[256];
|
|
|
|
char *ptr;
|
|
|
|
char *string=NULL;
|
|
|
|
int len=0;
|
|
|
|
int stringlen;
|
|
|
|
|
|
|
|
if(file) {
|
|
|
|
while(fgets(buffer, sizeof(buffer), file)) {
|
|
|
|
ptr= strchr(buffer, '\r');
|
|
|
|
if(ptr)
|
|
|
|
*ptr=0;
|
|
|
|
ptr= strchr(buffer, '\n');
|
|
|
|
if(ptr)
|
|
|
|
*ptr=0;
|
|
|
|
stringlen=strlen(buffer);
|
|
|
|
if(string)
|
|
|
|
string = realloc(string, len+stringlen+1);
|
|
|
|
else
|
|
|
|
string = malloc(stringlen+1);
|
|
|
|
|
|
|
|
strcpy(string+len, buffer);
|
|
|
|
|
|
|
|
len+=stringlen;
|
|
|
|
}
|
|
|
|
return string;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return NULL; /* no string */
|
|
|
|
}
|
|
|
|
|
2000-08-24 19:56:20 +02:00
|
|
|
static char *file2memory(FILE *file, long *size)
|
|
|
|
{
|
|
|
|
char buffer[1024];
|
|
|
|
char *ptr;
|
|
|
|
char *string=NULL;
|
|
|
|
char *newstring=NULL;
|
|
|
|
long len=0;
|
|
|
|
long stringlen=0;
|
|
|
|
|
|
|
|
if(file) {
|
|
|
|
while(len = fread(buffer, 1, sizeof(buffer), file)) {
|
|
|
|
if(string) {
|
|
|
|
newstring = realloc(string, len+stringlen);
|
|
|
|
if(newstring)
|
|
|
|
string = newstring;
|
|
|
|
else
|
|
|
|
break; /* no more strings attached! :-) */
|
|
|
|
}
|
|
|
|
else
|
|
|
|
string = malloc(len);
|
|
|
|
memcpy(&string[stringlen], buffer, len);
|
|
|
|
stringlen+=len;
|
|
|
|
}
|
|
|
|
*size = stringlen;
|
|
|
|
return string;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return NULL; /* no string */
|
|
|
|
}
|
|
|
|
|
2000-11-17 10:47:18 +01:00
|
|
|
typedef enum {
|
|
|
|
PARAM_OK,
|
|
|
|
PARAM_OPTION_AMBIGUOUS,
|
|
|
|
PARAM_OPTION_UNKNOWN,
|
|
|
|
PARAM_REQUIRES_PARAMETER,
|
|
|
|
PARAM_BAD_USE,
|
|
|
|
PARAM_HELP_REQUESTED,
|
|
|
|
PARAM_GOT_EXTRA_PARAMETER,
|
|
|
|
|
|
|
|
PARAM_LAST
|
|
|
|
} ParameterError;
|
|
|
|
|
|
|
|
static ParameterError getparameter(char *flag, /* f or -long-flag */
|
|
|
|
char *nextarg, /* NULL if unset */
|
|
|
|
bool *usedarg, /* set to TRUE if the arg
|
|
|
|
has been used */
|
|
|
|
struct Configurable *config)
|
1999-12-29 15:20:26 +01:00
|
|
|
{
|
|
|
|
char letter;
|
2000-08-24 19:56:20 +02:00
|
|
|
char subletter=0; /* subletters can only occur on long options */
|
|
|
|
|
1999-12-29 15:20:26 +01:00
|
|
|
char *parse=NULL;
|
|
|
|
int res;
|
|
|
|
int j;
|
|
|
|
time_t now;
|
|
|
|
int hit=-1;
|
2000-08-24 19:56:20 +02:00
|
|
|
bool longopt=FALSE;
|
1999-12-29 15:20:26 +01:00
|
|
|
|
|
|
|
/* single-letter,
|
|
|
|
long-name,
|
|
|
|
boolean whether it takes an additional argument
|
|
|
|
*/
|
|
|
|
struct LongShort aliases[]= {
|
|
|
|
{"9", "crlf", FALSE},
|
|
|
|
{"8", "stderr", TRUE},
|
2000-09-18 23:54:46 +02:00
|
|
|
{"7", "interface", TRUE},
|
2000-09-21 10:52:08 +02:00
|
|
|
{"6", "krb4", TRUE},
|
2000-11-17 10:47:18 +01:00
|
|
|
{"5", "url", TRUE},
|
1999-12-29 15:20:26 +01:00
|
|
|
|
|
|
|
{"2", "sslv2", FALSE},
|
|
|
|
{"3", "sslv3", FALSE},
|
|
|
|
{"a", "append", FALSE},
|
|
|
|
{"A", "user-agent", TRUE},
|
|
|
|
{"b", "cookie", TRUE},
|
2000-06-14 14:50:38 +02:00
|
|
|
{"B", "ftp-ascii", FALSE}, /* this long format is OBSOLETEE now! */
|
|
|
|
{"B", "use-ascii", FALSE},
|
1999-12-29 15:20:26 +01:00
|
|
|
{"c", "continue", FALSE},
|
|
|
|
{"C", "continue-at", TRUE},
|
|
|
|
{"d", "data", TRUE},
|
2000-08-24 19:56:20 +02:00
|
|
|
{"da", "data-ascii", TRUE},
|
|
|
|
{"db", "data-binary", TRUE},
|
1999-12-29 15:20:26 +01:00
|
|
|
{"D", "dump-header", TRUE},
|
|
|
|
{"e", "referer", TRUE},
|
|
|
|
{"E", "cert", TRUE},
|
|
|
|
{"f", "fail", FALSE},
|
|
|
|
{"F", "form", TRUE},
|
|
|
|
|
|
|
|
{"h", "help", FALSE},
|
|
|
|
{"H", "header", TRUE},
|
|
|
|
{"i", "include", FALSE},
|
|
|
|
{"I", "head", FALSE},
|
|
|
|
{"K", "config", TRUE},
|
|
|
|
{"l", "list-only", FALSE},
|
|
|
|
{"L", "location", FALSE},
|
|
|
|
{"m", "max-time", TRUE},
|
|
|
|
{"M", "manual", FALSE},
|
|
|
|
{"n", "netrc", FALSE},
|
2000-03-03 00:06:34 +01:00
|
|
|
{"N", "no-buffer", FALSE},
|
1999-12-29 15:20:26 +01:00
|
|
|
{"o", "output", TRUE},
|
|
|
|
{"O", "remote-name", FALSE},
|
2000-09-15 08:11:46 +02:00
|
|
|
{"p", "proxytunnel", FALSE},
|
1999-12-29 15:20:26 +01:00
|
|
|
{"P", "ftpport", TRUE},
|
|
|
|
{"q", "disable", FALSE},
|
|
|
|
{"Q", "quote", TRUE},
|
|
|
|
{"r", "range", TRUE},
|
|
|
|
{"s", "silent", FALSE},
|
|
|
|
{"S", "show-error", FALSE},
|
|
|
|
{"t", "upload", FALSE},
|
|
|
|
{"T", "upload-file", TRUE},
|
|
|
|
{"u", "user", TRUE},
|
|
|
|
{"U", "proxy-user", TRUE},
|
|
|
|
{"v", "verbose", FALSE},
|
|
|
|
{"V", "version", FALSE},
|
2000-02-16 01:05:26 +01:00
|
|
|
{"w", "write-out", TRUE},
|
1999-12-29 15:20:26 +01:00
|
|
|
{"x", "proxy", TRUE},
|
|
|
|
{"X", "request", TRUE},
|
|
|
|
{"X", "http-request", TRUE}, /* OBSOLETE VERSION */
|
2000-01-31 23:21:10 +01:00
|
|
|
{"Y", "speed-limit", TRUE},
|
|
|
|
{"y", "speed-time", TRUE},
|
1999-12-29 15:20:26 +01:00
|
|
|
{"z", "time-cond", TRUE},
|
|
|
|
{"#", "progress-bar",FALSE},
|
|
|
|
};
|
|
|
|
|
2000-11-17 10:47:18 +01:00
|
|
|
if(('-' != flag[0]) ||
|
|
|
|
(('-' == flag[0]) && ('-' == flag[1]))) {
|
|
|
|
/* this should be a long name */
|
|
|
|
char *word=('-' == flag[0])?flag+2:flag;
|
|
|
|
int fnam=strlen(word);
|
2000-10-06 14:45:05 +02:00
|
|
|
int numhits=0;
|
1999-12-29 15:20:26 +01:00
|
|
|
for(j=0; j< sizeof(aliases)/sizeof(aliases[0]); j++) {
|
2000-11-17 10:47:18 +01:00
|
|
|
if(strnequal(aliases[j].lname, word, fnam)) {
|
2000-08-24 19:56:20 +02:00
|
|
|
longopt = TRUE;
|
2000-10-06 14:45:05 +02:00
|
|
|
numhits++;
|
2000-11-17 10:47:18 +01:00
|
|
|
if(strequal(aliases[j].lname, word)) {
|
1999-12-29 15:20:26 +01:00
|
|
|
parse = aliases[j].letter;
|
|
|
|
hit = j;
|
2000-10-06 14:45:05 +02:00
|
|
|
numhits = 1; /* a single unique hit */
|
1999-12-29 15:20:26 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
parse = aliases[j].letter;
|
|
|
|
hit = j;
|
|
|
|
}
|
|
|
|
}
|
2000-10-06 14:45:05 +02:00
|
|
|
if(numhits>1) {
|
|
|
|
/* this is at least the second match! */
|
2000-11-17 10:47:18 +01:00
|
|
|
return PARAM_OPTION_AMBIGUOUS;
|
2000-10-06 14:45:05 +02:00
|
|
|
}
|
1999-12-29 15:20:26 +01:00
|
|
|
if(hit < 0) {
|
2000-11-17 10:47:18 +01:00
|
|
|
return PARAM_OPTION_UNKNOWN;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2000-11-17 10:47:18 +01:00
|
|
|
flag++; /* prefixed with one dash, pass it */
|
1999-12-29 15:20:26 +01:00
|
|
|
hit=-1;
|
|
|
|
parse = flag;
|
|
|
|
}
|
|
|
|
|
|
|
|
do {
|
|
|
|
/* we can loop here if we have multiple single-letters */
|
|
|
|
|
2000-08-24 19:56:20 +02:00
|
|
|
if(!longopt)
|
|
|
|
letter = parse?*parse:'\0';
|
|
|
|
else {
|
|
|
|
letter = parse[0];
|
|
|
|
subletter = parse[1];
|
|
|
|
}
|
1999-12-29 15:20:26 +01:00
|
|
|
*usedarg = FALSE; /* default is that we don't use the arg */
|
|
|
|
|
|
|
|
#if 0
|
|
|
|
fprintf(stderr, "OPTION: %c %s\n", letter, nextarg?nextarg:"<null>");
|
|
|
|
#endif
|
|
|
|
if(hit < 0) {
|
|
|
|
for(j=0; j< sizeof(aliases)/sizeof(aliases[0]); j++) {
|
2000-08-24 19:56:20 +02:00
|
|
|
if(letter == aliases[j].letter[0]) {
|
1999-12-29 15:20:26 +01:00
|
|
|
hit = j;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(hit < 0) {
|
2000-11-17 10:47:18 +01:00
|
|
|
return PARAM_OPTION_UNKNOWN;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if(hit < 0) {
|
2000-11-17 10:47:18 +01:00
|
|
|
return PARAM_OPTION_UNKNOWN;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
2000-11-17 10:47:18 +01:00
|
|
|
if((!nextarg || !*nextarg) && aliases[hit].extraparam) {
|
|
|
|
return PARAM_REQUIRES_PARAMETER;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
|
|
|
else if(nextarg && aliases[hit].extraparam)
|
|
|
|
*usedarg = TRUE; /* mark it as used */
|
|
|
|
|
|
|
|
switch(letter) {
|
|
|
|
case 'z': /* time condition coming up */
|
|
|
|
switch(*nextarg) {
|
|
|
|
case '+':
|
|
|
|
nextarg++;
|
|
|
|
default:
|
|
|
|
/* If-Modified-Since: (section 14.28 in RFC2068) */
|
|
|
|
config->timecond = TIMECOND_IFMODSINCE;
|
|
|
|
break;
|
|
|
|
case '-':
|
|
|
|
/* If-Unmodified-Since: (section 14.24 in RFC2068) */
|
|
|
|
config->timecond = TIMECOND_IFUNMODSINCE;
|
|
|
|
nextarg++;
|
|
|
|
break;
|
|
|
|
case '=':
|
|
|
|
/* Last-Modified: (section 14.29 in RFC2068) */
|
|
|
|
config->timecond = TIMECOND_LASTMOD;
|
|
|
|
nextarg++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
now=time(NULL);
|
2000-05-22 16:12:12 +02:00
|
|
|
config->condtime=curl_getdate(nextarg, &now);
|
1999-12-29 15:20:26 +01:00
|
|
|
if(-1 == config->condtime) {
|
|
|
|
/* now let's see if it is a file name to get the time from instead! */
|
|
|
|
struct stat statbuf;
|
|
|
|
if(-1 == stat(nextarg, &statbuf)) {
|
|
|
|
/* failed, remove time condition */
|
|
|
|
config->timecond = TIMECOND_NONE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* pull the time out from the file */
|
|
|
|
config->condtime = statbuf.st_mtime;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case '9': /* there is no short letter for this */
|
|
|
|
/* LF -> CRLF conversinon? */
|
|
|
|
config->crlf = TRUE;
|
|
|
|
break;
|
|
|
|
case '8': /* there is no short letter for this */
|
|
|
|
if(strcmp(nextarg, "-"))
|
|
|
|
config->errors = fopen(nextarg, "wt");
|
|
|
|
else
|
|
|
|
config->errors = stdout;
|
|
|
|
break;
|
2000-09-18 23:54:46 +02:00
|
|
|
case '7': /* there is no short letter for this */
|
|
|
|
/* interface */
|
2000-09-21 10:52:08 +02:00
|
|
|
GetStr(&config->iface, nextarg);
|
|
|
|
break;
|
|
|
|
case '6': /* there is no short letter for this */
|
|
|
|
/* krb4 level string */
|
|
|
|
GetStr(&config->krb4level, nextarg);
|
|
|
|
break;
|
2000-11-17 10:47:18 +01:00
|
|
|
case '5':
|
|
|
|
/* the URL! */
|
|
|
|
GetStr(&config->url, nextarg);
|
|
|
|
break;
|
1999-12-29 15:20:26 +01:00
|
|
|
case '#': /* added 19990617 larsa */
|
|
|
|
config->progressmode ^= CURL_PROGRESS_BAR;
|
|
|
|
break;
|
|
|
|
case '2':
|
|
|
|
/* SSL version 2 */
|
|
|
|
config->ssl_version = 2;
|
|
|
|
break;
|
|
|
|
case '3':
|
|
|
|
/* SSL version 2 */
|
|
|
|
config->ssl_version = 3;
|
|
|
|
break;
|
|
|
|
case 'a':
|
|
|
|
/* This makes the FTP sessions use APPE instead of STOR */
|
|
|
|
config->conf ^= CONF_FTPAPPEND;
|
|
|
|
break;
|
|
|
|
case 'A':
|
|
|
|
/* This specifies the User-Agent name */
|
|
|
|
GetStr(&config->useragent, nextarg);
|
|
|
|
break;
|
|
|
|
case 'b': /* cookie string coming up: */
|
2000-02-15 00:17:59 +01:00
|
|
|
if(nextarg[0] == '@') {
|
|
|
|
nextarg++;
|
|
|
|
}
|
|
|
|
else if(strchr(nextarg, '=')) {
|
1999-12-29 15:20:26 +01:00
|
|
|
/* A cookie string must have a =-letter */
|
|
|
|
GetStr(&config->cookie, nextarg);
|
2000-02-15 00:17:59 +01:00
|
|
|
break;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
2000-02-15 00:17:59 +01:00
|
|
|
/* We have a cookie file to read from! */
|
|
|
|
GetStr(&config->cookiefile, nextarg);
|
1999-12-29 15:20:26 +01:00
|
|
|
break;
|
|
|
|
case 'B':
|
2000-06-14 14:50:38 +02:00
|
|
|
/* use ASCII/text when transfering */
|
|
|
|
config->conf ^= CONF_GETTEXT;
|
1999-12-29 15:20:26 +01:00
|
|
|
break;
|
|
|
|
case 'c':
|
|
|
|
/* This makes us continue an ftp transfer */
|
|
|
|
config->use_resume^=TRUE;
|
2000-09-26 00:20:02 +02:00
|
|
|
fprintf(stderr, "-c is a deprecated switch, use '-C -' instead!\n");
|
1999-12-29 15:20:26 +01:00
|
|
|
break;
|
|
|
|
case 'C':
|
|
|
|
/* This makes us continue an ftp transfer at given position */
|
2000-09-26 00:20:02 +02:00
|
|
|
if(!strequal(nextarg, "-"))
|
|
|
|
config->resume_from= atoi(nextarg);
|
1999-12-29 15:20:26 +01:00
|
|
|
config->use_resume=TRUE;
|
|
|
|
break;
|
|
|
|
case 'd':
|
|
|
|
/* postfield data */
|
2000-10-26 09:06:52 +02:00
|
|
|
{
|
|
|
|
char *postdata=NULL;
|
|
|
|
|
|
|
|
if('@' == *nextarg) {
|
|
|
|
/* the data begins with a '@' letter, it means that a file name
|
|
|
|
or - (stdin) follows */
|
|
|
|
FILE *file;
|
|
|
|
char *ptr;
|
|
|
|
|
|
|
|
nextarg++; /* pass the @ */
|
|
|
|
|
|
|
|
if(strequal("-", nextarg))
|
|
|
|
file = stdin;
|
|
|
|
else
|
|
|
|
file = fopen(nextarg, "r");
|
|
|
|
|
|
|
|
if(subletter == 'b') /* forced binary */
|
|
|
|
postdata = file2memory(file, &config->postfieldsize);
|
|
|
|
else
|
|
|
|
postdata = file2string(file);
|
|
|
|
if(file && (file != stdin))
|
|
|
|
fclose(stdin);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
GetStr(&postdata, nextarg);
|
|
|
|
}
|
2000-08-24 19:56:20 +02:00
|
|
|
|
2000-10-26 09:06:52 +02:00
|
|
|
if(config->postfields && *config->postfields) {
|
|
|
|
/* we already have a string, we append this one
|
|
|
|
with a separating &-letter */
|
|
|
|
char *oldpost=config->postfields;
|
|
|
|
config->postfields=maprintf("%s&%s", oldpost, postdata);
|
|
|
|
free(oldpost);
|
|
|
|
free(postdata);
|
|
|
|
}
|
2000-08-24 19:56:20 +02:00
|
|
|
else
|
2000-10-26 09:06:52 +02:00
|
|
|
config->postfields=postdata;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
|
|
|
if(config->postfields)
|
|
|
|
config->conf |= CONF_POST;
|
2000-07-25 09:34:04 +02:00
|
|
|
if(SetHTTPrequest(HTTPREQ_SIMPLEPOST, &config->httpreq))
|
2000-11-17 10:47:18 +01:00
|
|
|
return PARAM_BAD_USE;
|
1999-12-29 15:20:26 +01:00
|
|
|
break;
|
|
|
|
case 'D':
|
|
|
|
/* dump-header to given file name */
|
|
|
|
GetStr(&config->headerfile, nextarg);
|
|
|
|
break;
|
|
|
|
case 'e':
|
2000-06-20 11:28:09 +02:00
|
|
|
{
|
|
|
|
char *ptr = strstr(nextarg, ";auto");
|
|
|
|
if(ptr) {
|
|
|
|
/* Automatic referer requested, this may be combined with a
|
|
|
|
set initial one */
|
|
|
|
config->conf |= CONF_AUTO_REFERER;
|
|
|
|
*ptr = 0; /* zero terminate here */
|
|
|
|
}
|
|
|
|
GetStr(&config->referer, nextarg);
|
|
|
|
}
|
1999-12-29 15:20:26 +01:00
|
|
|
break;
|
|
|
|
case 'E':
|
|
|
|
{
|
|
|
|
char *ptr = strchr(nextarg, ':');
|
|
|
|
if(ptr) {
|
|
|
|
/* we have a password too */
|
|
|
|
*ptr=0;
|
|
|
|
ptr++;
|
|
|
|
GetStr(&config->cert_passwd, ptr);
|
|
|
|
}
|
|
|
|
GetStr(&config->cert, nextarg);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'f':
|
|
|
|
/* fail hard on errors */
|
|
|
|
config->conf ^= CONF_FAILONERROR;
|
|
|
|
break;
|
|
|
|
case 'F':
|
|
|
|
/* "form data" simulation, this is a little advanced so lets do our best
|
|
|
|
to sort this out slowly and carefully */
|
2000-06-05 10:26:13 +02:00
|
|
|
if(curl_formparse(nextarg,
|
1999-12-29 15:20:26 +01:00
|
|
|
&config->httppost,
|
|
|
|
&config->last_post))
|
2000-11-17 10:47:18 +01:00
|
|
|
return PARAM_BAD_USE;
|
2000-07-25 09:34:04 +02:00
|
|
|
if(SetHTTPrequest(HTTPREQ_POST, &config->httpreq))
|
2000-11-17 10:47:18 +01:00
|
|
|
return PARAM_BAD_USE;
|
1999-12-29 15:20:26 +01:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'h': /* h for help */
|
|
|
|
help();
|
2000-11-17 10:47:18 +01:00
|
|
|
return PARAM_HELP_REQUESTED;
|
1999-12-29 15:20:26 +01:00
|
|
|
case 'H':
|
2000-06-05 10:26:13 +02:00
|
|
|
/* A custom header to append to a list */
|
|
|
|
config->headers = curl_slist_append(config->headers, nextarg);
|
1999-12-29 15:20:26 +01:00
|
|
|
break;
|
|
|
|
case 'i':
|
|
|
|
config->conf ^= CONF_HEADER; /* include the HTTP header as well */
|
|
|
|
break;
|
|
|
|
case 'I':
|
|
|
|
config->conf ^= CONF_HEADER; /* include the HTTP header in the output */
|
|
|
|
config->conf ^= CONF_NOBODY; /* don't fetch the body at all */
|
2000-07-25 09:34:04 +02:00
|
|
|
if(SetHTTPrequest(HTTPREQ_HEAD, &config->httpreq))
|
2000-11-17 10:47:18 +01:00
|
|
|
return PARAM_BAD_USE;
|
1999-12-29 15:20:26 +01:00
|
|
|
break;
|
|
|
|
case 'K':
|
|
|
|
res = parseconfig(nextarg, config);
|
|
|
|
config->configread = TRUE;
|
|
|
|
if(res)
|
|
|
|
return res;
|
|
|
|
break;
|
|
|
|
case 'l':
|
|
|
|
config->conf ^= CONF_FTPLISTONLY; /* only list the names of the FTP dir */
|
|
|
|
break;
|
|
|
|
case 'L':
|
|
|
|
config->conf ^= CONF_FOLLOWLOCATION; /* Follow Location: HTTP headers */
|
|
|
|
break;
|
|
|
|
case 'm':
|
|
|
|
/* specified max time */
|
|
|
|
config->timeout = atoi(nextarg);
|
|
|
|
break;
|
|
|
|
case 'M': /* M for manual, huge help */
|
|
|
|
hugehelp();
|
2000-11-17 10:47:18 +01:00
|
|
|
return PARAM_HELP_REQUESTED;
|
1999-12-29 15:20:26 +01:00
|
|
|
case 'n':
|
|
|
|
/* pick info from .netrc, if this is used for http, curl will
|
|
|
|
automatically enfore user+password with the request */
|
|
|
|
config->conf ^= CONF_NETRC;
|
|
|
|
break;
|
2000-03-03 00:06:34 +01:00
|
|
|
case 'N':
|
|
|
|
/* disable the output I/O buffering */
|
|
|
|
config->nobuffer ^= 1;
|
|
|
|
break;
|
1999-12-29 15:20:26 +01:00
|
|
|
case 'o':
|
|
|
|
/* output file */
|
|
|
|
GetStr(&config->outfile, nextarg); /* write to this file */
|
|
|
|
break;
|
|
|
|
case 'O':
|
|
|
|
/* output file */
|
|
|
|
config->remotefile ^= TRUE;
|
|
|
|
break;
|
|
|
|
case 'P':
|
|
|
|
/* This makes the FTP sessions use PORT instead of PASV */
|
|
|
|
/* use <eth0> or <192.168.10.10> style addresses. Anything except
|
|
|
|
this will make us try to get the "default" address.
|
|
|
|
NOTE: this is a changed behaviour since the released 4.1!
|
|
|
|
*/
|
|
|
|
GetStr(&config->ftpport, nextarg);
|
|
|
|
break;
|
|
|
|
case 'p':
|
2000-09-15 08:11:46 +02:00
|
|
|
/* proxy tunnel for non-http protocols */
|
|
|
|
config->proxytunnel ^= TRUE;
|
1999-12-29 15:20:26 +01:00
|
|
|
break;
|
2000-09-15 08:11:46 +02:00
|
|
|
|
1999-12-29 15:20:26 +01:00
|
|
|
case 'q': /* if used first, already taken care of, we do it like
|
|
|
|
this so we don't cause an error! */
|
|
|
|
break;
|
|
|
|
case 'Q':
|
|
|
|
/* QUOTE command to send to FTP server */
|
2000-01-11 00:36:14 +01:00
|
|
|
if(nextarg[0] == '-') {
|
|
|
|
/* prefixed with a dash makes it a POST TRANSFER one */
|
|
|
|
nextarg++;
|
|
|
|
config->postquote = curl_slist_append(config->postquote, nextarg);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
config->quote = curl_slist_append(config->quote, nextarg);
|
|
|
|
}
|
1999-12-29 15:20:26 +01:00
|
|
|
break;
|
|
|
|
case 'r':
|
|
|
|
/* byte range requested */
|
|
|
|
GetStr(&config->range, nextarg);
|
|
|
|
break;
|
|
|
|
case 's':
|
|
|
|
/* don't show progress meter, don't show errors : */
|
|
|
|
config->conf |= (CONF_MUTE|CONF_NOPROGRESS);
|
|
|
|
config->showerror ^= TRUE; /* toggle off */
|
|
|
|
break;
|
|
|
|
case 'S':
|
|
|
|
/* show errors */
|
|
|
|
config->showerror ^= TRUE; /* toggle on if used with -s */
|
|
|
|
break;
|
|
|
|
case 't':
|
|
|
|
/* we are uploading */
|
|
|
|
config->conf ^= CONF_UPLOAD;
|
2000-09-26 00:20:02 +02:00
|
|
|
fprintf(stderr, "-t is a deprecated switch, use '-T -' instead!\n");
|
1999-12-29 15:20:26 +01:00
|
|
|
break;
|
|
|
|
case 'T':
|
|
|
|
/* we are uploading */
|
|
|
|
config->conf |= CONF_UPLOAD;
|
2000-09-26 00:20:02 +02:00
|
|
|
if(!strequal("-", nextarg))
|
|
|
|
/* make - equal stdin */
|
|
|
|
GetStr(&config->infile, nextarg);
|
1999-12-29 15:20:26 +01:00
|
|
|
break;
|
|
|
|
case 'u':
|
|
|
|
/* user:password */
|
|
|
|
GetStr(&config->userpwd, nextarg);
|
|
|
|
break;
|
|
|
|
case 'U':
|
|
|
|
/* Proxy user:password */
|
|
|
|
GetStr(&config->proxyuserpwd, nextarg);
|
|
|
|
break;
|
|
|
|
case 'v':
|
|
|
|
config->conf ^= CONF_VERBOSE; /* talk a lot */
|
|
|
|
break;
|
|
|
|
case 'V':
|
|
|
|
printf(CURL_ID "%s\n", curl_version());
|
2000-11-17 10:47:18 +01:00
|
|
|
return PARAM_HELP_REQUESTED;
|
2000-02-16 01:05:26 +01:00
|
|
|
case 'w':
|
|
|
|
/* get the output string */
|
2000-03-01 23:44:46 +01:00
|
|
|
if('@' == *nextarg) {
|
|
|
|
/* the data begins with a '@' letter, it means that a file name
|
|
|
|
or - (stdin) follows */
|
|
|
|
FILE *file;
|
|
|
|
nextarg++; /* pass the @ */
|
|
|
|
if(strequal("-", nextarg))
|
|
|
|
file = stdin;
|
|
|
|
else
|
|
|
|
file = fopen(nextarg, "r");
|
|
|
|
config->writeout = file2string(file);
|
|
|
|
if(file && (file != stdin))
|
|
|
|
fclose(stdin);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
GetStr(&config->writeout, nextarg);
|
2000-02-16 01:05:26 +01:00
|
|
|
break;
|
1999-12-29 15:20:26 +01:00
|
|
|
case 'x':
|
|
|
|
/* proxy */
|
2000-05-22 16:12:12 +02:00
|
|
|
GetStr(&config->proxy, nextarg);
|
1999-12-29 15:20:26 +01:00
|
|
|
break;
|
|
|
|
case 'X':
|
|
|
|
/* HTTP request */
|
|
|
|
GetStr(&config->customrequest, nextarg);
|
2000-07-25 09:34:04 +02:00
|
|
|
if(SetHTTPrequest(HTTPREQ_CUSTOM, &config->httpreq))
|
2000-11-17 10:47:18 +01:00
|
|
|
return PARAM_BAD_USE;
|
1999-12-29 15:20:26 +01:00
|
|
|
break;
|
2000-01-31 23:21:10 +01:00
|
|
|
case 'y':
|
1999-12-29 15:20:26 +01:00
|
|
|
/* low speed time */
|
|
|
|
config->low_speed_time = atoi(nextarg);
|
|
|
|
if(!config->low_speed_limit)
|
|
|
|
config->low_speed_limit = 1;
|
|
|
|
break;
|
2000-01-31 23:21:10 +01:00
|
|
|
case 'Y':
|
1999-12-29 15:20:26 +01:00
|
|
|
/* low speed limit */
|
|
|
|
config->low_speed_limit = atoi(nextarg);
|
|
|
|
if(!config->low_speed_time)
|
|
|
|
config->low_speed_time=30;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default: /* unknown flag */
|
2000-11-17 10:47:18 +01:00
|
|
|
return PARAM_OPTION_UNKNOWN;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
|
|
|
hit = -1;
|
|
|
|
|
|
|
|
} while(*++parse && !*usedarg);
|
|
|
|
|
2000-11-17 10:47:18 +01:00
|
|
|
return PARAM_OK;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int parseconfig(char *filename,
|
|
|
|
struct Configurable *config)
|
|
|
|
{
|
|
|
|
int res;
|
|
|
|
FILE *file;
|
|
|
|
char configbuffer[4096];
|
|
|
|
char filebuffer[256];
|
|
|
|
bool usedarg;
|
2000-05-30 01:09:31 +02:00
|
|
|
char *home=NULL;
|
1999-12-29 15:20:26 +01:00
|
|
|
|
|
|
|
if(!filename || !*filename) {
|
|
|
|
/* NULL or no file name attempts to load .curlrc from the homedir! */
|
|
|
|
|
|
|
|
#define CURLRC DOT_CHAR "curlrc"
|
|
|
|
|
2000-06-05 10:26:13 +02:00
|
|
|
home = curl_getenv("HOME"); /* portable environment reader */
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2000-05-30 01:09:31 +02:00
|
|
|
if(!home)
|
2000-11-17 10:47:18 +01:00
|
|
|
return 0;
|
2000-05-30 01:09:31 +02:00
|
|
|
if(strlen(home)>(sizeof(filebuffer)-strlen(CURLRC))) {
|
|
|
|
free(home);
|
2000-11-17 10:47:18 +01:00
|
|
|
return 0;
|
2000-05-30 01:09:31 +02:00
|
|
|
}
|
1999-12-29 15:20:26 +01:00
|
|
|
|
|
|
|
sprintf(filebuffer, "%s%s%s", home, DIR_CHAR, CURLRC);
|
|
|
|
|
|
|
|
filename = filebuffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(strcmp(filename,"-"))
|
|
|
|
file = fopen(filename, "r");
|
|
|
|
else
|
|
|
|
file = stdin;
|
|
|
|
|
2000-10-26 10:15:13 +02:00
|
|
|
if(file) {
|
2000-02-11 00:03:08 +01:00
|
|
|
char *line;
|
2000-11-17 10:47:18 +01:00
|
|
|
char *aline;
|
|
|
|
char *option;
|
|
|
|
char *param;
|
|
|
|
int lineno=0;
|
|
|
|
bool alloced_param;
|
|
|
|
|
|
|
|
#define isseparator(x) (((x)=='=') || ((x) == ':'))
|
|
|
|
|
|
|
|
while (NULL != (aline = my_get_line(file))) {
|
|
|
|
lineno++;
|
|
|
|
line = aline;
|
|
|
|
alloced_param=FALSE;
|
|
|
|
|
2000-02-11 00:03:08 +01:00
|
|
|
/* lines with # in the fist column is a comment! */
|
2000-11-17 10:47:18 +01:00
|
|
|
while(isspace(*line))
|
|
|
|
line++;
|
|
|
|
|
|
|
|
switch(*line) {
|
|
|
|
case '#':
|
|
|
|
case '/':
|
|
|
|
case '\r':
|
|
|
|
case '\n':
|
|
|
|
case '*':
|
|
|
|
case '\0':
|
2000-02-11 00:03:08 +01:00
|
|
|
free(line);
|
|
|
|
continue;
|
|
|
|
}
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2000-11-17 10:47:18 +01:00
|
|
|
/* the option keywords starts here */
|
|
|
|
option = line;
|
|
|
|
while(*line && !isspace(*line) && !isseparator(*line))
|
|
|
|
line++;
|
|
|
|
/* ... and has ended here */
|
|
|
|
|
|
|
|
*line++=0; /* zero terminate, we have a local copy of the data */
|
|
|
|
|
|
|
|
#ifdef DEBUG_CONFIG
|
|
|
|
fprintf(stderr, "GOT: %s\n", option);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* pass spaces and separator(s) */
|
|
|
|
while(isspace(*line) || isseparator(*line))
|
|
|
|
line++;
|
|
|
|
|
|
|
|
/* the parameter starts here (unless quoted) */
|
|
|
|
if(*line == '\"') {
|
|
|
|
char *ptr;
|
|
|
|
/* quoted parameter, do the qoute dance */
|
|
|
|
line++;
|
|
|
|
param=strdup(line); /* parameter */
|
|
|
|
alloced_param=TRUE;
|
|
|
|
|
|
|
|
ptr=param;
|
|
|
|
while(*line && (*line != '\"')) {
|
|
|
|
if(*line == '\\') {
|
2000-11-17 11:08:39 +01:00
|
|
|
char out;
|
2000-11-17 10:47:18 +01:00
|
|
|
line++;
|
2000-11-17 11:08:39 +01:00
|
|
|
|
|
|
|
/* default is to output the letter after the backslah */
|
|
|
|
switch(out = *line) {
|
|
|
|
case '\0':
|
|
|
|
continue; /* this'll break out of the loop */
|
|
|
|
case 't':
|
|
|
|
out='\t';
|
|
|
|
break;
|
|
|
|
case 'n':
|
|
|
|
out='\n';
|
|
|
|
break;
|
|
|
|
case 'r':
|
|
|
|
out='\r';
|
|
|
|
break;
|
|
|
|
case 'v':
|
|
|
|
out='\v';
|
2000-11-17 10:47:18 +01:00
|
|
|
break;
|
|
|
|
}
|
2000-11-17 11:08:39 +01:00
|
|
|
*ptr++=out;
|
|
|
|
line++;
|
2000-11-17 10:47:18 +01:00
|
|
|
}
|
2000-11-17 11:08:39 +01:00
|
|
|
else
|
|
|
|
*ptr++=*line++;
|
2000-11-17 10:47:18 +01:00
|
|
|
}
|
|
|
|
*ptr=0; /* always zero terminate */
|
|
|
|
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
2000-11-17 10:47:18 +01:00
|
|
|
else {
|
|
|
|
param=line; /* parameter starts here */
|
|
|
|
while(*line && !isspace(*line))
|
|
|
|
line++;
|
|
|
|
*line=0; /* zero terminate */
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
2000-11-17 10:47:18 +01:00
|
|
|
#ifdef DEBUG_CONFIG
|
|
|
|
fprintf(stderr, "PARAM: \"%s\"\n", param);
|
|
|
|
#endif
|
|
|
|
res = getparameter(option, param, &usedarg, config);
|
2000-02-11 00:03:08 +01:00
|
|
|
|
2000-11-17 10:47:18 +01:00
|
|
|
if(*param && !usedarg)
|
|
|
|
/* we passed in a parameter that wasn't used! */
|
|
|
|
res = PARAM_GOT_EXTRA_PARAMETER;
|
2000-02-11 00:03:08 +01:00
|
|
|
|
2000-11-17 10:47:18 +01:00
|
|
|
if(res != PARAM_OK) {
|
|
|
|
/* the help request isn't really an error */
|
|
|
|
if(!strcmp(filename, "-")) {
|
|
|
|
filename="<stdin>";
|
|
|
|
}
|
|
|
|
if(PARAM_HELP_REQUESTED != res) {
|
|
|
|
char *reason;
|
|
|
|
switch(res) {
|
|
|
|
default:
|
|
|
|
case PARAM_GOT_EXTRA_PARAMETER:
|
|
|
|
reason = "had unsupported trailing garbage";
|
|
|
|
break;
|
|
|
|
case PARAM_OPTION_UNKNOWN:
|
|
|
|
reason = "is unknown";
|
|
|
|
break;
|
|
|
|
case PARAM_OPTION_AMBIGUOUS:
|
|
|
|
reason = "is ambiguous";
|
|
|
|
break;
|
|
|
|
case PARAM_REQUIRES_PARAMETER:
|
|
|
|
reason = "requires parameter";
|
|
|
|
break;
|
|
|
|
case PARAM_BAD_USE:
|
|
|
|
reason = "is badly used here";
|
2000-10-26 10:15:13 +02:00
|
|
|
break;
|
|
|
|
}
|
2000-11-17 10:47:18 +01:00
|
|
|
fprintf(stderr, "%s:%d: warning: '%s' %s\n",
|
|
|
|
filename, lineno, option, reason);
|
2000-10-26 10:15:13 +02:00
|
|
|
}
|
2000-02-11 00:03:08 +01:00
|
|
|
}
|
|
|
|
|
2000-11-17 10:47:18 +01:00
|
|
|
if(alloced_param)
|
|
|
|
free(param);
|
|
|
|
|
|
|
|
free(aline);
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
|
|
|
if(file != stdin)
|
|
|
|
fclose(file);
|
|
|
|
}
|
2000-05-30 01:09:31 +02:00
|
|
|
if(home)
|
|
|
|
free(home);
|
2000-11-17 10:47:18 +01:00
|
|
|
return 0;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
struct OutStruct {
|
|
|
|
char *filename;
|
|
|
|
FILE *stream;
|
|
|
|
};
|
|
|
|
|
2000-03-03 00:06:34 +01:00
|
|
|
/* having this global is a bit dirty, but hey, who said we weren't? ;-) */
|
|
|
|
struct Configurable config;
|
|
|
|
|
1999-12-29 15:20:26 +01:00
|
|
|
int my_fwrite(void *buffer, size_t size, size_t nmemb, FILE *stream)
|
|
|
|
{
|
|
|
|
struct OutStruct *out=(struct OutStruct *)stream;
|
|
|
|
if(out && !out->stream) {
|
|
|
|
/* open file for writing */
|
|
|
|
out->stream=fopen(out->filename, "wb");
|
|
|
|
if(!out->stream)
|
|
|
|
return -1; /* failure */
|
2000-03-03 00:06:34 +01:00
|
|
|
if(config.nobuffer) {
|
|
|
|
/* disable output buffering */
|
|
|
|
#ifdef HAVE_SETVBUF
|
|
|
|
setvbuf(out->stream, NULL, _IONBF, 0);
|
|
|
|
#endif
|
|
|
|
}
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
|
|
|
return fwrite(buffer, size, nmemb, out->stream);
|
|
|
|
}
|
|
|
|
|
2000-06-16 15:18:49 +02:00
|
|
|
struct ProgressData {
|
|
|
|
size_t total;
|
|
|
|
size_t prev;
|
|
|
|
size_t point;
|
|
|
|
int width;
|
|
|
|
};
|
|
|
|
|
|
|
|
int myprogress (void *clientp,
|
|
|
|
size_t dltotal,
|
|
|
|
size_t dlnow,
|
|
|
|
size_t ultotal,
|
|
|
|
size_t ulnow)
|
|
|
|
{
|
|
|
|
/* The original progress-bar source code was written for curl by Lars Aas,
|
|
|
|
and this new edition inherites some of his concepts. */
|
|
|
|
|
|
|
|
char line[256];
|
|
|
|
char outline[256];
|
|
|
|
char format[40];
|
|
|
|
float frac;
|
|
|
|
float percent;
|
|
|
|
int barwidth;
|
|
|
|
int num;
|
|
|
|
int i;
|
|
|
|
int prevblock;
|
|
|
|
int thisblock;
|
|
|
|
|
|
|
|
struct ProgressData *bar = (struct ProgressData *)clientp;
|
|
|
|
size_t total = dltotal + ultotal;
|
|
|
|
|
|
|
|
bar->point = dlnow + ulnow; /* we've come this far */
|
|
|
|
|
|
|
|
if(0 == total) {
|
|
|
|
int prevblock = bar->prev / 1024;
|
|
|
|
int thisblock = bar->point / 1024;
|
|
|
|
while ( thisblock > prevblock ) {
|
|
|
|
fprintf( stderr, "#" );
|
|
|
|
prevblock++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
frac = (float) bar->point / (float) total;
|
|
|
|
percent = frac * 100.0f;
|
|
|
|
barwidth = bar->width - 7;
|
|
|
|
num = (int) (((float)barwidth) * frac);
|
|
|
|
i = 0;
|
|
|
|
for ( i = 0; i < num; i++ ) {
|
|
|
|
line[i] = '#';
|
|
|
|
}
|
|
|
|
line[i] = '\0';
|
|
|
|
sprintf( format, "%%-%ds %%5.1f%%%%", barwidth );
|
|
|
|
sprintf( outline, format, line, percent );
|
|
|
|
fprintf( stderr, "\r%s", outline );
|
|
|
|
}
|
|
|
|
bar->prev = bar->point;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void progressbarinit(struct ProgressData *bar)
|
|
|
|
{
|
|
|
|
#ifdef __EMX__
|
|
|
|
/* 20000318 mgs */
|
|
|
|
int scr_size [2];
|
|
|
|
#endif
|
|
|
|
char *colp;
|
|
|
|
|
|
|
|
memset(bar, 0, sizeof(struct ProgressData));
|
|
|
|
|
|
|
|
/* TODO: get terminal width through ansi escapes or something similar.
|
|
|
|
try to update width when xterm is resized... - 19990617 larsa */
|
|
|
|
#ifndef __EMX__
|
|
|
|
/* 20000318 mgs
|
|
|
|
* OS/2 users most likely won't have this env var set, and besides that
|
|
|
|
* we're using our own way to determine screen width */
|
|
|
|
colp = curl_getenv("COLUMNS");
|
|
|
|
if (colp != NULL) {
|
|
|
|
bar->width = atoi(colp);
|
|
|
|
free(colp);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
bar->width = 79;
|
|
|
|
#else
|
|
|
|
/* 20000318 mgs
|
|
|
|
* We use this emx library call to get the screen width, and subtract
|
|
|
|
* one from what we got in order to avoid a problem with the cursor
|
|
|
|
* advancing to the next line if we print a string that is as long as
|
|
|
|
* the screen is wide. */
|
|
|
|
|
|
|
|
_scrsize(scr_size);
|
|
|
|
bar->width = scr_size[0] - 1;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
}
|
1999-12-29 15:20:26 +01:00
|
|
|
|
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
2000-05-26 15:58:10 +02:00
|
|
|
char errorbuffer[CURL_ERROR_SIZE];
|
2000-10-09 13:13:17 +02:00
|
|
|
char useragent[128]; /* buah, we don't want a larger default user agent */
|
2000-06-16 15:18:49 +02:00
|
|
|
struct ProgressData progressbar;
|
1999-12-29 15:20:26 +01:00
|
|
|
|
|
|
|
struct OutStruct outs;
|
2000-03-16 12:43:10 +01:00
|
|
|
struct OutStruct heads;
|
1999-12-29 15:20:26 +01:00
|
|
|
|
|
|
|
char *url = NULL;
|
2000-05-10 00:42:53 +02:00
|
|
|
|
1999-12-29 15:20:26 +01:00
|
|
|
URLGlob *urls;
|
|
|
|
int urlnum;
|
2000-06-14 16:26:53 +02:00
|
|
|
char *outfiles;
|
1999-12-29 15:20:26 +01:00
|
|
|
int separator = 0;
|
|
|
|
|
|
|
|
FILE *infd = stdin;
|
|
|
|
FILE *headerfilep = NULL;
|
|
|
|
char *urlbuffer=NULL;
|
|
|
|
int infilesize=-1; /* -1 means unknown */
|
|
|
|
bool stillflags=TRUE;
|
|
|
|
|
2000-11-17 15:03:58 +01:00
|
|
|
bool allocuseragent=FALSE;
|
|
|
|
|
2000-05-22 16:12:12 +02:00
|
|
|
CURL *curl;
|
2000-06-14 16:26:53 +02:00
|
|
|
int res;
|
1999-12-29 15:20:26 +01:00
|
|
|
int i;
|
|
|
|
|
|
|
|
outs.stream = stdout;
|
|
|
|
|
2000-10-09 13:13:17 +02:00
|
|
|
#ifdef MALLOCDEBUG
|
|
|
|
/* this sends all memory debug messages to a logfile named memdump */
|
|
|
|
curl_memdebug("memdump");
|
|
|
|
#endif
|
|
|
|
|
1999-12-29 15:20:26 +01:00
|
|
|
memset(&config, 0, sizeof(struct Configurable));
|
|
|
|
|
|
|
|
config.showerror=TRUE;
|
|
|
|
config.conf=CONF_DEFAULT;
|
2000-01-11 00:36:14 +01:00
|
|
|
#if 0
|
1999-12-29 15:20:26 +01:00
|
|
|
config.crlf=FALSE;
|
|
|
|
config.quote=NULL;
|
2000-01-11 00:36:14 +01:00
|
|
|
#endif
|
1999-12-29 15:20:26 +01:00
|
|
|
|
|
|
|
if(argc>1 &&
|
|
|
|
(!strnequal("--", argv[1], 2) && (argv[1][0] == '-')) &&
|
|
|
|
strchr(argv[1], 'q')) {
|
|
|
|
/*
|
|
|
|
* The first flag, that is not a verbose name, but a shortname
|
|
|
|
* and it includes the 'q' flag!
|
|
|
|
*/
|
|
|
|
#if 0
|
|
|
|
fprintf(stderr, "I TURNED OFF THE CRAP\n");
|
|
|
|
#endif
|
|
|
|
;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
res = parseconfig(NULL, &config);
|
|
|
|
if(res)
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((argc < 2) && !config.url) {
|
|
|
|
helpf(NULL);
|
2000-05-22 16:12:12 +02:00
|
|
|
return CURLE_FAILED_INIT;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Parse options */
|
|
|
|
for (i = 1; i < argc; i++) {
|
|
|
|
if(stillflags &&
|
|
|
|
('-' == argv[i][0])) {
|
|
|
|
char *nextarg;
|
|
|
|
bool passarg;
|
2000-11-17 10:47:18 +01:00
|
|
|
char *origopt=argv[i];
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2000-11-17 10:47:18 +01:00
|
|
|
char *flag = argv[i];
|
1999-12-29 15:20:26 +01:00
|
|
|
|
|
|
|
if(strequal("--", argv[i]))
|
|
|
|
/* this indicates the end of the flags and thus enables the
|
|
|
|
following (URL) argument to start with -. */
|
|
|
|
stillflags=FALSE;
|
|
|
|
else {
|
|
|
|
nextarg= (i < argc - 1)? argv[i+1]: NULL;
|
|
|
|
|
2000-11-17 10:47:18 +01:00
|
|
|
res = getparameter(flag, nextarg, &passarg, &config);
|
|
|
|
if(res) {
|
|
|
|
switch(res) {
|
|
|
|
case PARAM_OPTION_AMBIGUOUS:
|
|
|
|
helpf("option %s is ambiguous\n", origopt);
|
|
|
|
break;
|
|
|
|
case PARAM_OPTION_UNKNOWN:
|
|
|
|
helpf("option %s is unknown\n", origopt);
|
|
|
|
break;
|
|
|
|
case PARAM_REQUIRES_PARAMETER:
|
|
|
|
helpf("option %s requires an extra argument!\n", origopt);
|
|
|
|
break;
|
|
|
|
case PARAM_BAD_USE:
|
|
|
|
helpf("option %s was wrongly used!\n", origopt);
|
|
|
|
break;
|
|
|
|
case PARAM_HELP_REQUESTED:
|
|
|
|
/* no text */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return CURLE_FAILED_INIT;
|
|
|
|
}
|
1999-12-29 15:20:26 +01:00
|
|
|
|
|
|
|
if(passarg) /* we're supposed to skip this */
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if(url) {
|
|
|
|
helpf("only one URL is supported!\n");
|
2000-05-22 16:12:12 +02:00
|
|
|
return CURLE_FAILED_INIT;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
|
|
|
url = argv[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if no URL was specified and there was one in the config file, get that
|
|
|
|
one */
|
|
|
|
if(!url && config.url)
|
|
|
|
url = config.url;
|
|
|
|
|
|
|
|
if(!url) {
|
|
|
|
helpf("no URL specified!\n");
|
2000-05-22 16:12:12 +02:00
|
|
|
return CURLE_FAILED_INIT;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
2000-10-09 13:13:17 +02:00
|
|
|
if(NULL == config.useragent) {
|
|
|
|
/* set non-zero default values: */
|
|
|
|
snprintf(useragent, sizeof(useragent),
|
|
|
|
CURL_NAME "/" CURL_VERSION " (" OS ") " "%s", curl_version());
|
|
|
|
config.useragent= useragent;
|
|
|
|
}
|
2000-11-17 15:03:58 +01:00
|
|
|
else
|
|
|
|
allocuseragent = TRUE;
|
1999-12-29 15:20:26 +01:00
|
|
|
#if 0
|
2000-05-09 14:29:28 +02:00
|
|
|
fprintf(stderr, "URL: %s PROXY: %s\n", url, config.proxy?config.proxy:"none");
|
1999-12-29 15:20:26 +01:00
|
|
|
#endif
|
|
|
|
|
2000-05-09 14:29:28 +02:00
|
|
|
/* expand '{...}' and '[...]' expressions and return total number of URLs
|
|
|
|
in pattern set */
|
|
|
|
res = glob_url(&urls, url, &urlnum);
|
2000-05-22 16:12:12 +02:00
|
|
|
if(res != CURLE_OK)
|
2000-05-09 14:29:28 +02:00
|
|
|
return res;
|
|
|
|
|
2000-11-17 15:03:58 +01:00
|
|
|
/* save outfile pattern befor expansion */
|
|
|
|
outfiles = strdup(config.outfile);
|
|
|
|
|
1999-12-29 15:20:26 +01:00
|
|
|
if (!outfiles && !config.remotefile && urlnum > 1) {
|
|
|
|
#ifdef CURL_SEPARATORS
|
|
|
|
/* multiple files extracted to stdout, insert separators! */
|
|
|
|
separator = 1;
|
|
|
|
#endif
|
|
|
|
#ifdef MIME_SEPARATORS
|
|
|
|
/* multiple files extracted to stdout, insert MIME separators! */
|
|
|
|
separator = 1;
|
|
|
|
printf("MIME-Version: 1.0\n");
|
|
|
|
printf("Content-Type: multipart/mixed; boundary=%s\n\n", MIMEseparator);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
for (i = 0; (url = next_url(urls)); ++i) {
|
2000-11-17 15:03:58 +01:00
|
|
|
if (outfiles) {
|
|
|
|
free(config.outfile);
|
|
|
|
config.outfile = outfiles;
|
|
|
|
}
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2000-11-17 15:03:58 +01:00
|
|
|
if (config.outfile || config.remotefile) {
|
|
|
|
/*
|
|
|
|
* We have specified a file name to store the result in, or we have
|
|
|
|
* decided we want to use the remote file name.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if(!config.outfile && config.remotefile) {
|
|
|
|
/* Find and get the remote file name */
|
|
|
|
config.outfile=strstr(url, "://");
|
|
|
|
if(config.outfile)
|
|
|
|
config.outfile+=3;
|
|
|
|
else
|
|
|
|
config.outfile=url;
|
|
|
|
config.outfile = strdup(strrchr(config.outfile, '/'));
|
|
|
|
if(!config.outfile || !strlen(++config.outfile)) {
|
|
|
|
helpf("Remote file name has no length!\n");
|
|
|
|
return CURLE_WRITE_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* fill '#1' ... '#9' terms from URL pattern */
|
|
|
|
char *outfile = config.outfile;
|
|
|
|
config.outfile = match_url(config.outfile, *urls);
|
|
|
|
free(outfile);
|
|
|
|
}
|
|
|
|
|
|
|
|
if((0 == config.resume_from) && config.use_resume) {
|
|
|
|
/* we're told to continue where we are now, then we get the size of the
|
|
|
|
file as it is now and open it for append instead */
|
|
|
|
struct stat fileinfo;
|
|
|
|
|
|
|
|
if(0 == stat(config.outfile, &fileinfo)) {
|
|
|
|
/* set offset to current file size: */
|
|
|
|
config.resume_from = fileinfo.st_size;
|
|
|
|
}
|
|
|
|
/* else let offset remain 0 */
|
|
|
|
}
|
|
|
|
|
|
|
|
if(config.resume_from) {
|
|
|
|
/* open file for output: */
|
|
|
|
outs.stream=(FILE *) fopen(config.outfile, config.resume_from?"ab":"wb");
|
|
|
|
if (!outs.stream) {
|
|
|
|
helpf("Can't open '%s'!\n", config.outfile);
|
|
|
|
return CURLE_WRITE_ERROR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
outs.filename = config.outfile;
|
|
|
|
outs.stream = NULL; /* open when needed */
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
|
|
|
}
|
2000-11-17 15:03:58 +01:00
|
|
|
if (config.infile) {
|
|
|
|
/*
|
|
|
|
* We have specified a file to upload
|
|
|
|
*/
|
1999-12-29 15:20:26 +01:00
|
|
|
struct stat fileinfo;
|
|
|
|
|
2000-11-17 15:03:58 +01:00
|
|
|
/* If no file name part is given in the URL, we add this file name */
|
|
|
|
char *ptr=strstr(url, "://");
|
|
|
|
if(ptr)
|
|
|
|
ptr+=3;
|
|
|
|
else
|
|
|
|
ptr=url;
|
|
|
|
ptr = strrchr(ptr, '/');
|
|
|
|
if(!ptr || !strlen(++ptr)) {
|
|
|
|
/* The URL has no file name part, add the local file name. In order to
|
|
|
|
be able to do so, we have to create a new URL in another buffer.*/
|
|
|
|
|
|
|
|
urlbuffer=(char *)malloc(strlen(url) + strlen(config.infile) + 3);
|
|
|
|
if(!urlbuffer) {
|
|
|
|
helpf("out of memory\n");
|
|
|
|
return CURLE_OUT_OF_MEMORY;
|
|
|
|
}
|
|
|
|
if(ptr)
|
|
|
|
/* there is a trailing slash on the URL */
|
|
|
|
sprintf(urlbuffer, "%s%s", url, config.infile);
|
|
|
|
else
|
|
|
|
/* thers is no trailing slash on the URL */
|
|
|
|
sprintf(urlbuffer, "%s/%s", url, config.infile);
|
|
|
|
|
|
|
|
url = urlbuffer; /* use our new URL instead! */
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
|
|
|
|
2000-11-17 15:03:58 +01:00
|
|
|
infd=(FILE *) fopen(config.infile, "rb");
|
|
|
|
if (!infd || stat(config.infile, &fileinfo)) {
|
|
|
|
helpf("Can't open '%s'!\n", config.infile);
|
|
|
|
return CURLE_READ_ERROR;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
2000-11-17 15:03:58 +01:00
|
|
|
infilesize=fileinfo.st_size;
|
|
|
|
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
2000-11-17 15:03:58 +01:00
|
|
|
if((config.conf&CONF_UPLOAD) &&
|
|
|
|
config.use_resume &&
|
|
|
|
(0==config.resume_from)) {
|
|
|
|
config.resume_from = -1; /* -1 will then force get-it-yourself */
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
2000-11-17 15:03:58 +01:00
|
|
|
if(config.headerfile) {
|
|
|
|
/* open file for output: */
|
|
|
|
if(strcmp(config.headerfile,"-")) {
|
|
|
|
heads.filename = config.headerfile;
|
|
|
|
headerfilep=NULL;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
|
|
|
else
|
2000-11-17 15:03:58 +01:00
|
|
|
headerfilep=stdout;
|
|
|
|
heads.stream = headerfilep;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
2000-11-17 15:03:58 +01:00
|
|
|
|
|
|
|
if(outs.stream && isatty(fileno(outs.stream)) &&
|
|
|
|
!(config.conf&(CONF_UPLOAD|CONF_HTTPPOST)))
|
|
|
|
/* we send the output to a tty and it isn't an upload operation,
|
|
|
|
therefore we switch off the progress meter */
|
|
|
|
config.conf |= CONF_NOPROGRESS;
|
|
|
|
|
2000-05-10 00:42:53 +02:00
|
|
|
|
2000-11-17 15:03:58 +01:00
|
|
|
if (urlnum > 1) {
|
|
|
|
fprintf(stderr, "\n[%d/%d]: %s --> %s\n",
|
|
|
|
i+1, urlnum, url, config.outfile ? config.outfile : "<stdout>");
|
|
|
|
if (separator) {
|
1999-12-29 15:20:26 +01:00
|
|
|
#ifdef CURL_SEPARATORS
|
2000-11-17 15:03:58 +01:00
|
|
|
printf("%s%s\n", CURLseparator, url);
|
1999-12-29 15:20:26 +01:00
|
|
|
#endif
|
|
|
|
#ifdef MIME_SEPARATORS
|
2000-11-17 15:03:58 +01:00
|
|
|
printf("--%s\n", MIMEseparator);
|
|
|
|
printf("Content-ID: %s\n\n", url);
|
1999-12-29 15:20:26 +01:00
|
|
|
#endif
|
2000-11-17 15:03:58 +01:00
|
|
|
}
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
|
|
|
|
2000-11-17 15:03:58 +01:00
|
|
|
if(!config.errors)
|
|
|
|
config.errors = stderr;
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2000-06-14 14:50:38 +02:00
|
|
|
#ifdef WIN32
|
2000-11-17 15:03:58 +01:00
|
|
|
if(!config.outfile && !(config.conf & CONF_GETTEXT)) {
|
|
|
|
/* We get the output to stdout and we have not got the ASCII/text flag,
|
|
|
|
then set stdout to be binary */
|
|
|
|
setmode( 1, O_BINARY );
|
2000-06-16 15:18:49 +02:00
|
|
|
}
|
2000-11-17 15:03:58 +01:00
|
|
|
#endif
|
2000-06-16 15:18:49 +02:00
|
|
|
|
2000-05-22 16:12:12 +02:00
|
|
|
|
2000-11-17 15:03:58 +01:00
|
|
|
main_init();
|
|
|
|
|
|
|
|
/* The new, v7-style easy-interface! */
|
|
|
|
curl = curl_easy_init();
|
|
|
|
if(curl) {
|
|
|
|
curl_easy_setopt(curl, CURLOPT_FILE, (FILE *)&outs); /* where to store */
|
|
|
|
/* what call to write: */
|
|
|
|
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, my_fwrite);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_INFILE, infd); /* for uploads */
|
|
|
|
/* size of uploaded file: */
|
|
|
|
curl_easy_setopt(curl, CURLOPT_INFILESIZE, infilesize);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_URL, url); /* what to fetch */
|
|
|
|
curl_easy_setopt(curl, CURLOPT_PROXY, config.proxy); /* proxy to use */
|
|
|
|
curl_easy_setopt(curl, CURLOPT_VERBOSE, config.conf&CONF_VERBOSE);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_HEADER, config.conf&CONF_HEADER);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_NOPROGRESS, config.conf&CONF_NOPROGRESS);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_NOBODY, config.conf&CONF_NOBODY);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_FAILONERROR,
|
|
|
|
config.conf&CONF_FAILONERROR);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_UPLOAD, config.conf&CONF_UPLOAD);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_POST, config.conf&CONF_POST);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_FTPLISTONLY,
|
|
|
|
config.conf&CONF_FTPLISTONLY);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_FTPAPPEND, config.conf&CONF_FTPAPPEND);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_NETRC, config.conf&CONF_NETRC);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION,
|
|
|
|
config.conf&CONF_FOLLOWLOCATION);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_TRANSFERTEXT, config.conf&CONF_GETTEXT);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_PUT, config.conf&CONF_PUT);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_MUTE, config.conf&CONF_MUTE);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_USERPWD, config.userpwd);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_PROXYUSERPWD, config.proxyuserpwd);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_RANGE, config.range);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_ERRORBUFFER, errorbuffer);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_TIMEOUT, config.timeout);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_POSTFIELDS, config.postfields);
|
|
|
|
|
|
|
|
/* new in libcurl 7.2: */
|
|
|
|
curl_easy_setopt(curl, CURLOPT_POSTFIELDSIZE, config.postfieldsize);
|
|
|
|
|
|
|
|
curl_easy_setopt(curl, CURLOPT_REFERER, config.referer);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_AUTOREFERER,
|
|
|
|
config.conf&CONF_AUTO_REFERER);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_USERAGENT, config.useragent);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_FTPPORT, config.ftpport);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_LOW_SPEED_LIMIT, config.low_speed_limit);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_LOW_SPEED_TIME, config.low_speed_time);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_RESUME_FROM,
|
|
|
|
config.use_resume?config.resume_from:0);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_COOKIE, config.cookie);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_HTTPHEADER, config.headers);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_HTTPPOST, config.httppost);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_SSLCERT, config.cert);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_SSLCERTPASSWD, config.cert_passwd);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_CRLF, config.crlf);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_QUOTE, config.quote);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_POSTQUOTE, config.postquote);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_WRITEHEADER,
|
|
|
|
config.headerfile?&heads:NULL);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_COOKIEFILE, config.cookiefile);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_SSLVERSION, config.ssl_version);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_TIMECONDITION, config.timecond);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_TIMEVALUE, config.condtime);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_CUSTOMREQUEST, config.customrequest);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_STDERR, config.errors);
|
|
|
|
|
|
|
|
/* three new ones in libcurl 7.3: */
|
|
|
|
curl_easy_setopt(curl, CURLOPT_HTTPPROXYTUNNEL, config.proxytunnel);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_INTERFACE, config.iface);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_KRB4LEVEL, config.krb4level);
|
|
|
|
|
|
|
|
if((config.progressmode == CURL_PROGRESS_BAR) &&
|
|
|
|
!(config.conf&(CONF_NOPROGRESS|CONF_MUTE))) {
|
|
|
|
/* we want the alternative style, then we have to implement it
|
|
|
|
ourselves! */
|
|
|
|
progressbarinit(&progressbar);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_PROGRESSFUNCTION, myprogress);
|
|
|
|
curl_easy_setopt(curl, CURLOPT_PROGRESSDATA, &progressbar);
|
|
|
|
}
|
2000-05-22 16:12:12 +02:00
|
|
|
|
2000-11-17 15:03:58 +01:00
|
|
|
res = curl_easy_perform(curl);
|
2000-05-22 16:12:12 +02:00
|
|
|
|
2000-11-17 15:03:58 +01:00
|
|
|
if(config.writeout) {
|
|
|
|
ourWriteOut(curl, config.writeout);
|
|
|
|
}
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2000-11-17 15:03:58 +01:00
|
|
|
/* always cleanup */
|
|
|
|
curl_easy_cleanup(curl);
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2000-11-17 15:03:58 +01:00
|
|
|
if((res!=CURLE_OK) && config.showerror)
|
|
|
|
fprintf(config.errors, "curl: (%d) %s\n", res, errorbuffer);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
fprintf(config.errors, "curl: failed to init libcurl!\n");
|
2000-03-16 12:43:10 +01:00
|
|
|
|
2000-11-17 15:03:58 +01:00
|
|
|
main_free();
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2000-11-17 15:03:58 +01:00
|
|
|
if((config.errors != stderr) &&
|
|
|
|
(config.errors != stdout))
|
|
|
|
/* it wasn't directed to stdout or stderr so close the file! */
|
|
|
|
fclose(config.errors);
|
|
|
|
|
|
|
|
if(config.headerfile && !headerfilep && heads.stream)
|
|
|
|
fclose(heads.stream);
|
|
|
|
|
|
|
|
if(urlbuffer)
|
|
|
|
free(urlbuffer);
|
|
|
|
if (config.outfile && outs.stream)
|
|
|
|
fclose(outs.stream);
|
|
|
|
if (config.infile)
|
|
|
|
fclose(infd);
|
|
|
|
if(headerfilep)
|
|
|
|
fclose(headerfilep);
|
|
|
|
|
|
|
|
if(url)
|
|
|
|
free(url);
|
1999-12-29 15:20:26 +01:00
|
|
|
|
|
|
|
}
|
|
|
|
#ifdef MIME_SEPARATORS
|
|
|
|
if (separator)
|
|
|
|
printf("--%s--\n", MIMEseparator);
|
|
|
|
#endif
|
|
|
|
|
2000-11-17 15:03:58 +01:00
|
|
|
if(config.url)
|
|
|
|
free(config.url);
|
|
|
|
if(config.userpwd)
|
|
|
|
free(config.userpwd);
|
|
|
|
if(config.postfields)
|
|
|
|
free(config.postfields);
|
|
|
|
if(config.proxy)
|
|
|
|
free(config.proxy);
|
|
|
|
if(config.proxyuserpwd)
|
|
|
|
free(config.proxyuserpwd);
|
|
|
|
if(config.cookie)
|
|
|
|
free(config.cookie);
|
|
|
|
if(config.cookiefile)
|
|
|
|
free(config.cookiefile);
|
|
|
|
if(config.krb4level)
|
|
|
|
free(config.krb4level);
|
|
|
|
if(config.headerfile)
|
|
|
|
free(config.headerfile);
|
|
|
|
if(config.outfile)
|
|
|
|
free(config.outfile);
|
|
|
|
if(config.infile)
|
|
|
|
free(config.infile);
|
|
|
|
if(config.range)
|
|
|
|
free(config.range);
|
|
|
|
if(config.customrequest)
|
|
|
|
free(config.customrequest);
|
|
|
|
if(config.writeout)
|
|
|
|
free(config.writeout);
|
|
|
|
|
|
|
|
if(config.httppost)
|
|
|
|
curl_formfree(config.httppost);
|
|
|
|
|
|
|
|
if(allocuseragent)
|
|
|
|
free(config.useragent);
|
|
|
|
|
2000-10-09 23:35:40 +02:00
|
|
|
/* cleanup memory used for URL globbing patterns */
|
|
|
|
glob_cleanup(urls);
|
|
|
|
|
1999-12-29 15:20:26 +01:00
|
|
|
curl_slist_free_all(config.quote); /* the checks for config.quote == NULL */
|
2000-06-05 10:26:13 +02:00
|
|
|
curl_slist_free_all(config.postquote); /* */
|
|
|
|
curl_slist_free_all(config.headers); /* */
|
1999-12-29 15:20:26 +01:00
|
|
|
|
2000-11-17 15:03:58 +01:00
|
|
|
return res;
|
1999-12-29 15:20:26 +01:00
|
|
|
}
|
2000-02-11 00:03:08 +01:00
|
|
|
|
|
|
|
static char *my_get_line(FILE *fp)
|
|
|
|
{
|
|
|
|
char buf[4096];
|
|
|
|
char *nl = NULL;
|
|
|
|
char *retval = NULL;
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
if (NULL == fgets(buf, sizeof(buf), fp))
|
|
|
|
break;
|
|
|
|
if (NULL == retval)
|
|
|
|
retval = strdup(buf);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (NULL == (retval = realloc(retval,
|
|
|
|
strlen(retval) + strlen(buf) + 1)))
|
|
|
|
break;
|
|
|
|
strcat(retval, buf);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (NULL == (nl = strchr(retval, '\n')));
|
|
|
|
|
|
|
|
if (NULL != nl)
|
|
|
|
*nl = '\0';
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
static char *my_get_token(const char *line)
|
|
|
|
{
|
|
|
|
static const char *save = NULL;
|
|
|
|
const char *first = NULL;
|
|
|
|
const char *last = NULL;
|
|
|
|
char *retval = NULL;
|
|
|
|
size_t size;
|
|
|
|
|
|
|
|
if (NULL == line)
|
|
|
|
line = save;
|
|
|
|
if (NULL == line)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
while (('\0' != *line) && (isspace(*line)))
|
|
|
|
line++;
|
|
|
|
first = line;
|
|
|
|
while (('\0' != *line) && (!isspace(*line)))
|
|
|
|
line++;
|
|
|
|
save = line;
|
|
|
|
while ('\0' != *line)
|
|
|
|
line++;
|
|
|
|
last = line;
|
|
|
|
|
|
|
|
size = last - first;
|
|
|
|
if (0 == size)
|
|
|
|
return NULL;
|
|
|
|
if (NULL == (retval = malloc(size + 1)))
|
|
|
|
return NULL;
|
|
|
|
memcpy(retval, first, size);
|
|
|
|
retval[size] = '\0';
|
|
|
|
return retval;
|
|
|
|
}
|