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