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