]> sjero.net Git - wget/blob - src/main.c
Merge build info with head.
[wget] / src / main.c
1 /* Command line parsing.
2    Copyright (C) 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003,
3    2004, 2005, 2006, 2007, 2008 Free Software Foundation, Inc.
4
5 This file is part of GNU Wget.
6
7 GNU Wget is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 3 of the License, or
10 (at your option) any later version.
11
12 GNU Wget is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15 GNU General Public License for more details.
16
17 You should have received a copy of the GNU General Public License
18 along with Wget.  If not, see <http://www.gnu.org/licenses/>.
19
20 Additional permission under GNU GPL version 3 section 7
21
22 If you modify this program, or any covered work, by linking or
23 combining it with the OpenSSL project's OpenSSL library (or a
24 modified version of that library), containing parts covered by the
25 terms of the OpenSSL or SSLeay licenses, the Free Software Foundation
26 grants you additional permission to convey the resulting work.
27 Corresponding Source for a non-source form of such a combination
28 shall include the source code for the parts of OpenSSL used as well
29 as that of the covered work.  */
30
31 #include "wget.h"
32
33 #include <stdio.h>
34 #include <stdlib.h>
35 #ifdef HAVE_UNISTD_H
36 # include <unistd.h>
37 #endif /* HAVE_UNISTD_H */
38 #include <string.h>
39 #include <signal.h>
40 #ifdef ENABLE_NLS
41 # include <locale.h>
42 #endif
43 #include <assert.h>
44 #include <errno.h>
45 #include <time.h>
46
47 #include "utils.h"
48 #include "init.h"
49 #include "retr.h"
50 #include "recur.h"
51 #include "host.h"
52 #include "url.h"
53 #include "progress.h"           /* for progress_handle_sigwinch */
54 #include "convert.h"
55 #include "spider.h"
56 #include "http.h"               /* for save_cookies */
57
58 #include <getopt.h>
59 #include <getpass.h>
60 #include <quote.h>
61
62 #ifndef PATH_SEPARATOR
63 # define PATH_SEPARATOR '/'
64 #endif
65
66 struct options opt;
67
68 /* defined in version.c */
69 extern char *version_string;
70 extern char *compilation_string;
71 extern char *system_getrc;
72 extern char *link_string;
73 /* defined in build_info.c */
74 extern char *compiled_features[];
75 extern char *system_wgetrc;
76 extern char *locale_dir;
77 /* Used for --version output in print_version */
78 static const int max_chars_per_line = 72;
79
80 #if defined(SIGHUP) || defined(SIGUSR1)
81 static void redirect_output_signal (int);
82 #endif
83
84 const char *exec_name;
85 \f
86 #ifndef TESTING
87 /* Initialize I18N/L10N.  That amounts to invoking setlocale, and
88    setting up gettext's message catalog using bindtextdomain and
89    textdomain.  Does nothing if NLS is disabled or missing.  */
90
91 static void
92 i18n_initialize (void)
93 {
94   /* ENABLE_NLS implies existence of functions invoked here.  */
95 #ifdef ENABLE_NLS
96   /* Set the current locale.  */
97   setlocale (LC_ALL, "");
98   /* Set the text message domain.  */
99   bindtextdomain ("wget", LOCALEDIR);
100   textdomain ("wget");
101 #endif /* ENABLE_NLS */
102 }
103 \f
104 /* Definition of command-line options. */
105
106 static void print_help (void);
107 static void print_version (void);
108
109 #ifdef HAVE_SSL
110 # define IF_SSL(x) x
111 #else
112 # define IF_SSL(x) NULL
113 #endif
114
115 #ifdef ENABLE_DEBUG
116 # define WHEN_DEBUG(x) x
117 #else
118 # define WHEN_DEBUG(x) NULL
119 #endif
120
121 struct cmdline_option {
122   const char *long_name;
123   char short_name;
124   enum {
125     OPT_VALUE,
126     OPT_BOOLEAN,
127     OPT_FUNCALL,
128     /* Non-standard options that have to be handled specially in
129        main().  */
130     OPT__APPEND_OUTPUT,
131     OPT__CLOBBER,
132     OPT__DONT_REMOVE_LISTING,
133     OPT__EXECUTE,
134     OPT__NO,
135     OPT__PARENT
136   } type;
137   const void *data;             /* for standard options */
138   int argtype;                  /* for non-standard options */
139 };
140
141 static struct cmdline_option option_data[] =
142   {
143     { "accept", 'A', OPT_VALUE, "accept", -1 },
144     { "append-output", 'a', OPT__APPEND_OUTPUT, NULL, required_argument },
145     { "ask-password", 0, OPT_BOOLEAN, "askpassword", -1 },
146     { "auth-no-challenge", 0, OPT_BOOLEAN, "authnochallenge", -1 },
147     { "background", 'b', OPT_BOOLEAN, "background", -1 },
148     { "backup-converted", 'K', OPT_BOOLEAN, "backupconverted", -1 },
149     { "backups", 0, OPT_BOOLEAN, "backups", -1 },
150     { "base", 'B', OPT_VALUE, "base", -1 },
151     { "bind-address", 0, OPT_VALUE, "bindaddress", -1 },
152     { IF_SSL ("ca-certificate"), 0, OPT_VALUE, "cacertificate", -1 },
153     { IF_SSL ("ca-directory"), 0, OPT_VALUE, "cadirectory", -1 },
154     { "cache", 0, OPT_BOOLEAN, "cache", -1 },
155     { IF_SSL ("certificate"), 0, OPT_VALUE, "certificate", -1 },
156     { IF_SSL ("certificate-type"), 0, OPT_VALUE, "certificatetype", -1 },
157     { IF_SSL ("check-certificate"), 0, OPT_BOOLEAN, "checkcertificate", -1 },
158     { "clobber", 0, OPT__CLOBBER, NULL, optional_argument },
159     { "connect-timeout", 0, OPT_VALUE, "connecttimeout", -1 },
160     { "continue", 'c', OPT_BOOLEAN, "continue", -1 },
161     { "convert-links", 'k', OPT_BOOLEAN, "convertlinks", -1 },
162     { "content-disposition", 0, OPT_BOOLEAN, "contentdisposition", -1 },
163     { "cookies", 0, OPT_BOOLEAN, "cookies", -1 },
164     { "cut-dirs", 0, OPT_VALUE, "cutdirs", -1 },
165     { WHEN_DEBUG ("debug"), 'd', OPT_BOOLEAN, "debug", -1 },
166     { "delete-after", 0, OPT_BOOLEAN, "deleteafter", -1 },
167     { "directories", 0, OPT_BOOLEAN, "dirstruct", -1 },
168     { "directory-prefix", 'P', OPT_VALUE, "dirprefix", -1 },
169     { "dns-cache", 0, OPT_BOOLEAN, "dnscache", -1 },
170     { "dns-timeout", 0, OPT_VALUE, "dnstimeout", -1 },
171     { "domains", 'D', OPT_VALUE, "domains", -1 },
172     { "dont-remove-listing", 0, OPT__DONT_REMOVE_LISTING, NULL, no_argument },
173     { "dot-style", 0, OPT_VALUE, "dotstyle", -1 },
174     { "egd-file", 0, OPT_VALUE, "egdfile", -1 },
175     { "exclude-directories", 'X', OPT_VALUE, "excludedirectories", -1 },
176     { "exclude-domains", 0, OPT_VALUE, "excludedomains", -1 },
177     { "execute", 'e', OPT__EXECUTE, NULL, required_argument },
178     { "follow-ftp", 0, OPT_BOOLEAN, "followftp", -1 },
179     { "follow-tags", 0, OPT_VALUE, "followtags", -1 },
180     { "force-directories", 'x', OPT_BOOLEAN, "dirstruct", -1 },
181     { "force-html", 'F', OPT_BOOLEAN, "forcehtml", -1 },
182     { "ftp-password", 0, OPT_VALUE, "ftppassword", -1 },
183     { "ftp-user", 0, OPT_VALUE, "ftpuser", -1 },
184     { "glob", 0, OPT_BOOLEAN, "glob", -1 },
185     { "header", 0, OPT_VALUE, "header", -1 },
186     { "help", 'h', OPT_FUNCALL, (void *)print_help, no_argument },
187     { "host-directories", 0, OPT_BOOLEAN, "addhostdir", -1 },
188     { "html-extension", 'E', OPT_BOOLEAN, "htmlextension", -1 },
189     { "htmlify", 0, OPT_BOOLEAN, "htmlify", -1 },
190     { "http-keep-alive", 0, OPT_BOOLEAN, "httpkeepalive", -1 },
191     { "http-passwd", 0, OPT_VALUE, "httppassword", -1 }, /* deprecated */
192     { "http-password", 0, OPT_VALUE, "httppassword", -1 },
193     { "http-user", 0, OPT_VALUE, "httpuser", -1 },
194     { "ignore-case", 0, OPT_BOOLEAN, "ignorecase", -1 },
195     { "ignore-length", 0, OPT_BOOLEAN, "ignorelength", -1 },
196     { "ignore-tags", 0, OPT_VALUE, "ignoretags", -1 },
197     { "include-directories", 'I', OPT_VALUE, "includedirectories", -1 },
198 #ifdef ENABLE_IPV6
199     { "inet4-only", '4', OPT_BOOLEAN, "inet4only", -1 },
200     { "inet6-only", '6', OPT_BOOLEAN, "inet6only", -1 },
201 #endif
202     { "input-file", 'i', OPT_VALUE, "input", -1 },
203     { "keep-session-cookies", 0, OPT_BOOLEAN, "keepsessioncookies", -1 },
204     { "level", 'l', OPT_VALUE, "reclevel", -1 },
205     { "limit-rate", 0, OPT_VALUE, "limitrate", -1 },
206     { "load-cookies", 0, OPT_VALUE, "loadcookies", -1 },
207     { "max-redirect", 0, OPT_VALUE, "maxredirect", -1 },
208     { "mirror", 'm', OPT_BOOLEAN, "mirror", -1 },
209     { "no", 'n', OPT__NO, NULL, required_argument },
210     { "no-clobber", 0, OPT_BOOLEAN, "noclobber", -1 },
211     { "no-parent", 0, OPT_BOOLEAN, "noparent", -1 },
212     { "output-document", 'O', OPT_VALUE, "outputdocument", -1 },
213     { "output-file", 'o', OPT_VALUE, "logfile", -1 },
214     { "page-requisites", 'p', OPT_BOOLEAN, "pagerequisites", -1 },
215     { "parent", 0, OPT__PARENT, NULL, optional_argument },
216     { "passive-ftp", 0, OPT_BOOLEAN, "passiveftp", -1 },
217     { "password", 0, OPT_VALUE, "password", -1 },
218     { "post-data", 0, OPT_VALUE, "postdata", -1 },
219     { "post-file", 0, OPT_VALUE, "postfile", -1 },
220     { "prefer-family", 0, OPT_VALUE, "preferfamily", -1 },
221     { "preserve-permissions", 0, OPT_BOOLEAN, "preservepermissions", -1 },
222     { IF_SSL ("private-key"), 0, OPT_VALUE, "privatekey", -1 },
223     { IF_SSL ("private-key-type"), 0, OPT_VALUE, "privatekeytype", -1 },
224     { "progress", 0, OPT_VALUE, "progress", -1 },
225     { "protocol-directories", 0, OPT_BOOLEAN, "protocoldirectories", -1 },
226     { "proxy", 0, OPT_BOOLEAN, "useproxy", -1 },
227     { "proxy__compat", 'Y', OPT_VALUE, "useproxy", -1 }, /* back-compatible */
228     { "proxy-passwd", 0, OPT_VALUE, "proxypassword", -1 }, /* deprecated */
229     { "proxy-password", 0, OPT_VALUE, "proxypassword", -1 },
230     { "proxy-user", 0, OPT_VALUE, "proxyuser", -1 },
231     { "quiet", 'q', OPT_BOOLEAN, "quiet", -1 },
232     { "quota", 'Q', OPT_VALUE, "quota", -1 },
233     { "random-file", 0, OPT_VALUE, "randomfile", -1 },
234     { "random-wait", 0, OPT_BOOLEAN, "randomwait", -1 },
235     { "read-timeout", 0, OPT_VALUE, "readtimeout", -1 },
236     { "recursive", 'r', OPT_BOOLEAN, "recursive", -1 },
237     { "referer", 0, OPT_VALUE, "referer", -1 },
238     { "reject", 'R', OPT_VALUE, "reject", -1 },
239     { "relative", 'L', OPT_BOOLEAN, "relativeonly", -1 },
240     { "remove-listing", 0, OPT_BOOLEAN, "removelisting", -1 },
241     { "restrict-file-names", 0, OPT_BOOLEAN, "restrictfilenames", -1 },
242     { "retr-symlinks", 0, OPT_BOOLEAN, "retrsymlinks", -1 },
243     { "retry-connrefused", 0, OPT_BOOLEAN, "retryconnrefused", -1 },
244     { "save-cookies", 0, OPT_VALUE, "savecookies", -1 },
245     { "save-headers", 0, OPT_BOOLEAN, "saveheaders", -1 },
246     { IF_SSL ("secure-protocol"), 0, OPT_VALUE, "secureprotocol", -1 },
247     { "server-response", 'S', OPT_BOOLEAN, "serverresponse", -1 },
248     { "span-hosts", 'H', OPT_BOOLEAN, "spanhosts", -1 },
249     { "spider", 0, OPT_BOOLEAN, "spider", -1 },
250     { "strict-comments", 0, OPT_BOOLEAN, "strictcomments", -1 },
251     { "timeout", 'T', OPT_VALUE, "timeout", -1 },
252     { "timestamping", 'N', OPT_BOOLEAN, "timestamping", -1 },
253     { "tries", 't', OPT_VALUE, "tries", -1 },
254     { "user", 0, OPT_VALUE, "user", -1 },
255     { "user-agent", 'U', OPT_VALUE, "useragent", -1 },
256     { "verbose", 'v', OPT_BOOLEAN, "verbose", -1 },
257     { "verbose", 0, OPT_BOOLEAN, "verbose", -1 },
258     { "version", 'V', OPT_FUNCALL, (void *) print_version, no_argument },
259     { "wait", 'w', OPT_VALUE, "wait", -1 },
260     { "waitretry", 0, OPT_VALUE, "waitretry", -1 },
261 #ifdef MSDOS
262     { "wdebug", 0, OPT_BOOLEAN, "wdebug", -1 },
263 #endif
264   };
265
266 #undef WHEN_DEBUG
267 #undef IF_SSL
268
269 /* Return a string that contains S with "no-" prepended.  The string
270    is NUL-terminated and allocated off static storage at Wget
271    startup.  */
272
273 static char *
274 no_prefix (const char *s)
275 {
276   static char buffer[1024];
277   static char *p = buffer;
278
279   char *cp = p;
280   int size = 3 + strlen (s) + 1;  /* "no-STRING\0" */
281   if (p + size >= buffer + sizeof (buffer))
282     abort ();
283
284   cp[0] = 'n', cp[1] = 'o', cp[2] = '-';
285   strcpy (cp + 3, s);
286   p += size;
287   return cp;
288 }
289
290 /* The arguments that that main passes to getopt_long. */
291 static struct option long_options[2 * countof (option_data) + 1];
292 static char short_options[128];
293
294 /* Mapping between short option chars and option_data indices. */
295 static unsigned char optmap[96];
296
297 /* Marker for `--no-FOO' values in long_options.  */
298 #define BOOLEAN_NEG_MARKER 1024
299
300 /* Initialize the long_options array used by getopt_long from the data
301    in option_data.  */
302
303 static void
304 init_switches (void)
305 {
306   char *p = short_options;
307   size_t i, o = 0;
308   for (i = 0; i < countof (option_data); i++)
309     {
310       struct cmdline_option *opt = &option_data[i];
311       struct option *longopt;
312
313       if (!opt->long_name)
314         /* The option is disabled. */
315         continue;
316
317       longopt = &long_options[o++];
318       longopt->name = opt->long_name;
319       longopt->val = i;
320       if (opt->short_name)
321         {
322           *p++ = opt->short_name;
323           optmap[opt->short_name - 32] = longopt - long_options;
324         }
325       switch (opt->type)
326         {
327         case OPT_VALUE:
328           longopt->has_arg = required_argument;
329           if (opt->short_name)
330             *p++ = ':';
331           break;
332         case OPT_BOOLEAN:
333           /* Specify an optional argument for long options, so that
334              --option=off works the same as --no-option, for
335              compatibility with pre-1.10 Wget.  However, don't specify
336              optional arguments short-option booleans because they
337              prevent combining of short options.  */
338           longopt->has_arg = optional_argument;
339           /* For Boolean options, add the "--no-FOO" variant, which is
340              identical to "--foo", except it has opposite meaning and
341              it doesn't allow an argument.  */
342           longopt = &long_options[o++];
343           longopt->name = no_prefix (opt->long_name);
344           longopt->has_arg = no_argument;
345           /* Mask the value so we'll be able to recognize that we're
346              dealing with the false value.  */
347           longopt->val = i | BOOLEAN_NEG_MARKER;
348           break;
349         default:
350           assert (opt->argtype != -1);
351           longopt->has_arg = opt->argtype;
352           if (opt->short_name)
353             {
354               if (longopt->has_arg == required_argument)
355                 *p++ = ':';
356               /* Don't handle optional_argument */
357             }
358         }
359     }
360   /* Terminate short_options. */
361   *p = '\0';
362   /* No need for xzero(long_options[o]) because its storage is static
363      and it will be zeroed by default.  */
364   assert (o <= countof (long_options));
365 }
366
367 /* Print the usage message.  */
368 static void
369 print_usage (void)
370 {
371   printf (_("Usage: %s [OPTION]... [URL]...\n"), exec_name);
372 }
373
374 /* Print the help message, describing all the available options.  If
375    you add an option, be sure to update this list.  */
376 static void
377 print_help (void)
378 {
379   /* We split the help text this way to ease translation of individual
380      entries.  */
381   static const char *help[] = {
382     "\n",
383     N_("\
384 Mandatory arguments to long options are mandatory for short options too.\n\n"),
385     N_("\
386 Startup:\n"),
387     N_("\
388   -V,  --version           display the version of Wget and exit.\n"),
389     N_("\
390   -h,  --help              print this help.\n"),
391     N_("\
392   -b,  --background        go to background after startup.\n"),
393     N_("\
394   -e,  --execute=COMMAND   execute a `.wgetrc'-style command.\n"),
395     "\n",
396
397     N_("\
398 Logging and input file:\n"),
399     N_("\
400   -o,  --output-file=FILE    log messages to FILE.\n"),
401     N_("\
402   -a,  --append-output=FILE  append messages to FILE.\n"),
403 #ifdef ENABLE_DEBUG
404     N_("\
405   -d,  --debug               print lots of debugging information.\n"),
406 #endif
407 #ifdef MSDOS
408     N_("\
409        --wdebug              print Watt-32 debug output.\n"),
410 #endif
411     N_("\
412   -q,  --quiet               quiet (no output).\n"),
413     N_("\
414   -v,  --verbose             be verbose (this is the default).\n"),
415     N_("\
416   -nv, --no-verbose          turn off verboseness, without being quiet.\n"),
417     N_("\
418   -i,  --input-file=FILE     download URLs found in FILE.\n"),
419     N_("\
420   -F,  --force-html          treat input file as HTML.\n"),
421     N_("\
422   -B,  --base=URL            prepends URL to relative links in -F -i file.\n"),
423     "\n",
424
425     N_("\
426 Download:\n"),
427     N_("\
428   -t,  --tries=NUMBER            set number of retries to NUMBER (0 unlimits).\n"),
429     N_("\
430        --retry-connrefused       retry even if connection is refused.\n"),
431     N_("\
432   -O,  --output-document=FILE    write documents to FILE.\n"),
433     N_("\
434   -nc, --no-clobber              skip downloads that would download to\n\
435                                  existing files.\n"),
436     N_("\
437   -c,  --continue                resume getting a partially-downloaded file.\n"),
438     N_("\
439        --progress=TYPE           select progress gauge type.\n"),
440     N_("\
441   -N,  --timestamping            don't re-retrieve files unless newer than\n\
442                                  local.\n"),
443     N_("\
444   -S,  --server-response         print server response.\n"),
445     N_("\
446        --spider                  don't download anything.\n"),
447     N_("\
448   -T,  --timeout=SECONDS         set all timeout values to SECONDS.\n"),
449     N_("\
450        --dns-timeout=SECS        set the DNS lookup timeout to SECS.\n"),
451     N_("\
452        --connect-timeout=SECS    set the connect timeout to SECS.\n"),
453     N_("\
454        --read-timeout=SECS       set the read timeout to SECS.\n"),
455     N_("\
456   -w,  --wait=SECONDS            wait SECONDS between retrievals.\n"),
457     N_("\
458        --waitretry=SECONDS       wait 1..SECONDS between retries of a retrieval.\n"),
459     N_("\
460        --random-wait             wait from 0...2*WAIT secs between retrievals.\n"),
461     N_("\
462        --no-proxy                explicitly turn off proxy.\n"),
463     N_("\
464   -Q,  --quota=NUMBER            set retrieval quota to NUMBER.\n"),
465     N_("\
466        --bind-address=ADDRESS    bind to ADDRESS (hostname or IP) on local host.\n"),
467     N_("\
468        --limit-rate=RATE         limit download rate to RATE.\n"),
469     N_("\
470        --no-dns-cache            disable caching DNS lookups.\n"),
471     N_("\
472        --restrict-file-names=OS  restrict chars in file names to ones OS allows.\n"),
473     N_("\
474        --ignore-case             ignore case when matching files/directories.\n"),
475 #ifdef ENABLE_IPV6
476     N_("\
477   -4,  --inet4-only              connect only to IPv4 addresses.\n"),
478     N_("\
479   -6,  --inet6-only              connect only to IPv6 addresses.\n"),
480     N_("\
481        --prefer-family=FAMILY    connect first to addresses of specified family,\n\
482                                  one of IPv6, IPv4, or none.\n"),
483 #endif
484     N_("\
485        --user=USER               set both ftp and http user to USER.\n"),
486     N_("\
487        --password=PASS           set both ftp and http password to PASS.\n"),
488     N_("\
489        --ask-password            prompt for passwords.\n"),
490     "\n",
491
492     N_("\
493 Directories:\n"),
494     N_("\
495   -nd, --no-directories           don't create directories.\n"),
496     N_("\
497   -x,  --force-directories        force creation of directories.\n"),
498     N_("\
499   -nH, --no-host-directories      don't create host directories.\n"),
500     N_("\
501        --protocol-directories     use protocol name in directories.\n"),
502     N_("\
503   -P,  --directory-prefix=PREFIX  save files to PREFIX/...\n"),
504     N_("\
505        --cut-dirs=NUMBER          ignore NUMBER remote directory components.\n"),
506     "\n",
507
508     N_("\
509 HTTP options:\n"),
510     N_("\
511        --http-user=USER        set http user to USER.\n"),
512     N_("\
513        --http-password=PASS    set http password to PASS.\n"),
514     N_("\
515        --no-cache              disallow server-cached data.\n"),
516     N_("\
517   -E,  --html-extension        save HTML documents with `.html' extension.\n"),
518     N_("\
519        --ignore-length         ignore `Content-Length' header field.\n"),
520     N_("\
521        --header=STRING         insert STRING among the headers.\n"),
522     N_("\
523        --max-redirect          maximum redirections allowed per page.\n"),
524     N_("\
525        --proxy-user=USER       set USER as proxy username.\n"),
526     N_("\
527        --proxy-password=PASS   set PASS as proxy password.\n"),
528     N_("\
529        --referer=URL           include `Referer: URL' header in HTTP request.\n"),
530     N_("\
531        --save-headers          save the HTTP headers to file.\n"),
532     N_("\
533   -U,  --user-agent=AGENT      identify as AGENT instead of Wget/VERSION.\n"),
534     N_("\
535        --no-http-keep-alive    disable HTTP keep-alive (persistent connections).\n"),
536     N_("\
537        --no-cookies            don't use cookies.\n"),
538     N_("\
539        --load-cookies=FILE     load cookies from FILE before session.\n"),
540     N_("\
541        --save-cookies=FILE     save cookies to FILE after session.\n"),
542     N_("\
543        --keep-session-cookies  load and save session (non-permanent) cookies.\n"),
544     N_("\
545        --post-data=STRING      use the POST method; send STRING as the data.\n"),
546     N_("\
547        --post-file=FILE        use the POST method; send contents of FILE.\n"),
548     N_("\
549        --content-disposition   honor the Content-Disposition header when\n\
550                                choosing local file names (EXPERIMENTAL).\n"),
551     N_("\
552        --auth-no-challenge     Send Basic HTTP authentication information\n\
553                                without first waiting for the server's\n\
554                                challenge.\n"),
555     "\n",
556
557 #ifdef HAVE_SSL
558     N_("\
559 HTTPS (SSL/TLS) options:\n"),
560     N_("\
561        --secure-protocol=PR     choose secure protocol, one of auto, SSLv2,\n\
562                                 SSLv3, and TLSv1.\n"),
563     N_("\
564        --no-check-certificate   don't validate the server's certificate.\n"),
565     N_("\
566        --certificate=FILE       client certificate file.\n"),
567     N_("\
568        --certificate-type=TYPE  client certificate type, PEM or DER.\n"),
569     N_("\
570        --private-key=FILE       private key file.\n"),
571     N_("\
572        --private-key-type=TYPE  private key type, PEM or DER.\n"),
573     N_("\
574        --ca-certificate=FILE    file with the bundle of CA's.\n"),
575     N_("\
576        --ca-directory=DIR       directory where hash list of CA's is stored.\n"),
577     N_("\
578        --random-file=FILE       file with random data for seeding the SSL PRNG.\n"),
579     N_("\
580        --egd-file=FILE          file naming the EGD socket with random data.\n"),
581     "\n",
582 #endif /* HAVE_SSL */
583
584     N_("\
585 FTP options:\n"),
586     N_("\
587        --ftp-user=USER         set ftp user to USER.\n"),
588     N_("\
589        --ftp-password=PASS     set ftp password to PASS.\n"),
590     N_("\
591        --no-remove-listing     don't remove `.listing' files.\n"),
592     N_("\
593        --no-glob               turn off FTP file name globbing.\n"),
594     N_("\
595        --no-passive-ftp        disable the \"passive\" transfer mode.\n"),
596     N_("\
597        --retr-symlinks         when recursing, get linked-to files (not dir).\n"),
598     N_("\
599        --preserve-permissions  preserve remote file permissions.\n"),
600     "\n",
601
602     N_("\
603 Recursive download:\n"),
604     N_("\
605   -r,  --recursive          specify recursive download.\n"),
606     N_("\
607   -l,  --level=NUMBER       maximum recursion depth (inf or 0 for infinite).\n"),
608     N_("\
609        --delete-after       delete files locally after downloading them.\n"),
610     N_("\
611   -k,  --convert-links      make links in downloaded HTML point to local files.\n"),
612     N_("\
613   -K,  --backup-converted   before converting file X, back up as X.orig.\n"),
614     N_("\
615   -m,  --mirror             shortcut for -N -r -l inf --no-remove-listing.\n"),
616     N_("\
617   -p,  --page-requisites    get all images, etc. needed to display HTML page.\n"),
618     N_("\
619        --strict-comments    turn on strict (SGML) handling of HTML comments.\n"),
620     "\n",
621
622     N_("\
623 Recursive accept/reject:\n"),
624     N_("\
625   -A,  --accept=LIST               comma-separated list of accepted extensions.\n"),
626     N_("\
627   -R,  --reject=LIST               comma-separated list of rejected extensions.\n"),
628     N_("\
629   -D,  --domains=LIST              comma-separated list of accepted domains.\n"),
630     N_("\
631        --exclude-domains=LIST      comma-separated list of rejected domains.\n"),
632     N_("\
633        --follow-ftp                follow FTP links from HTML documents.\n"),
634     N_("\
635        --follow-tags=LIST          comma-separated list of followed HTML tags.\n"),
636     N_("\
637        --ignore-tags=LIST          comma-separated list of ignored HTML tags.\n"),
638     N_("\
639   -H,  --span-hosts                go to foreign hosts when recursive.\n"),
640     N_("\
641   -L,  --relative                  follow relative links only.\n"),
642     N_("\
643   -I,  --include-directories=LIST  list of allowed directories.\n"),
644     N_("\
645   -X,  --exclude-directories=LIST  list of excluded directories.\n"),
646     N_("\
647   -np, --no-parent                 don't ascend to the parent directory.\n"),
648     "\n",
649
650     N_("Mail bug reports and suggestions to <bug-wget@gnu.org>.\n")
651   };
652
653   size_t i;
654
655   printf (_("GNU Wget %s, a non-interactive network retriever.\n"),
656           version_string);
657   print_usage ();
658
659   for (i = 0; i < countof (help); i++)
660     fputs (_(help[i]), stdout);
661
662   exit (0);
663 }
664
665 /* Return a human-readable printed representation of INTERVAL,
666    measured in seconds.  */
667
668 static char *
669 secs_to_human_time (double interval)
670 {
671   static char buf[32];
672   int secs = (int) (interval + 0.5);
673   int hours, mins, days;
674
675   days = secs / 86400, secs %= 86400;
676   hours = secs / 3600, secs %= 3600;
677   mins = secs / 60, secs %= 60;
678
679   if (days)
680     sprintf (buf, "%dd %dh %dm %ds", days, hours, mins, secs);
681   else if (hours)
682     sprintf (buf, "%dh %dm %ds", hours, mins, secs);
683   else if (mins)
684     sprintf (buf, "%dm %ds", mins, secs);
685   else
686     sprintf (buf, "%ss", print_decimal (interval));
687
688   return buf;
689 }
690
691 static char *
692 prompt_for_password (void)
693 {
694   if (opt.user)
695     printf (_("Password for user %s: "), quote (opt.user));
696   else
697     printf (_("Password: "));
698   return getpass("");
699 }
700
701 /* Function that prints the line argument while limiting it
702    to at most line_length. prefix is printed on the first line
703    and an appropriate number of spaces are added on subsequent
704    lines.*/
705 static void
706 format_and_print_line (char* prefix, char* line,
707                        int line_length) 
708 {
709   assert (prefix != NULL);
710   assert (line != NULL);
711
712   if (line_length <= 0)
713     line_length = max_chars_per_line;
714
715   const int leading_spaces = strlen (prefix);
716   printf ("%s", prefix);
717   int remaining_chars = line_length - leading_spaces;
718   /* We break on spaces. */
719   char* token = strtok (line, " ");
720   while (token != NULL) 
721     {
722       /* If however a token is much larger than the maximum
723          line length, all bets are off and we simply print the
724          token on the next line. */
725       if (remaining_chars <= strlen (token)) 
726         {
727           printf ("\n");
728           int j = 0;
729           for (j = 0; j < leading_spaces; j++) 
730             {
731               printf (" ");
732             }
733           remaining_chars = line_length - leading_spaces;
734         }
735       printf ("%s ", token);
736       remaining_chars -= strlen (token) + 1;  // account for " "
737       token = strtok (NULL, " ");
738     }
739
740   printf ("\n");
741   xfree (prefix);
742   xfree (line);
743 }
744
745 static void
746 print_version (void)
747 {
748   const char *options_title = "Options    : ";
749   const char *wgetrc_title  = "Wgetrc     : ";
750   const char *locale_title  = "Locale     : ";
751   const char *compile_title = "Compile    : ";
752   const char *link_title    = "Link       : ";
753   const char *prefix_spaces = "             ";
754   const int prefix_space_length = strlen (prefix_spaces);
755
756   printf ("GNU Wget %s\n", version_string);
757   printf (options_title);
758   /* compiled_features is a char*[]. We limit the characters per
759      line to max_chars_per_line and prefix each line with a constant
760      number of spaces for proper alignment. */
761   int i =0;
762   for (i = 0; compiled_features[i] != NULL; ) 
763     {
764       int line_length = max_chars_per_line - prefix_space_length;
765       while ((line_length > 0) && (compiled_features[i] != NULL)) 
766         {
767           printf ("%s ", compiled_features[i]);
768           line_length -= strlen (compiled_features[i]) + 2;
769           i++;
770         }
771       printf ("\n");
772       if (compiled_features[i] != NULL) 
773         {
774           printf (prefix_spaces);
775         }
776     }
777   /* Handle the case when $WGETRC is unset and $HOME/.wgetrc is 
778      absent. */
779   printf (wgetrc_title);
780   char *env_wgetrc = wgetrc_env_file_name ();
781   if (env_wgetrc && *env_wgetrc) 
782     {
783       printf ("%s (env)\n%s", env_wgetrc, prefix_spaces);
784       xfree (env_wgetrc);
785     }
786   char *user_wgetrc = wgetrc_user_file_name ();
787   if (user_wgetrc) 
788     {
789       printf ("%s (user)\n%s", user_wgetrc, prefix_spaces);
790       xfree (user_wgetrc);
791     }
792   printf ("%s (system)\n", system_wgetrc);
793
794   format_and_print_line (strdup (locale_title),
795                          strdup (locale_dir), 
796                          max_chars_per_line);
797   
798   format_and_print_line (strdup (compile_title),
799                          strdup (compilation_string),
800                          max_chars_per_line);
801
802   format_and_print_line (strdup (link_title),
803                          strdup (link_string),
804                          max_chars_per_line);
805   printf ("\n");
806   /* TRANSLATORS: When available, an actual copyright character
807      (cirle-c) should be used in preference to "(C)". */
808   fputs (_("\
809 Copyright (C) 2008 Free Software Foundation, Inc.\n"), stdout);
810   fputs (_("\
811 License GPLv3+: GNU GPL version 3 or later\n\
812 <http://www.gnu.org/licenses/gpl.html>.\n\
813 This is free software: you are free to change and redistribute it.\n\
814 There is NO WARRANTY, to the extent permitted by law.\n"), stdout);
815   /* TRANSLATORS: When available, please use the proper diacritics for
816      names such as this one. See en_US.po for reference. */
817   fputs (_("\nOriginally written by Hrvoje Niksic <hniksic@xemacs.org>.\n"),
818          stdout);
819   fputs (_("Currently maintained by Micah Cowan <micah@cowan.name>.\n"),
820          stdout);
821   exit (0);
822 }
823
824 int
825 main (int argc, char **argv)
826 {
827   char **url, **t;
828   int i, ret, longindex;
829   int nurl, status;
830   bool append_to_log = false;
831
832   i18n_initialize ();
833
834   /* Construct the name of the executable, without the directory part.  */
835   exec_name = strrchr (argv[0], PATH_SEPARATOR);
836   if (!exec_name)
837     exec_name = argv[0];
838   else
839     ++exec_name;
840
841 #ifdef WINDOWS
842   /* Drop extension (typically .EXE) from executable filename. */
843   windows_main ((char **) &exec_name);
844 #endif
845
846   /* Set option defaults; read the system wgetrc and ~/.wgetrc.  */
847   initialize ();
848
849   init_switches ();
850   longindex = -1;
851   while ((ret = getopt_long (argc, argv,
852                              short_options, long_options, &longindex)) != -1)
853     {
854       int val;
855       struct cmdline_option *opt;
856
857       /* If LONGINDEX is unchanged, it means RET is referring a short
858          option.  */
859       if (longindex == -1)
860         {
861           if (ret == '?')
862             {
863               print_usage ();
864               printf ("\n");
865               printf (_("Try `%s --help' for more options.\n"), exec_name);
866               exit (2);
867             }
868           /* Find the short option character in the mapping.  */
869           longindex = optmap[ret - 32];
870         }
871       val = long_options[longindex].val;
872
873       /* Use the retrieved value to locate the option in the
874          option_data array, and to see if we're dealing with the
875          negated "--no-FOO" variant of the boolean option "--foo".  */
876       opt = &option_data[val & ~BOOLEAN_NEG_MARKER];
877       switch (opt->type)
878         {
879         case OPT_VALUE:
880           setoptval (opt->data, optarg, opt->long_name);
881           break;
882         case OPT_BOOLEAN:
883           if (optarg)
884             /* The user has specified a value -- use it. */
885             setoptval (opt->data, optarg, opt->long_name);
886           else
887             {
888               /* NEG is true for `--no-FOO' style boolean options. */
889               bool neg = !!(val & BOOLEAN_NEG_MARKER);
890               setoptval (opt->data, neg ? "0" : "1", opt->long_name);
891             }
892           break;
893         case OPT_FUNCALL:
894           {
895             void (*func) (void) = (void (*) (void)) opt->data;
896             func ();
897           }
898           break;
899         case OPT__APPEND_OUTPUT:
900           setoptval ("logfile", optarg, opt->long_name);
901           append_to_log = true;
902           break;
903         case OPT__EXECUTE:
904           run_command (optarg);
905           break;
906         case OPT__NO:
907           {
908             /* We support real --no-FOO flags now, but keep these
909                short options for convenience and backward
910                compatibility.  */
911             char *p;
912             for (p = optarg; *p; p++)
913               switch (*p)
914                 {
915                 case 'v':
916                   setoptval ("verbose", "0", opt->long_name);
917                   break;
918                 case 'H':
919                   setoptval ("addhostdir", "0", opt->long_name);
920                   break;
921                 case 'd':
922                   setoptval ("dirstruct", "0", opt->long_name);
923                   break;
924                 case 'c':
925                   setoptval ("noclobber", "1", opt->long_name);
926                   break;
927                 case 'p':
928                   setoptval ("noparent", "1", opt->long_name);
929                   break;
930                 default:
931                   printf (_("%s: illegal option -- `-n%c'\n"), exec_name, *p);
932                   print_usage ();
933                   printf ("\n");
934                   printf (_("Try `%s --help' for more options.\n"), exec_name);
935                   exit (1);
936                 }
937             break;
938           }
939         case OPT__PARENT:
940         case OPT__CLOBBER:
941           {
942             /* The wgetrc commands are named noparent and noclobber,
943                so we must revert the meaning of the cmdline options
944                before passing the value to setoptval.  */
945             bool flag = true;
946             if (optarg)
947               flag = (*optarg == '1' || c_tolower (*optarg) == 'y'
948                       || (c_tolower (optarg[0]) == 'o'
949                           && c_tolower (optarg[1]) == 'n'));
950             setoptval (opt->type == OPT__PARENT ? "noparent" : "noclobber",
951                        flag ? "0" : "1", opt->long_name);
952             break;
953           }
954         case OPT__DONT_REMOVE_LISTING:
955           setoptval ("removelisting", "0", opt->long_name);
956           break;
957         }
958
959       longindex = -1;
960     }
961
962   nurl = argc - optind;
963
964   /* All user options have now been processed, so it's now safe to do
965      interoption dependency checks. */
966
967   if (opt.reclevel == 0)
968       opt.reclevel = INFINITE_RECURSION; /* see recur.h for commentary */
969
970   if (opt.spider || opt.delete_after)
971       opt.no_dirstruct = true;
972
973   if (opt.page_requisites && !opt.recursive)
974     {
975       /* Don't set opt.recursive here because it would confuse the FTP
976          code.  Instead, call retrieve_tree below when either
977          page_requisites or recursive is requested.  */
978       opt.reclevel = 0;
979       if (!opt.no_dirstruct)
980         opt.dirstruct = 1;      /* normally handled by cmd_spec_recursive() */
981     }
982
983   if (opt.verbose == -1)
984     opt.verbose = !opt.quiet;
985
986   /* Sanity checks.  */
987   if (opt.verbose && opt.quiet)
988     {
989       printf (_("Can't be verbose and quiet at the same time.\n"));
990       print_usage ();
991       exit (1);
992     }
993   if (opt.timestamping && opt.noclobber)
994     {
995       printf (_("\
996 Can't timestamp and not clobber old files at the same time.\n"));
997       print_usage ();
998       exit (1);
999     }
1000 #ifdef ENABLE_IPV6
1001   if (opt.ipv4_only && opt.ipv6_only)
1002     {
1003       printf (_("Cannot specify both --inet4-only and --inet6-only.\n"));
1004       print_usage ();
1005       exit (1);
1006     }
1007 #endif
1008   if (opt.output_document)
1009     {
1010       if (opt.convert_links 
1011           && (nurl > 1 || opt.page_requisites || opt.recursive))
1012         {
1013           fputs (_("\
1014 Cannot specify both -k and -O if multiple URLs are given, or in combination\n\
1015 with -p or -r. See the manual for details.\n\n"), stdout);
1016           print_usage ();
1017           exit (1);
1018         }
1019       if (opt.page_requisites
1020           || opt.recursive)
1021         {
1022           logprintf (LOG_NOTQUIET, "%s", _("\
1023 WARNING: combining -O with -r or -p will mean that all downloaded content\n\
1024 will be placed in the single file you specified.\n\n"));
1025         }
1026       if (opt.timestamping)
1027         {
1028           logprintf (LOG_NOTQUIET, "%s", _("\
1029 WARNING: timestamping does nothing in combination with -O. See the manual\n\
1030 for details.\n\n"));
1031           opt.timestamping = false;
1032         }
1033       if (opt.noclobber && file_exists_p(opt.output_document)) 
1034            { 
1035               /* Check if output file exists; if it does, exit. */
1036               logprintf (LOG_VERBOSE, _("File `%s' already there; not retrieving.\n"), opt.output_document);
1037               exit(1);
1038            }  
1039     }
1040
1041   if (opt.ask_passwd && opt.passwd)
1042     {
1043       printf (_("Cannot specify both --ask-password and --password.\n"));
1044       print_usage ();
1045       exit (1);
1046     }
1047
1048   if (!nurl && !opt.input_filename)
1049     {
1050       /* No URL specified.  */
1051       printf (_("%s: missing URL\n"), exec_name);
1052       print_usage ();
1053       printf ("\n");
1054       /* #### Something nicer should be printed here -- similar to the
1055          pre-1.5 `--help' page.  */
1056       printf (_("Try `%s --help' for more options.\n"), exec_name);
1057       exit (1);
1058     }
1059
1060   if (opt.ask_passwd)
1061     {
1062       opt.passwd = prompt_for_password ();
1063
1064       if (opt.passwd == NULL || opt.passwd[0] == '\0')
1065         exit (1);
1066     }
1067
1068 #ifdef MSDOS
1069   if (opt.wdebug)
1070      dbug_init();
1071   sock_init();
1072 #else
1073   if (opt.background)
1074     fork_to_background ();
1075 #endif
1076
1077   /* Initialize progress.  Have to do this after the options are
1078      processed so we know where the log file is.  */
1079   if (opt.verbose)
1080     set_progress_implementation (opt.progress_type);
1081
1082   /* Fill in the arguments.  */
1083   url = alloca_array (char *, nurl + 1);
1084   for (i = 0; i < nurl; i++, optind++)
1085     {
1086       char *rewritten = rewrite_shorthand_url (argv[optind]);
1087       if (rewritten)
1088         url[i] = rewritten;
1089       else
1090         url[i] = xstrdup (argv[optind]);
1091     }
1092   url[i] = NULL;
1093
1094   /* Initialize logging.  */
1095   log_init (opt.lfilename, append_to_log);
1096
1097   DEBUGP (("DEBUG output created by Wget %s on %s.\n\n", version_string,
1098            OS_TYPE));
1099
1100   /* Open the output filename if necessary.  */
1101   if (opt.output_document)
1102     {
1103       if (HYPHENP (opt.output_document))
1104         {
1105 #ifdef WINDOWS
1106           FILE *result;
1107           result = freopen (NULL, "wb", stdout);
1108           if (result == NULL)
1109             {
1110               logputs (LOG_NOTQUIET, _("\
1111 WARNING: Can't reopen standard output in binary mode;\n\
1112          downloaded file may contain inappropriate line endings.\n"));
1113             }
1114 #endif
1115           output_stream = stdout;
1116         }
1117       else
1118         {
1119           struct_fstat st;
1120           output_stream = fopen (opt.output_document,
1121                                  opt.always_rest ? "ab" : "wb");
1122           if (output_stream == NULL)
1123             {
1124               perror (opt.output_document);
1125               exit (1);
1126             }
1127           if (fstat (fileno (output_stream), &st) == 0 && S_ISREG (st.st_mode))
1128             output_stream_regular = true;
1129         }
1130     }
1131
1132 #ifdef WINDOWS
1133   ws_startup ();
1134 #endif
1135
1136 #ifdef SIGHUP
1137   /* Setup the signal handler to redirect output when hangup is
1138      received.  */
1139   if (signal(SIGHUP, SIG_IGN) != SIG_IGN)
1140     signal(SIGHUP, redirect_output_signal);
1141 #endif
1142   /* ...and do the same for SIGUSR1.  */
1143 #ifdef SIGUSR1
1144   signal (SIGUSR1, redirect_output_signal);
1145 #endif
1146 #ifdef SIGPIPE
1147   /* Writing to a closed socket normally signals SIGPIPE, and the
1148      process exits.  What we want is to ignore SIGPIPE and just check
1149      for the return value of write().  */
1150   signal (SIGPIPE, SIG_IGN);
1151 #endif
1152 #ifdef SIGWINCH
1153   signal (SIGWINCH, progress_handle_sigwinch);
1154 #endif
1155
1156   status = RETROK;              /* initialize it, just-in-case */
1157   /* Retrieve the URLs from argument list.  */
1158   for (t = url; *t; t++)
1159     {
1160       char *filename = NULL, *redirected_URL = NULL;
1161       int dt;
1162
1163       if ((opt.recursive || opt.page_requisites)
1164           && (url_scheme (*t) != SCHEME_FTP || url_uses_proxy (*t)))
1165         {
1166           int old_follow_ftp = opt.follow_ftp;
1167
1168           /* Turn opt.follow_ftp on in case of recursive FTP retrieval */
1169           if (url_scheme (*t) == SCHEME_FTP) 
1170             opt.follow_ftp = 1;
1171           
1172           status = retrieve_tree (*t);
1173
1174           opt.follow_ftp = old_follow_ftp;
1175         }
1176       else
1177         status = retrieve_url (*t, &filename, &redirected_URL, NULL, &dt, opt.recursive);
1178
1179       if (opt.delete_after && file_exists_p(filename))
1180         {
1181           DEBUGP (("Removing file due to --delete-after in main():\n"));
1182           logprintf (LOG_VERBOSE, _("Removing %s.\n"), filename);
1183           if (unlink (filename))
1184             logprintf (LOG_NOTQUIET, "unlink: %s\n", strerror (errno));
1185         }
1186
1187       xfree_null (redirected_URL);
1188       xfree_null (filename);
1189     }
1190
1191   /* And then from the input file, if any.  */
1192   if (opt.input_filename)
1193     {
1194       int count;
1195       status = retrieve_from_file (opt.input_filename, opt.force_html, &count);
1196       if (!count)
1197         logprintf (LOG_NOTQUIET, _("No URLs found in %s.\n"),
1198                    opt.input_filename);
1199     }
1200
1201   /* Print broken links. */
1202   if (opt.recursive && opt.spider)
1203     {
1204       print_broken_links();
1205     }
1206   
1207   /* Print the downloaded sum.  */
1208   if ((opt.recursive || opt.page_requisites
1209        || nurl > 1
1210        || (opt.input_filename && total_downloaded_bytes != 0))
1211       &&
1212       total_downloaded_bytes != 0)
1213     {
1214       logprintf (LOG_NOTQUIET,
1215                  _("FINISHED --%s--\nDownloaded: %d files, %s in %s (%s)\n"),
1216                  datetime_str (time (NULL)),
1217                  opt.numurls,
1218                  human_readable (total_downloaded_bytes),
1219                  secs_to_human_time (total_download_time),
1220                  retr_rate (total_downloaded_bytes, total_download_time));
1221       /* Print quota warning, if exceeded.  */
1222       if (opt.quota && total_downloaded_bytes > opt.quota)
1223         logprintf (LOG_NOTQUIET,
1224                    _("Download quota of %s EXCEEDED!\n"),
1225                    human_readable (opt.quota));
1226     }
1227
1228   if (opt.cookies_output)
1229     save_cookies ();
1230
1231   if (opt.convert_links && !opt.delete_after)
1232     convert_all_links ();
1233
1234   log_close ();
1235   for (i = 0; i < nurl; i++)
1236     xfree (url[i]);
1237   cleanup ();
1238
1239 #ifdef DEBUG_MALLOC
1240   print_malloc_debug_stats ();
1241 #endif
1242   if (status == RETROK)
1243     return 0;
1244   else
1245     return 1;
1246 }
1247 #endif /* TESTING */
1248 \f
1249 #if defined(SIGHUP) || defined(SIGUSR1)
1250
1251 /* So the signal_name check doesn't blow when only one is available. */
1252 #ifndef SIGHUP
1253 # define SIGHUP -1
1254 #endif
1255 #ifndef SIGUSR1
1256 # define SIGUSR1 -1
1257 #endif
1258
1259 /* Hangup signal handler.  When wget receives SIGHUP or SIGUSR1, it
1260    will proceed operation as usual, trying to write into a log file.
1261    If that is impossible, the output will be turned off.  */
1262
1263 static void
1264 redirect_output_signal (int sig)
1265 {
1266   const char *signal_name = (sig == SIGHUP ? "SIGHUP" :
1267                              (sig == SIGUSR1 ? "SIGUSR1" :
1268                               "WTF?!"));
1269   log_request_redirect_output (signal_name);
1270   progress_schedule_redirect ();
1271   signal (sig, redirect_output_signal);
1272 }
1273 #endif
1274
1275 /*
1276  * vim: et ts=2 sw=2
1277  */