getent.c revision 1.5 1 1.5 lukem /* $NetBSD: getent.c,v 1.5 2004/11/29 05:02:40 lukem Exp $ */
2 1.1 lukem
3 1.1 lukem /*-
4 1.1 lukem * Copyright (c) 2004 The NetBSD Foundation, Inc.
5 1.1 lukem * All rights reserved.
6 1.1 lukem *
7 1.1 lukem * This code is derived from software contributed to The NetBSD Foundation
8 1.1 lukem * by Luke Mewburn.
9 1.1 lukem *
10 1.1 lukem * Redistribution and use in source and binary forms, with or without
11 1.1 lukem * modification, are permitted provided that the following conditions
12 1.1 lukem * are met:
13 1.1 lukem * 1. Redistributions of source code must retain the above copyright
14 1.1 lukem * notice, this list of conditions and the following disclaimer.
15 1.1 lukem * 2. Redistributions in binary form must reproduce the above copyright
16 1.1 lukem * notice, this list of conditions and the following disclaimer in the
17 1.1 lukem * documentation and/or other materials provided with the distribution.
18 1.1 lukem * 3. All advertising materials mentioning features or use of this software
19 1.1 lukem * must display the following acknowledgement:
20 1.1 lukem * This product includes software developed by the NetBSD
21 1.1 lukem * Foundation, Inc. and its contributors.
22 1.1 lukem * 4. Neither the name of The NetBSD Foundation nor the names of its
23 1.1 lukem * contributors may be used to endorse or promote products derived
24 1.1 lukem * from this software without specific prior written permission.
25 1.1 lukem *
26 1.1 lukem * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
27 1.1 lukem * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
28 1.1 lukem * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
29 1.1 lukem * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
30 1.1 lukem * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
31 1.1 lukem * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
32 1.1 lukem * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
33 1.1 lukem * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
34 1.1 lukem * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
35 1.1 lukem * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
36 1.1 lukem * POSSIBILITY OF SUCH DAMAGE.
37 1.1 lukem */
38 1.1 lukem
39 1.1 lukem #include <sys/cdefs.h>
40 1.1 lukem #ifndef lint
41 1.5 lukem __RCSID("$NetBSD: getent.c,v 1.5 2004/11/29 05:02:40 lukem Exp $");
42 1.1 lukem #endif /* not lint */
43 1.1 lukem
44 1.1 lukem #include <sys/socket.h>
45 1.1 lukem
46 1.1 lukem #include <assert.h>
47 1.1 lukem #include <ctype.h>
48 1.1 lukem #include <errno.h>
49 1.1 lukem #include <grp.h>
50 1.1 lukem #include <limits.h>
51 1.1 lukem #include <netdb.h>
52 1.1 lukem #include <pwd.h>
53 1.1 lukem #include <stdio.h>
54 1.5 lukem #include <stdarg.h>
55 1.1 lukem #include <stdlib.h>
56 1.1 lukem #include <string.h>
57 1.1 lukem #include <unistd.h>
58 1.1 lukem
59 1.1 lukem #include <arpa/inet.h>
60 1.1 lukem #include <arpa/nameser.h>
61 1.1 lukem
62 1.1 lukem #include <netinet/in.h> /* for INET6_ADDRSTRLEN */
63 1.1 lukem
64 1.1 lukem static int usage(void);
65 1.1 lukem static int parsenum(const char *, unsigned long *);
66 1.1 lukem static int group(int, char *[]);
67 1.1 lukem static int hosts(int, char *[]);
68 1.2 lukem static int networks(int, char *[]);
69 1.1 lukem static int passwd(int, char *[]);
70 1.3 lukem static int protocols(int, char *[]);
71 1.4 lukem static int services(int, char *[]);
72 1.1 lukem static int shells(int, char *[]);
73 1.1 lukem
74 1.1 lukem enum {
75 1.1 lukem RV_OK = 0,
76 1.1 lukem RV_USAGE = 1,
77 1.1 lukem RV_NOTFOUND = 2,
78 1.1 lukem RV_NOENUM = 3,
79 1.1 lukem };
80 1.1 lukem
81 1.5 lukem static struct getentdb {
82 1.5 lukem const char *name;
83 1.5 lukem int (*callback)(int, char *[]);
84 1.5 lukem } databases[] = {
85 1.5 lukem { "group", group, },
86 1.5 lukem { "hosts", hosts, },
87 1.5 lukem { "networks", networks, },
88 1.5 lukem { "passwd", passwd, },
89 1.5 lukem { "protocols", protocols, },
90 1.5 lukem { "services", services, },
91 1.5 lukem { "shells", shells, },
92 1.5 lukem
93 1.5 lukem { NULL, NULL, },
94 1.5 lukem };
95 1.5 lukem
96 1.5 lukem
97 1.1 lukem int
98 1.1 lukem main(int argc, char *argv[])
99 1.1 lukem {
100 1.5 lukem struct getentdb *curdb;
101 1.1 lukem
102 1.1 lukem setprogname(argv[0]);
103 1.1 lukem
104 1.1 lukem if (argc < 2)
105 1.1 lukem usage();
106 1.5 lukem for (curdb = databases; curdb->name != NULL; curdb++) {
107 1.1 lukem if (strcmp(curdb->name, argv[1]) == 0) {
108 1.1 lukem exit(curdb->callback(argc, argv));
109 1.1 lukem break;
110 1.1 lukem }
111 1.1 lukem }
112 1.1 lukem fprintf(stderr, "Unknown database: %s\n", argv[1]);
113 1.1 lukem usage();
114 1.1 lukem /* NOTREACHED */
115 1.1 lukem return RV_USAGE;
116 1.1 lukem }
117 1.1 lukem
118 1.1 lukem static int
119 1.1 lukem usage(void)
120 1.1 lukem {
121 1.5 lukem struct getentdb *curdb;
122 1.1 lukem
123 1.1 lukem fprintf(stderr, "Usage: %s database [key ...]\n",
124 1.1 lukem getprogname());
125 1.5 lukem fprintf(stderr, " database may be one of:\n\t");
126 1.5 lukem for (curdb = databases; curdb->name != NULL; curdb++) {
127 1.5 lukem fprintf(stderr, " %s", curdb->name);
128 1.5 lukem }
129 1.5 lukem fprintf(stderr, "\n");
130 1.1 lukem exit(RV_USAGE);
131 1.1 lukem /* NOTREACHED */
132 1.1 lukem }
133 1.1 lukem
134 1.1 lukem static int
135 1.1 lukem parsenum(const char *word, unsigned long *result)
136 1.1 lukem {
137 1.1 lukem unsigned long num;
138 1.1 lukem char *ep;
139 1.1 lukem
140 1.1 lukem assert(word != NULL);
141 1.1 lukem assert(result != NULL);
142 1.1 lukem
143 1.1 lukem if (!isdigit((unsigned char)word[0]))
144 1.1 lukem return 0;
145 1.1 lukem errno = 0;
146 1.1 lukem num = strtoul(word, &ep, 10);
147 1.1 lukem if (num == ULONG_MAX && errno == ERANGE)
148 1.1 lukem return 0;
149 1.1 lukem if (*ep != '\0')
150 1.1 lukem return 0;
151 1.1 lukem *result = num;
152 1.1 lukem return 1;
153 1.1 lukem }
154 1.1 lukem
155 1.5 lukem /*
156 1.5 lukem * printfmtstrings --
157 1.5 lukem * vprintf(format, ...),
158 1.5 lukem * then the aliases (beginning with prefix, separated by sep),
159 1.5 lukem * then a newline
160 1.5 lukem */
161 1.5 lukem static void
162 1.5 lukem printfmtstrings(char *strings[], const char *prefix, const char *sep,
163 1.5 lukem const char *fmt, ...)
164 1.5 lukem {
165 1.5 lukem va_list ap;
166 1.5 lukem const char *curpref;
167 1.5 lukem int i;
168 1.1 lukem
169 1.5 lukem va_start(ap, fmt);
170 1.5 lukem vprintf(fmt, ap);
171 1.1 lukem
172 1.5 lukem curpref = prefix;
173 1.5 lukem for (i = 0; strings[i] != NULL; i++) {
174 1.5 lukem printf("%s%s", curpref, strings[i]);
175 1.5 lukem curpref = sep;
176 1.1 lukem }
177 1.1 lukem printf("\n");
178 1.1 lukem }
179 1.1 lukem
180 1.5 lukem
181 1.5 lukem /*
182 1.5 lukem * group
183 1.5 lukem */
184 1.5 lukem
185 1.1 lukem static int
186 1.1 lukem group(int argc, char *argv[])
187 1.1 lukem {
188 1.1 lukem struct group *gr;
189 1.1 lukem unsigned long id;
190 1.1 lukem int i, rv;
191 1.1 lukem
192 1.1 lukem assert(argc > 1);
193 1.1 lukem assert(argv != NULL);
194 1.1 lukem
195 1.5 lukem #define GROUPPRINT printfmtstrings(gr->gr_mem, ":", ",", "%s:%s:%u", \
196 1.5 lukem gr->gr_name, gr->gr_passwd, gr->gr_gid)
197 1.5 lukem
198 1.1 lukem setgroupent(1);
199 1.1 lukem rv = RV_OK;
200 1.1 lukem if (argc == 2) {
201 1.1 lukem while ((gr = getgrent()) != NULL)
202 1.5 lukem GROUPPRINT;
203 1.1 lukem } else {
204 1.1 lukem for (i = 2; i < argc; i++) {
205 1.1 lukem if (parsenum(argv[i], &id))
206 1.1 lukem gr = getgrgid((gid_t)id);
207 1.1 lukem else
208 1.1 lukem gr = getgrnam(argv[i]);
209 1.1 lukem if (gr != NULL)
210 1.5 lukem GROUPPRINT;
211 1.1 lukem else {
212 1.1 lukem rv = RV_NOTFOUND;
213 1.1 lukem break;
214 1.1 lukem }
215 1.1 lukem }
216 1.1 lukem }
217 1.1 lukem endgrent();
218 1.1 lukem return rv;
219 1.1 lukem }
220 1.1 lukem
221 1.3 lukem
222 1.1 lukem /*
223 1.1 lukem * hosts
224 1.1 lukem */
225 1.1 lukem
226 1.1 lukem static void
227 1.1 lukem hostsprint(const struct hostent *he)
228 1.1 lukem {
229 1.1 lukem char buf[INET6_ADDRSTRLEN];
230 1.1 lukem
231 1.1 lukem assert(he != NULL);
232 1.1 lukem if (inet_ntop(he->h_addrtype, he->h_addr, buf, sizeof(buf)) == NULL)
233 1.1 lukem strlcpy(buf, "# unknown", sizeof(buf));
234 1.5 lukem printfmtstrings(he->h_aliases, " ", " ", "%-16s %s", buf, he->h_name);
235 1.1 lukem }
236 1.1 lukem
237 1.1 lukem static int
238 1.1 lukem hosts(int argc, char *argv[])
239 1.1 lukem {
240 1.1 lukem struct hostent *he;
241 1.1 lukem char addr[IN6ADDRSZ];
242 1.1 lukem int i, rv;
243 1.1 lukem
244 1.1 lukem assert(argc > 1);
245 1.1 lukem assert(argv != NULL);
246 1.1 lukem
247 1.1 lukem sethostent(1);
248 1.1 lukem rv = RV_OK;
249 1.1 lukem if (argc == 2) {
250 1.1 lukem while ((he = gethostent()) != NULL)
251 1.1 lukem hostsprint(he);
252 1.1 lukem } else {
253 1.1 lukem for (i = 2; i < argc; i++) {
254 1.1 lukem if (inet_pton(AF_INET6, argv[i], (void *)addr) > 0)
255 1.1 lukem he = gethostbyaddr(addr, IN6ADDRSZ, AF_INET6);
256 1.1 lukem else if (inet_pton(AF_INET, argv[i], (void *)addr) > 0)
257 1.1 lukem he = gethostbyaddr(addr, INADDRSZ, AF_INET);
258 1.1 lukem else
259 1.1 lukem he = gethostbyname(argv[i]);
260 1.1 lukem if (he != NULL)
261 1.1 lukem hostsprint(he);
262 1.1 lukem else {
263 1.1 lukem rv = RV_NOTFOUND;
264 1.1 lukem break;
265 1.1 lukem }
266 1.1 lukem }
267 1.1 lukem }
268 1.1 lukem endhostent();
269 1.1 lukem return rv;
270 1.1 lukem }
271 1.1 lukem
272 1.2 lukem
273 1.2 lukem /*
274 1.2 lukem * networks
275 1.2 lukem */
276 1.2 lukem
277 1.2 lukem static void
278 1.2 lukem networksprint(const struct netent *ne)
279 1.2 lukem {
280 1.2 lukem char buf[INET6_ADDRSTRLEN];
281 1.2 lukem struct in_addr ianet;
282 1.2 lukem
283 1.2 lukem assert(ne != NULL);
284 1.2 lukem ianet = inet_makeaddr(ne->n_net, 0);
285 1.2 lukem if (inet_ntop(ne->n_addrtype, &ianet, buf, sizeof(buf)) == NULL)
286 1.2 lukem strlcpy(buf, "# unknown", sizeof(buf));
287 1.5 lukem printfmtstrings(ne->n_aliases, " ", " ", "%-16s %s", ne->n_name, buf);
288 1.2 lukem }
289 1.2 lukem
290 1.2 lukem static int
291 1.2 lukem networks(int argc, char *argv[])
292 1.2 lukem {
293 1.2 lukem struct netent *ne;
294 1.2 lukem in_addr_t net;
295 1.2 lukem int i, rv;
296 1.2 lukem
297 1.2 lukem assert(argc > 1);
298 1.2 lukem assert(argv != NULL);
299 1.2 lukem
300 1.2 lukem setnetent(1);
301 1.2 lukem rv = RV_OK;
302 1.2 lukem if (argc == 2) {
303 1.2 lukem while ((ne = getnetent()) != NULL)
304 1.2 lukem networksprint(ne);
305 1.2 lukem } else {
306 1.2 lukem for (i = 2; i < argc; i++) {
307 1.2 lukem net = inet_network(argv[i]);
308 1.2 lukem if (net != INADDR_NONE)
309 1.2 lukem ne = getnetbyaddr(net, AF_INET);
310 1.2 lukem else
311 1.2 lukem ne = getnetbyname(argv[i]);
312 1.2 lukem if (ne != NULL)
313 1.2 lukem networksprint(ne);
314 1.2 lukem else {
315 1.2 lukem rv = RV_NOTFOUND;
316 1.2 lukem break;
317 1.2 lukem }
318 1.2 lukem }
319 1.2 lukem }
320 1.2 lukem endnetent();
321 1.2 lukem return rv;
322 1.2 lukem }
323 1.2 lukem
324 1.3 lukem
325 1.1 lukem /*
326 1.1 lukem * passwd
327 1.1 lukem */
328 1.1 lukem
329 1.1 lukem static int
330 1.1 lukem passwd(int argc, char *argv[])
331 1.1 lukem {
332 1.1 lukem struct passwd *pw;
333 1.1 lukem unsigned long id;
334 1.1 lukem int i, rv;
335 1.1 lukem
336 1.1 lukem assert(argc > 1);
337 1.1 lukem assert(argv != NULL);
338 1.1 lukem
339 1.5 lukem #define PASSWDPRINT printf("%s:%s:%u:%u:%s:%s:%s\n", \
340 1.5 lukem pw->pw_name, pw->pw_passwd, pw->pw_uid, \
341 1.5 lukem pw->pw_gid, pw->pw_gecos, pw->pw_dir, pw->pw_shell)
342 1.5 lukem
343 1.1 lukem setpassent(1);
344 1.1 lukem rv = RV_OK;
345 1.1 lukem if (argc == 2) {
346 1.1 lukem while ((pw = getpwent()) != NULL)
347 1.5 lukem PASSWDPRINT;
348 1.1 lukem } else {
349 1.1 lukem for (i = 2; i < argc; i++) {
350 1.1 lukem if (parsenum(argv[i], &id))
351 1.1 lukem pw = getpwuid((uid_t)id);
352 1.1 lukem else
353 1.1 lukem pw = getpwnam(argv[i]);
354 1.1 lukem if (pw != NULL)
355 1.5 lukem PASSWDPRINT;
356 1.1 lukem else {
357 1.1 lukem rv = RV_NOTFOUND;
358 1.1 lukem break;
359 1.1 lukem }
360 1.1 lukem }
361 1.1 lukem }
362 1.1 lukem endpwent();
363 1.1 lukem return rv;
364 1.1 lukem }
365 1.1 lukem
366 1.3 lukem
367 1.3 lukem /*
368 1.3 lukem * protocols
369 1.3 lukem */
370 1.3 lukem
371 1.3 lukem static int
372 1.3 lukem protocols(int argc, char *argv[])
373 1.3 lukem {
374 1.3 lukem struct protoent *pe;
375 1.3 lukem unsigned long id;
376 1.3 lukem int i, rv;
377 1.3 lukem
378 1.3 lukem assert(argc > 1);
379 1.3 lukem assert(argv != NULL);
380 1.3 lukem
381 1.5 lukem #define PROTOCOLSPRINT printfmtstrings(pe->p_aliases, " ", " ", \
382 1.5 lukem "%-16s %5d", pe->p_name, pe->p_proto)
383 1.5 lukem
384 1.3 lukem setprotoent(1);
385 1.3 lukem rv = RV_OK;
386 1.3 lukem if (argc == 2) {
387 1.3 lukem while ((pe = getprotoent()) != NULL)
388 1.5 lukem PROTOCOLSPRINT;
389 1.3 lukem } else {
390 1.3 lukem for (i = 2; i < argc; i++) {
391 1.3 lukem if (parsenum(argv[i], &id))
392 1.3 lukem pe = getprotobynumber((int)id);
393 1.3 lukem else
394 1.3 lukem pe = getprotobyname(argv[i]);
395 1.3 lukem if (pe != NULL)
396 1.5 lukem PROTOCOLSPRINT;
397 1.3 lukem else {
398 1.3 lukem rv = RV_NOTFOUND;
399 1.3 lukem break;
400 1.3 lukem }
401 1.3 lukem }
402 1.3 lukem }
403 1.3 lukem endprotoent();
404 1.3 lukem return rv;
405 1.3 lukem }
406 1.3 lukem
407 1.3 lukem
408 1.1 lukem /*
409 1.4 lukem * services
410 1.4 lukem */
411 1.4 lukem
412 1.4 lukem static int
413 1.4 lukem services(int argc, char *argv[])
414 1.4 lukem {
415 1.4 lukem struct servent *se;
416 1.4 lukem unsigned long id;
417 1.4 lukem char *proto;
418 1.4 lukem int i, rv;
419 1.4 lukem
420 1.4 lukem assert(argc > 1);
421 1.4 lukem assert(argv != NULL);
422 1.4 lukem
423 1.5 lukem #define SERVICESPRINT printfmtstrings(se->s_aliases, " ", " ", \
424 1.5 lukem "%-16s %5d/%s", \
425 1.5 lukem se->s_name, ntohs(se->s_port), se->s_proto)
426 1.5 lukem
427 1.4 lukem setservent(1);
428 1.4 lukem rv = RV_OK;
429 1.4 lukem if (argc == 2) {
430 1.4 lukem while ((se = getservent()) != NULL)
431 1.5 lukem SERVICESPRINT;
432 1.4 lukem } else {
433 1.4 lukem for (i = 2; i < argc; i++) {
434 1.4 lukem proto = strchr(argv[i], '/');
435 1.4 lukem if (proto != NULL)
436 1.4 lukem *proto++ = '\0';
437 1.4 lukem if (parsenum(argv[i], &id))
438 1.4 lukem se = getservbyport((int)id, proto);
439 1.4 lukem else
440 1.4 lukem se = getservbyname(argv[i], proto);
441 1.4 lukem if (se != NULL)
442 1.5 lukem SERVICESPRINT;
443 1.4 lukem else {
444 1.4 lukem rv = RV_NOTFOUND;
445 1.4 lukem break;
446 1.4 lukem }
447 1.4 lukem }
448 1.4 lukem }
449 1.4 lukem endservent();
450 1.4 lukem return rv;
451 1.4 lukem }
452 1.4 lukem
453 1.4 lukem
454 1.4 lukem /*
455 1.1 lukem * shells
456 1.1 lukem */
457 1.1 lukem
458 1.1 lukem static int
459 1.1 lukem shells(int argc, char *argv[])
460 1.1 lukem {
461 1.1 lukem const char *sh;
462 1.1 lukem int i, rv;
463 1.1 lukem
464 1.1 lukem assert(argc > 1);
465 1.1 lukem assert(argv != NULL);
466 1.1 lukem
467 1.5 lukem #define SHELLSPRINT printf("%s\n", sh)
468 1.5 lukem
469 1.1 lukem setusershell();
470 1.1 lukem rv = RV_OK;
471 1.1 lukem if (argc == 2) {
472 1.1 lukem while ((sh = getusershell()) != NULL)
473 1.5 lukem SHELLSPRINT;
474 1.1 lukem } else {
475 1.1 lukem for (i = 2; i < argc; i++) {
476 1.1 lukem setusershell();
477 1.1 lukem while ((sh = getusershell()) != NULL) {
478 1.1 lukem if (strcmp(sh, argv[i]) == 0) {
479 1.5 lukem SHELLSPRINT;
480 1.1 lukem break;
481 1.1 lukem }
482 1.1 lukem }
483 1.1 lukem if (sh == NULL) {
484 1.1 lukem rv = RV_NOTFOUND;
485 1.1 lukem break;
486 1.1 lukem }
487 1.1 lukem }
488 1.1 lukem }
489 1.1 lukem endusershell();
490 1.1 lukem return rv;
491 1.1 lukem }
492