mirror of
https://github.com/moparisthebest/curl
synced 2024-12-22 08:08:50 -05:00
- Added "--trace [file]" to the command line tool. It makes a very detailed
trace dump get stored, with a full protocol dump that includes all received and transmitted data. This could be a very effective tool for debugging what goes wrong. This dump includes every byte the way it is sent to/received from the server. The dump is the plain-text version, so SSL transfers will still be readable.
This commit is contained in:
parent
86cc34c0de
commit
57ff28c9b7
91
src/main.c
91
src/main.c
@ -370,9 +370,10 @@ static void help(void)
|
|||||||
puts(" -r/--range <range> Retrieve a byte range from a HTTP/1.1 or FTP server\n"
|
puts(" -r/--range <range> Retrieve a byte range from a HTTP/1.1 or FTP server\n"
|
||||||
" -R/--remote-time Set the remote file's time on the local output\n"
|
" -R/--remote-time Set the remote file's time on the local output\n"
|
||||||
" -s/--silent Silent mode. Don't output anything\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"
|
" -S/--show-error Show error. With -s, make curl show errors when they occur");
|
||||||
" --stderr <file> Where to redirect stderr. - means stdout.\n"
|
puts(" --stderr <file> Where to redirect stderr. - means stdout.\n"
|
||||||
" -t/--telnet-option <OPT=val> Set telnet option\n"
|
" -t/--telnet-option <OPT=val> Set telnet option\n"
|
||||||
|
" --trace <file> Dump a network/debug trace to the given file\n"
|
||||||
" -T/--upload-file <file> Transfer/upload <file> to remote site\n"
|
" -T/--upload-file <file> Transfer/upload <file> to remote site\n"
|
||||||
" --url <URL> Another way to specify URL to work with");
|
" --url <URL> Another way to specify URL to work with");
|
||||||
puts(" -u/--user <user[:password]> Specify user and password to use\n"
|
puts(" -u/--user <user[:password]> Specify user and password to use\n"
|
||||||
@ -452,6 +453,8 @@ struct Configurable {
|
|||||||
bool crlf;
|
bool crlf;
|
||||||
char *customrequest;
|
char *customrequest;
|
||||||
char *krb4level;
|
char *krb4level;
|
||||||
|
char *trace_dump; /* file to dump the network trace to, or NULL */
|
||||||
|
FILE *trace_stream;
|
||||||
long httpversion;
|
long httpversion;
|
||||||
bool progressmode;
|
bool progressmode;
|
||||||
bool nobuffer;
|
bool nobuffer;
|
||||||
@ -960,6 +963,7 @@ static ParameterError getparameter(char *flag, /* f or -long-flag */
|
|||||||
#ifdef USE_ENVIRONMENT
|
#ifdef USE_ENVIRONMENT
|
||||||
{"5f", "environment", FALSE},
|
{"5f", "environment", FALSE},
|
||||||
#endif
|
#endif
|
||||||
|
{"5g", "trace", TRUE},
|
||||||
{"0", "http1.0", FALSE},
|
{"0", "http1.0", FALSE},
|
||||||
{"1", "tlsv1", FALSE},
|
{"1", "tlsv1", FALSE},
|
||||||
{"2", "sslv2", FALSE},
|
{"2", "sslv2", FALSE},
|
||||||
@ -1140,6 +1144,10 @@ static ParameterError getparameter(char *flag, /* f or -long-flag */
|
|||||||
config->writeenv ^= TRUE;
|
config->writeenv ^= TRUE;
|
||||||
break;
|
break;
|
||||||
#endif
|
#endif
|
||||||
|
case 'g': /* --trace */
|
||||||
|
GetStr(&config->trace_dump, nextarg);
|
||||||
|
config->conf ^= CONF_VERBOSE; /* talk a lot */
|
||||||
|
break;
|
||||||
default: /* the URL! */
|
default: /* the URL! */
|
||||||
{
|
{
|
||||||
struct getout *url;
|
struct getout *url;
|
||||||
@ -1504,7 +1512,8 @@ static ParameterError getparameter(char *flag, /* f or -long-flag */
|
|||||||
break;
|
break;
|
||||||
case 't':
|
case 't':
|
||||||
/* Telnet options */
|
/* Telnet options */
|
||||||
config->telnet_options = curl_slist_append(config->telnet_options, nextarg);
|
config->telnet_options =
|
||||||
|
curl_slist_append(config->telnet_options, nextarg);
|
||||||
break;
|
break;
|
||||||
case 'T':
|
case 'T':
|
||||||
/* we are uploading */
|
/* we are uploading */
|
||||||
@ -1914,6 +1923,73 @@ void progressbarinit(struct ProgressData *bar,
|
|||||||
bar->out = config->errors;
|
bar->out = config->errors;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
void dump(const char *text,
|
||||||
|
FILE *stream, unsigned char *ptr, size_t size)
|
||||||
|
{
|
||||||
|
size_t i;
|
||||||
|
size_t c;
|
||||||
|
|
||||||
|
#define DUMP_BYTES 16 /* per line */
|
||||||
|
|
||||||
|
fprintf(stream, "%s %d (0x%x) bytes\n", text, size, size);
|
||||||
|
|
||||||
|
for(i=0; i<size; i+= DUMP_BYTES) {
|
||||||
|
|
||||||
|
fprintf(stream, "%04x: ", i);
|
||||||
|
|
||||||
|
for(c = 0; c < DUMP_BYTES; c++)
|
||||||
|
if(i+c < size)
|
||||||
|
fprintf(stream, "%02x ", ptr[i+c]);
|
||||||
|
else
|
||||||
|
fputs(" ", stream);
|
||||||
|
|
||||||
|
for(c = 0; (c < DUMP_BYTES) && (i+c < size); c++)
|
||||||
|
fprintf(stream, "%c",
|
||||||
|
(ptr[i+c]>=0x20) && (ptr[i+c]<0x80)?ptr[i+c]:'.');
|
||||||
|
|
||||||
|
fputc('\n', stream); /* newline */
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
int my_trace(CURL *handle, curl_infotype type,
|
||||||
|
unsigned char *data, size_t size,
|
||||||
|
void *userp)
|
||||||
|
{
|
||||||
|
struct Configurable *config = (struct Configurable *)userp;
|
||||||
|
FILE *output=config->errors;
|
||||||
|
|
||||||
|
(void)handle; /* prevent compiler warning */
|
||||||
|
|
||||||
|
if(!config->trace_stream)
|
||||||
|
/* open for append */
|
||||||
|
config->trace_stream = fopen(config->trace_dump, "w");
|
||||||
|
|
||||||
|
if(config->trace_stream)
|
||||||
|
output = config->trace_stream;
|
||||||
|
|
||||||
|
switch (type) {
|
||||||
|
case CURLINFO_TEXT:
|
||||||
|
fprintf(output, "== Info: %s", data);
|
||||||
|
break;
|
||||||
|
case CURLINFO_HEADER_OUT:
|
||||||
|
dump("=> Send header", output, data, size);
|
||||||
|
break;
|
||||||
|
case CURLINFO_DATA_OUT:
|
||||||
|
dump("=> Send data ", output, data, size);
|
||||||
|
break;
|
||||||
|
case CURLINFO_HEADER_IN:
|
||||||
|
dump("<= Recv header", output, data, size);
|
||||||
|
break;
|
||||||
|
case CURLINFO_DATA_IN:
|
||||||
|
dump("<= Recv data", output, data, size);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
void free_config_fields(struct Configurable *config)
|
void free_config_fields(struct Configurable *config)
|
||||||
{
|
{
|
||||||
if(config->random_file)
|
if(config->random_file)
|
||||||
@ -2486,6 +2562,12 @@ operate(struct Configurable *config, int argc, char *argv[])
|
|||||||
/* disable it */
|
/* disable it */
|
||||||
curl_easy_setopt(curl, CURLOPT_FTP_USE_EPSV, FALSE);
|
curl_easy_setopt(curl, CURLOPT_FTP_USE_EPSV, FALSE);
|
||||||
|
|
||||||
|
/* new in curl 7.9.7 */
|
||||||
|
if(config->trace_dump) {
|
||||||
|
curl_easy_setopt(curl, CURLOPT_DEBUGFUNCTION, my_trace);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_DEBUGDATA, config);
|
||||||
|
}
|
||||||
|
|
||||||
res = curl_easy_perform(curl);
|
res = curl_easy_perform(curl);
|
||||||
|
|
||||||
if((config->progressmode == CURL_PROGRESS_BAR) &&
|
if((config->progressmode == CURL_PROGRESS_BAR) &&
|
||||||
@ -2568,6 +2650,9 @@ operate(struct Configurable *config, int argc, char *argv[])
|
|||||||
if(config->headerfile && !headerfilep && heads.stream)
|
if(config->headerfile && !headerfilep && heads.stream)
|
||||||
fclose(heads.stream);
|
fclose(heads.stream);
|
||||||
|
|
||||||
|
if(config->trace_stream)
|
||||||
|
fclose(config->trace_stream);
|
||||||
|
|
||||||
if(allocuseragent)
|
if(allocuseragent)
|
||||||
free(config->useragent);
|
free(config->useragent);
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user