chiark / gitweb /
srv processing written (except for sorting); now to be debugged
[adns] / client / adh-opts.c
CommitLineData
39d64e6e 1/*
b3543b6b 2 * adh-opts.c
39d64e6e 3 * - useful general-purpose resolver client program
b3543b6b 4 * option handling tables etc.
39d64e6e 5 */
6/*
a79ac5ba 7 * This file is
89435c42 8 * Copyright (C) 1997-2000 Ian Jackson <ian@davenant.greenend.org.uk>
a79ac5ba 9 *
10 * It is part of adns, which is
89435c42 11 * Copyright (C) 1997-2000 Ian Jackson <ian@davenant.greenend.org.uk>
c6826df6 12 * Copyright (C) 1999-2000 Tony Finch <dot@dotat.at>
39d64e6e 13 *
14 * This program is free software; you can redistribute it and/or modify
15 * it under the terms of the GNU General Public License as published by
16 * the Free Software Foundation; either version 2, or (at your option)
17 * any later version.
18 *
19 * This program is distributed in the hope that it will be useful,
20 * but WITHOUT ANY WARRANTY; without even the implied warranty of
21 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
22 * GNU General Public License for more details.
23 *
24 * You should have received a copy of the GNU General Public License
25 * along with this program; if not, write to the Free Software Foundation,
26 * Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
27 */
28
aa1d7195 29#include "adnshost.h"
30
b3543b6b 31int ov_env=1, ov_pipe=0, ov_asynch=0;
32int ov_verbose= 0;
aa1d7195 33adns_rrtype ov_type= adns_r_none;
b3543b6b 34int ov_search=0, ov_qc_query=0, ov_qc_anshost=0, ov_qc_cname=1;
e02793ec 35int ov_tcp=0, ov_cname=0, ov_format=fmt_default;
aa1d7195 36char *ov_id= 0;
37struct perqueryflags_remember ov_pqfr = { 1,1,1, tm_none };
91ae575b 38
aa1d7195 39static const struct optioninfo global_options[]= {
91ae575b 40 { ot_desconly, "global binary options:" },
41 { ot_flag, "Do not look at environment variables at all",
42 "e", "env", &ov_env, 0 },
43 { ot_flag, "Read queries on stdin instead of using args",
44 "f", "pipe", &ov_pipe, 1 },
45 { ot_flag, "Allow answers to be reordered",
46 "a", "asynch", &ov_asynch, 1 },
e02793ec 47
48 { ot_desconly, "answer/error output format and destination (see below):" },
49 { ot_value, "Answers to stdout, errors as messages to stderr (default)",
50 "Fs", "fmt-simple", &ov_format, fmt_simple },
51 { ot_value, "Answers and errors both to stdout in parseable format",
52 "Fi", "fmt-inline", &ov_format, fmt_inline },
53 { ot_value, "Fully-parseable output format (default for --asynch)",
54 "Fa", "fmt-asynch", &ov_format, fmt_asynch },
91ae575b 55
56 { ot_desconly, "global verbosity level:" },
57 { ot_value, "Do not print anything to stderr",
58 "Vq", "quiet", &ov_verbose, adns_if_noerrprint },
59 { ot_value, "Report unexpected kinds of problem only (default)",
60 "Vn", "no-quiet", &ov_verbose, 0 },
61 { ot_value, "Debugging mode",
62 "Vd", "debug", &ov_verbose, adns_if_debug },
63
64 { ot_desconly, "other global options:" },
09aee00b 65 { ot_funcarg, "Configuration to use instead of /etc/resolv.conf",
66 0, "config", 0,0, of_config, "<config-text>" },
8a5e5147 67 { ot_func, "Print version number",
68 0, "version", 0,0, of_version },
91ae575b 69 { ot_func, "Print usage information",
70 0, "help", 0,0, of_help },
71
72 { ot_end }
73};
74
aa1d7195 75static const struct optioninfo perquery_options[]= {
91ae575b 76 { ot_desconly, "per-query options:" },
77 { ot_funcarg, "Query type (see below)",
78 "t", "type", 0,0, &of_type, "type" },
0b535035 79 { ot_funcarg, "Do reverse query (address -> name lookup)",
5e6342f0 80 "i", "ptr", 0,0, &of_ptr, "addr" },
d2f6d877 81 { ot_funcarg2, "Lookup in in-addr-like `zone' (eg MAPS RBL)",
82 0, "reverse", 0,0, &of_reverse, "addr","zone" },
91ae575b 83
84 { ot_desconly, "per-query binary options:" },
85 { ot_flag, "Use the search list",
86 "s", "search", &ov_search, 1 },
87 { ot_flag, "Let query domains contain quote-requiring chars",
88 "Qq", "qc-query", &ov_qc_query, 1 },
89 { ot_flag, "Let hostnames in answers contain ...",
90 "Qa", "qc-anshost", &ov_qc_anshost, 1 },
91 { ot_flag, "Prevent CNAME target domains from containing ...",
92 "Qc", "qc-cname", &ov_qc_cname, 0 },
93 { ot_flag, "Force use of a virtual circuit",
94 "u", "tcp", &ov_tcp, 1 },
95 { ot_flag, "Do not display owner name in output",
b3543b6b 96 "Do", "show-owner", &ov_pqfr.show_owner, 0 },
91ae575b 97 { ot_flag, "Do not display RR type in output",
b3543b6b 98 "Dt", "show-type", &ov_pqfr.show_type, 0 },
91ae575b 99 { ot_flag, "Do not display CNAME target in output",
b3543b6b 100 "Dc", "show-cname", &ov_pqfr.show_cname, 0 },
91ae575b 101
102 { ot_desconly, "per-query TTL mode (NB TTL is minimum across all info in reply):" },
103 { ot_value, "Show the TTL as a TTL",
b3543b6b 104 "Tt", "ttl-ttl", &ov_pqfr.ttl, tm_rel },
91ae575b 105 { ot_value, "Show the TTL as a time_t when the data might expire",
b3543b6b 106 "Ta", "ttl-abs", &ov_pqfr.ttl, tm_abs },
91ae575b 107 { ot_value, "Do not show the TTL (default)",
b3543b6b 108 "Tn", "no-ttl", &ov_pqfr.ttl, tm_none },
91ae575b 109
110 { ot_desconly, "per-query CNAME handling mode:" },
111 { ot_value, "Call it an error if a CNAME is found",
112 "Cf", "cname-reject", &ov_cname, adns_qf_cname_forbid },
113 { ot_value, "Allow references to CNAMEs in other RRs",
114 "Cl", "cname-loose", &ov_cname, adns_qf_cname_loose },
115 { ot_value, "CNAME ok for query domain, but not in RRs (default)",
116 "Cs", "cname-ok", &ov_cname, 0 },
117
b3543b6b 118 { ot_desconly, "asynchronous/pipe mode options:" },
91ae575b 119 { ot_funcarg, "Set <id>, default is decimal sequence starting 0",
0b535035 120 0, "asynch-id", 0,0, &of_asynch_id, "id" },
5692efb1 121 { ot_funcarg, "Cancel the query with id <id> (no error if not found)",
91ae575b 122 0, "cancel-id", 0,0, &of_cancel_id, "id" },
123
124 { ot_end }
125};
39d64e6e 126
127static void printusage(void) {
aa1d7195 128 static const struct optioninfo *const all_optiontables[]= {
b3543b6b 129 global_options, perquery_options, 0
91ae575b 130 };
131
aa1d7195 132 const struct optioninfo *const *oiap, *oip=0;
91ae575b 133 int maxsopt, maxlopt, l;
134
135 maxsopt= maxlopt= 0;
136
aa1d7195 137 for (oiap=all_optiontables; *oiap; oiap++) {
91ae575b 138 for (oip=*oiap; oip->type != ot_end; oip++) {
139 if (oip->type == ot_funcarg) continue;
140 if (oip->sopt) { l= strlen(oip->sopt); if (l>maxsopt) maxsopt= l; }
141 if (oip->lopt) {
142 l= strlen(oip->lopt);
143 if (oip->type == ot_flag && !oip->value) l+= 3;
144 if (l>maxlopt) maxlopt= l;
145 }
146 }
147 }
148
149 fputs("usage: adnshost [global-opts] [query-opts] query-domain\n"
150 " [[query-opts] query-domain ...]\n"
151 " adnshost [global-opts] [query-opts] -f|--pipe\n",
152 stdout);
153
aa1d7195 154 for (oiap=all_optiontables; *oiap; oiap++) {
91ae575b 155 putchar('\n');
156 for (oip=*oiap; oip->type != ot_end; oip++) {
157 switch (oip->type) {
158 case ot_flag:
159 if (!oip->value) {
160 if (oip->sopt) {
161 printf(" +%-*s --no-%-*s %s\n",
162 maxsopt, oip->sopt,
163 maxlopt-2, oip->lopt,
164 oip->desc);
165 } else {
166 printf(" --no-%-*s %s\n",
167 maxlopt+maxsopt+1, oip->lopt,
168 oip->desc);
169 }
170 break;
171 }
172 case ot_value: case ot_func: /* fall through */
173 if (oip->sopt) {
174 printf(" -%-*s --%-*s %s\n",
175 maxsopt, oip->sopt,
176 maxlopt+1, oip->lopt,
177 oip->desc);
178 } else {
179 printf(" --%-*s %s\n",
180 maxlopt+maxsopt+3, oip->lopt,
181 oip->desc);
182 }
183 break;
184 case ot_funcarg:
185 if (oip->sopt) {
186 l= (maxlopt + maxsopt - 9 -
187 (strlen(oip->sopt) + strlen(oip->lopt) + 2*strlen(oip->argdesc)));
188 printf(" -%s<%s> / --%s <%s>%*s%s\n",
189 oip->sopt, oip->argdesc, oip->lopt, oip->argdesc,
190 l>2 ? l : 2, "",
191 oip->desc);
192 } else {
193 l= (maxlopt + maxsopt + 1 -
194 (strlen(oip->lopt) + strlen(oip->argdesc)));
195 printf(" --%s <%s>%*s%s\n",
196 oip->lopt, oip->argdesc,
197 l>2 ? l : 2, "",
198 oip->desc);
199 }
200 break;
d2f6d877 201 case ot_funcarg2:
202 assert(!oip->sopt);
203 l= (maxlopt + maxsopt - 2 -
204 (strlen(oip->lopt) + strlen(oip->argdesc) + strlen(oip->argdesc2)));
205 printf(" --%s <%s> <%s>%*s%s\n",
206 oip->lopt, oip->argdesc, oip->argdesc2,
207 l>2 ? l : 2, "",
208 oip->desc);
209 break;
91ae575b 210 case ot_desconly:
211 printf("%s\n", oip->desc);
212 break;
213 default:
214 abort();
215 }
216 }
217 }
218
219 printf("\nEscaping domains which might start with `-':\n"
220 " - %-*s Next argument is a domain, but more options may follow\n",
221 maxlopt+maxsopt+3, "<domain>");
222
223 fputs("\n"
224 "Query domains should always be quoted according to master file format.\n"
225 "\n"
226 "For binary options, --FOO and --no-FOO are opposites, as are\n"
227 "-X and +X. In each case the default is the one not listed.\n"
228 "Per query options stay set a particular way until they are reset,\n"
229 "whether they appear on the command line or on stdin.\n"
230 "All global options must preceed the first query domain.\n"
231 "\n"
232 "With -f, the input should be lines with either an option, possibly\n"
233 "with a value argument (separated from the option by a space if it's a long\n"
234 "option), or a domain (possibly preceded by a hyphen and a space to\n"
235 "distinguish it from an option).\n"
236 "\n"
237 "Output format is master file format without class or TTL by default:\n"
238 " [<owner>] [<ttl>] [<type>] <data>\n"
239 "or if the <owner> domain refers to a CNAME and --show-cname is on\n"
240 " [<owner>] [<ttl>] CNAME <cname>\n"
241 " [<cname>] [<ttl>] <type> <data>\n"
e02793ec 242 "When a query fails you get an error message to stderr (with --fmt-simple).\n"
243 "Specify --fmt-inline for lines like this (broken here for readability):\n"
2798305f 244 " ; failed <statustype> <statusnum> <statusabbrev> \\\n"
245 " [<owner>] [<ttl>] [<cname>] \"<status string>\"\n"
e02793ec 246 "If you use --fmt-asynch, which is the default for --asynch,\n"
247 "each answer (success or failure) is preceded by a line\n"
2798305f 248 " <id> <nrrs> <statustype> <statusnum> <statusabbrev> \\\n"
249 " [<owner>] [<ttl>] [<cname>] \"<status string>\"\n"
91ae575b 250 "where <nrrs> is the number of RRs that follow and <cname> will be `$' or\n"
251 "the CNAME target; the CNAME indirection and error formats above are not used.\n"
252 "\n"
253 "Exit status:\n"
254 " 0 all went well\n"
255 " 1-6 at least one query failed with statustype:\n"
256 " 1 localfail )\n"
257 " 2 remotefail ) temporary errors\n"
258 " 3 tempfail __)_________________\n"
259 " 4 misconfig )\n"
260 " 5 misquery ) permanent errors\n"
261 " 6 permfail )\n"
262 " 10 system trouble\n"
263 " 11 usage problems\n"
264 "\n"
265 "Query types (see adns.h; default is addr):\n"
2248cdf4 266 " ns soa ptr mx rp srv addr - enhanced versions\n"
267 " cname hinfo txt - types with only one version\n"
268 " a ns- soa- ptr- mx- rp- srv- - _raw versions\n"
0b535035 269 "Default is addr, or ptr for -i/--ptr queries\n",
91ae575b 270 stdout);
271 if (ferror(stdout)) sysfail("write usage message",errno);
39d64e6e 272}
273
8a5e5147 274void of_version(const struct optioninfo *oi, const char *arg, const char *arg2) {
275 VERSION_PRINT_QUIT("adnshost");
276}
277
d2f6d877 278void of_help(const struct optioninfo *oi, const char *arg, const char *arg2) {
39d64e6e 279 printusage();
280 if (fclose(stdout)) sysfail("finish writing output",errno);
09aee00b 281 quitnow(0);
39d64e6e 282}
91ae575b 283
aa1d7195 284typedef int comparer_type(const char **optp, const struct optioninfo *entry);
285
286static int oc_long(const char **optp, const struct optioninfo *entry) {
287 return entry->lopt && !strcmp(*optp,entry->lopt);
288}
289
290static int oc_short(const char **optp, const struct optioninfo *entry) {
291 const char *sopt;
292 int l;
293
294 sopt= entry->sopt;
295 if (!sopt) return 0;
296 l= strlen(sopt);
297 if (memcmp(*optp,sopt,l)) return 0;
298 (*optp) += l;
299 return 1;
300}
301
302static const struct optioninfo *find1(const char **optp,
303 const struct optioninfo *table,
304 comparer_type *comparer) {
305 for (;;) {
306 if (table->type == ot_end) return 0;
307 if (comparer(optp,table)) return table;
308 table++;
309 }
310}
311
312static const struct optioninfo *find(const char **optp,
313 const char *prefix,
314 comparer_type *comparer) {
315 const struct optioninfo *oip;
44fb8e6f 316 const char *opt;
aa1d7195 317
44fb8e6f 318 opt= *optp;
aa1d7195 319 oip= find1(optp,perquery_options,comparer);
320 if (oip) return oip;
321 oip= find1(optp,global_options,comparer);
44fb8e6f 322 if (!oip) usageerr("unknown option %s%s",prefix,opt);
323 if (ads) usageerr("global option %s%s specified after query domain(s)",prefix,opt);
aa1d7195 324 return oip;
325}
326
327const struct optioninfo *opt_findl(const char *opt) { return find(&opt,"--",oc_long); }
328const struct optioninfo *opt_finds(const char **optp) { return find(optp,"-",oc_short); }
329
196e9104 330static void noninvert(const struct optioninfo *oip) NONRETURNING;
331static void noninvert(const struct optioninfo *oip) {
332 usageerr("option %s%s%s%s%s may not be inverted",
333 oip->sopt ? "-" : "", oip->sopt ? oip->sopt : "",
334 oip->lopt && oip->sopt ? " / " : "",
335 oip->lopt ? "--" : "", oip->lopt ? oip->lopt : "");
336}
337
d2f6d877 338void opt_do(const struct optioninfo *oip, int invert,
339 const char *arg, const char *arg2) {
aa1d7195 340 switch (oip->type) {
196e9104 341 case ot_flag:
342 assert(!arg);
caa16c6a 343 *oip->storep= !invert;
196e9104 344 return;
345 case ot_value:
aa1d7195 346 assert(!arg);
196e9104 347 if (invert) noninvert(oip);
aa1d7195 348 *oip->storep= oip->value;
349 return;
d2f6d877 350 case ot_func: case ot_funcarg: case ot_funcarg2:
196e9104 351 if (invert) noninvert(oip);
d2f6d877 352 oip->func(oip,arg,arg2);
aa1d7195 353 return;
354 default:
355 abort();
356 }
357}