getent.c revision 1.4 1 1.4 lukem /* $NetBSD: getent.c,v 1.4 2004/11/29 04:13:15 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.4 lukem __RCSID("$NetBSD: getent.c,v 1.4 2004/11/29 04:13:15 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.1 lukem #include <stdlib.h>
55 1.1 lukem #include <string.h>
56 1.1 lukem #include <unistd.h>
57 1.1 lukem
58 1.1 lukem #include <arpa/inet.h>
59 1.1 lukem #include <arpa/nameser.h>
60 1.1 lukem
61 1.1 lukem #include <netinet/in.h> /* for INET6_ADDRSTRLEN */
62 1.1 lukem
63 1.1 lukem static int usage(void);
64 1.1 lukem static int parsenum(const char *, unsigned long *);
65 1.1 lukem static int group(int, char *[]);
66 1.1 lukem static int hosts(int, char *[]);
67 1.2 lukem static int networks(int, char *[]);
68 1.1 lukem static int passwd(int, char *[]);
69 1.3 lukem static int protocols(int, char *[]);
70 1.4 lukem static int services(int, char *[]);
71 1.1 lukem static int shells(int, char *[]);
72 1.1 lukem
73 1.1 lukem enum {
74 1.1 lukem RV_OK = 0,
75 1.1 lukem RV_USAGE = 1,
76 1.1 lukem RV_NOTFOUND = 2,
77 1.1 lukem RV_NOENUM = 3,
78 1.1 lukem };
79 1.1 lukem
80 1.1 lukem int
81 1.1 lukem main(int argc, char *argv[])
82 1.1 lukem {
83 1.1 lukem static struct {
84 1.1 lukem const char *name;
85 1.1 lukem int (*callback)(int, char *[]);
86 1.1 lukem } *curdb, dbs[] = {
87 1.1 lukem { "group", group, },
88 1.1 lukem { "hosts", hosts, },
89 1.2 lukem { "networks", networks, },
90 1.1 lukem { "passwd", passwd, },
91 1.3 lukem { "protocols", protocols, },
92 1.4 lukem { "services", services, },
93 1.1 lukem { "shells", shells, },
94 1.1 lukem
95 1.1 lukem { NULL, NULL, },
96 1.1 lukem };
97 1.1 lukem
98 1.1 lukem setprogname(argv[0]);
99 1.1 lukem
100 1.1 lukem if (argc < 2)
101 1.1 lukem usage();
102 1.1 lukem for (curdb = dbs; curdb->name != NULL; curdb++) {
103 1.1 lukem if (strcmp(curdb->name, argv[1]) == 0) {
104 1.1 lukem exit(curdb->callback(argc, argv));
105 1.1 lukem break;
106 1.1 lukem }
107 1.1 lukem }
108 1.1 lukem fprintf(stderr, "Unknown database: %s\n", argv[1]);
109 1.1 lukem usage();
110 1.1 lukem /* NOTREACHED */
111 1.1 lukem return RV_USAGE;
112 1.1 lukem }
113 1.1 lukem
114 1.1 lukem static int
115 1.1 lukem usage(void)
116 1.1 lukem {
117 1.1 lukem
118 1.1 lukem fprintf(stderr, "Usage: %s database [key ...]\n",
119 1.1 lukem getprogname());
120 1.1 lukem exit(RV_USAGE);
121 1.1 lukem /* NOTREACHED */
122 1.1 lukem }
123 1.1 lukem
124 1.1 lukem static int
125 1.1 lukem parsenum(const char *word, unsigned long *result)
126 1.1 lukem {
127 1.1 lukem unsigned long num;
128 1.1 lukem char *ep;
129 1.1 lukem
130 1.1 lukem assert(word != NULL);
131 1.1 lukem assert(result != NULL);
132 1.1 lukem
133 1.1 lukem if (!isdigit((unsigned char)word[0]))
134 1.1 lukem return 0;
135 1.1 lukem errno = 0;
136 1.1 lukem num = strtoul(word, &ep, 10);
137 1.1 lukem if (num == ULONG_MAX && errno == ERANGE)
138 1.1 lukem return 0;
139 1.1 lukem if (*ep != '\0')
140 1.1 lukem return 0;
141 1.1 lukem *result = num;
142 1.1 lukem return 1;
143 1.1 lukem }
144 1.1 lukem
145 1.1 lukem
146 1.1 lukem /*
147 1.1 lukem * group
148 1.1 lukem */
149 1.1 lukem
150 1.1 lukem static void
151 1.1 lukem groupprint(const struct group *gr)
152 1.1 lukem {
153 1.1 lukem char prefix;
154 1.1 lukem int i;
155 1.1 lukem
156 1.1 lukem assert(gr != NULL);
157 1.1 lukem printf("%s:%s:%u",
158 1.1 lukem gr->gr_name, gr->gr_passwd, gr->gr_gid);
159 1.1 lukem prefix = ':';
160 1.1 lukem for (i = 0; gr->gr_mem[i] != NULL; i++) {
161 1.1 lukem printf("%c%s", prefix, gr->gr_mem[i]);
162 1.1 lukem prefix = ',';
163 1.1 lukem }
164 1.1 lukem printf("\n");
165 1.1 lukem }
166 1.1 lukem
167 1.1 lukem static int
168 1.1 lukem group(int argc, char *argv[])
169 1.1 lukem {
170 1.1 lukem struct group *gr;
171 1.1 lukem unsigned long id;
172 1.1 lukem int i, rv;
173 1.1 lukem
174 1.1 lukem assert(argc > 1);
175 1.1 lukem assert(argv != NULL);
176 1.1 lukem
177 1.1 lukem setgroupent(1);
178 1.1 lukem rv = RV_OK;
179 1.1 lukem if (argc == 2) {
180 1.1 lukem while ((gr = getgrent()) != NULL)
181 1.1 lukem groupprint(gr);
182 1.1 lukem } else {
183 1.1 lukem for (i = 2; i < argc; i++) {
184 1.1 lukem if (parsenum(argv[i], &id))
185 1.1 lukem gr = getgrgid((gid_t)id);
186 1.1 lukem else
187 1.1 lukem gr = getgrnam(argv[i]);
188 1.1 lukem if (gr != NULL)
189 1.1 lukem groupprint(gr);
190 1.1 lukem else {
191 1.1 lukem rv = RV_NOTFOUND;
192 1.1 lukem break;
193 1.1 lukem }
194 1.1 lukem }
195 1.1 lukem }
196 1.1 lukem endgrent();
197 1.1 lukem return rv;
198 1.1 lukem }
199 1.1 lukem
200 1.3 lukem
201 1.1 lukem /*
202 1.1 lukem * hosts
203 1.1 lukem */
204 1.1 lukem
205 1.1 lukem static void
206 1.1 lukem hostsprint(const struct hostent *he)
207 1.1 lukem {
208 1.1 lukem char buf[INET6_ADDRSTRLEN];
209 1.1 lukem int i;
210 1.1 lukem
211 1.1 lukem assert(he != NULL);
212 1.1 lukem if (inet_ntop(he->h_addrtype, he->h_addr, buf, sizeof(buf)) == NULL)
213 1.1 lukem strlcpy(buf, "# unknown", sizeof(buf));
214 1.1 lukem printf("%s\t%s",
215 1.1 lukem buf, he->h_name);
216 1.1 lukem for (i = 0; he->h_aliases[i] != NULL; i++) {
217 1.1 lukem printf(" %s", he->h_aliases[i]);
218 1.1 lukem }
219 1.1 lukem printf("\n");
220 1.1 lukem }
221 1.1 lukem
222 1.1 lukem static int
223 1.1 lukem hosts(int argc, char *argv[])
224 1.1 lukem {
225 1.1 lukem struct hostent *he;
226 1.1 lukem char addr[IN6ADDRSZ];
227 1.1 lukem int i, rv;
228 1.1 lukem
229 1.1 lukem assert(argc > 1);
230 1.1 lukem assert(argv != NULL);
231 1.1 lukem
232 1.1 lukem sethostent(1);
233 1.1 lukem rv = RV_OK;
234 1.1 lukem if (argc == 2) {
235 1.1 lukem while ((he = gethostent()) != NULL)
236 1.1 lukem hostsprint(he);
237 1.1 lukem } else {
238 1.1 lukem for (i = 2; i < argc; i++) {
239 1.1 lukem if (inet_pton(AF_INET6, argv[i], (void *)addr) > 0)
240 1.1 lukem he = gethostbyaddr(addr, IN6ADDRSZ, AF_INET6);
241 1.1 lukem else if (inet_pton(AF_INET, argv[i], (void *)addr) > 0)
242 1.1 lukem he = gethostbyaddr(addr, INADDRSZ, AF_INET);
243 1.1 lukem else
244 1.1 lukem he = gethostbyname(argv[i]);
245 1.1 lukem if (he != NULL)
246 1.1 lukem hostsprint(he);
247 1.1 lukem else {
248 1.1 lukem rv = RV_NOTFOUND;
249 1.1 lukem break;
250 1.1 lukem }
251 1.1 lukem }
252 1.1 lukem }
253 1.1 lukem endhostent();
254 1.1 lukem return rv;
255 1.1 lukem }
256 1.1 lukem
257 1.2 lukem
258 1.2 lukem /*
259 1.2 lukem * networks
260 1.2 lukem */
261 1.2 lukem
262 1.2 lukem static void
263 1.2 lukem networksprint(const struct netent *ne)
264 1.2 lukem {
265 1.2 lukem char buf[INET6_ADDRSTRLEN];
266 1.2 lukem struct in_addr ianet;
267 1.2 lukem int i;
268 1.2 lukem char prefix;
269 1.2 lukem
270 1.2 lukem assert(ne != NULL);
271 1.2 lukem ianet = inet_makeaddr(ne->n_net, 0);
272 1.2 lukem if (inet_ntop(ne->n_addrtype, &ianet, buf, sizeof(buf)) == NULL)
273 1.2 lukem strlcpy(buf, "# unknown", sizeof(buf));
274 1.2 lukem printf("%s\t%s",
275 1.2 lukem ne->n_name, buf);
276 1.2 lukem prefix = '\t';
277 1.2 lukem for (i = 0; ne->n_aliases[i] != NULL; i++) {
278 1.2 lukem printf("%c%s", prefix, ne->n_aliases[i]);
279 1.2 lukem prefix = ' ';
280 1.2 lukem }
281 1.2 lukem printf("\n");
282 1.2 lukem }
283 1.2 lukem
284 1.2 lukem static int
285 1.2 lukem networks(int argc, char *argv[])
286 1.2 lukem {
287 1.2 lukem struct netent *ne;
288 1.2 lukem in_addr_t net;
289 1.2 lukem int i, rv;
290 1.2 lukem
291 1.2 lukem assert(argc > 1);
292 1.2 lukem assert(argv != NULL);
293 1.2 lukem
294 1.2 lukem setnetent(1);
295 1.2 lukem rv = RV_OK;
296 1.2 lukem if (argc == 2) {
297 1.2 lukem while ((ne = getnetent()) != NULL)
298 1.2 lukem networksprint(ne);
299 1.2 lukem } else {
300 1.2 lukem for (i = 2; i < argc; i++) {
301 1.2 lukem net = inet_network(argv[i]);
302 1.2 lukem if (net != INADDR_NONE)
303 1.2 lukem ne = getnetbyaddr(net, AF_INET);
304 1.2 lukem else
305 1.2 lukem ne = getnetbyname(argv[i]);
306 1.2 lukem if (ne != NULL)
307 1.2 lukem networksprint(ne);
308 1.2 lukem else {
309 1.2 lukem rv = RV_NOTFOUND;
310 1.2 lukem break;
311 1.2 lukem }
312 1.2 lukem }
313 1.2 lukem }
314 1.2 lukem endnetent();
315 1.2 lukem return rv;
316 1.2 lukem }
317 1.2 lukem
318 1.3 lukem
319 1.1 lukem /*
320 1.1 lukem * passwd
321 1.1 lukem */
322 1.1 lukem
323 1.1 lukem static void
324 1.1 lukem passwdprint(const struct passwd *pw)
325 1.1 lukem {
326 1.1 lukem
327 1.1 lukem assert(pw != NULL);
328 1.1 lukem printf("%s:%s:%u:%u:%s:%s:%s\n",
329 1.1 lukem pw->pw_name, pw->pw_passwd, pw->pw_uid, pw->pw_gid,
330 1.1 lukem pw->pw_gecos, pw->pw_dir, pw->pw_shell);
331 1.1 lukem }
332 1.1 lukem
333 1.1 lukem static int
334 1.1 lukem passwd(int argc, char *argv[])
335 1.1 lukem {
336 1.1 lukem struct passwd *pw;
337 1.1 lukem unsigned long id;
338 1.1 lukem int i, rv;
339 1.1 lukem
340 1.1 lukem assert(argc > 1);
341 1.1 lukem assert(argv != NULL);
342 1.1 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.1 lukem passwdprint(pw);
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.1 lukem passwdprint(pw);
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 void
372 1.3 lukem protocolsprint(const struct protoent *pe)
373 1.3 lukem {
374 1.3 lukem int i;
375 1.3 lukem
376 1.3 lukem assert(pe != NULL);
377 1.3 lukem printf("%s\t%d", pe->p_name, pe->p_proto);
378 1.3 lukem for (i = 0; pe->p_aliases[i] != NULL; i++) {
379 1.3 lukem printf(" %s", pe->p_aliases[i]);
380 1.3 lukem }
381 1.3 lukem printf("\n");
382 1.3 lukem }
383 1.3 lukem
384 1.3 lukem static int
385 1.3 lukem protocols(int argc, char *argv[])
386 1.3 lukem {
387 1.3 lukem struct protoent *pe;
388 1.3 lukem unsigned long id;
389 1.3 lukem int i, rv;
390 1.3 lukem
391 1.3 lukem assert(argc > 1);
392 1.3 lukem assert(argv != NULL);
393 1.3 lukem
394 1.3 lukem setprotoent(1);
395 1.3 lukem rv = RV_OK;
396 1.3 lukem if (argc == 2) {
397 1.3 lukem while ((pe = getprotoent()) != NULL)
398 1.3 lukem protocolsprint(pe);
399 1.3 lukem } else {
400 1.3 lukem for (i = 2; i < argc; i++) {
401 1.3 lukem if (parsenum(argv[i], &id))
402 1.3 lukem pe = getprotobynumber((int)id);
403 1.3 lukem else
404 1.3 lukem pe = getprotobyname(argv[i]);
405 1.3 lukem if (pe != NULL)
406 1.3 lukem protocolsprint(pe);
407 1.3 lukem else {
408 1.3 lukem rv = RV_NOTFOUND;
409 1.3 lukem break;
410 1.3 lukem }
411 1.3 lukem }
412 1.3 lukem }
413 1.3 lukem endprotoent();
414 1.3 lukem return rv;
415 1.3 lukem }
416 1.3 lukem
417 1.3 lukem
418 1.1 lukem /*
419 1.4 lukem * services
420 1.4 lukem */
421 1.4 lukem
422 1.4 lukem static void
423 1.4 lukem servicesprint(const struct servent *se)
424 1.4 lukem {
425 1.4 lukem int i;
426 1.4 lukem
427 1.4 lukem assert(se != NULL);
428 1.4 lukem printf("%s\t%d/%s", se->s_name, ntohs(se->s_port), se->s_proto);
429 1.4 lukem for (i = 0; se->s_aliases[i] != NULL; i++) {
430 1.4 lukem printf(" %s", se->s_aliases[i]);
431 1.4 lukem }
432 1.4 lukem printf("\n");
433 1.4 lukem }
434 1.4 lukem
435 1.4 lukem static int
436 1.4 lukem services(int argc, char *argv[])
437 1.4 lukem {
438 1.4 lukem struct servent *se;
439 1.4 lukem unsigned long id;
440 1.4 lukem char *proto;
441 1.4 lukem int i, rv;
442 1.4 lukem
443 1.4 lukem assert(argc > 1);
444 1.4 lukem assert(argv != NULL);
445 1.4 lukem
446 1.4 lukem setservent(1);
447 1.4 lukem rv = RV_OK;
448 1.4 lukem if (argc == 2) {
449 1.4 lukem while ((se = getservent()) != NULL)
450 1.4 lukem servicesprint(se);
451 1.4 lukem } else {
452 1.4 lukem for (i = 2; i < argc; i++) {
453 1.4 lukem proto = strchr(argv[i], '/');
454 1.4 lukem if (proto != NULL)
455 1.4 lukem *proto++ = '\0';
456 1.4 lukem if (parsenum(argv[i], &id))
457 1.4 lukem se = getservbyport((int)id, proto);
458 1.4 lukem else
459 1.4 lukem se = getservbyname(argv[i], proto);
460 1.4 lukem if (se != NULL)
461 1.4 lukem servicesprint(se);
462 1.4 lukem else {
463 1.4 lukem rv = RV_NOTFOUND;
464 1.4 lukem break;
465 1.4 lukem }
466 1.4 lukem }
467 1.4 lukem }
468 1.4 lukem endservent();
469 1.4 lukem return rv;
470 1.4 lukem }
471 1.4 lukem
472 1.4 lukem
473 1.4 lukem /*
474 1.1 lukem * shells
475 1.1 lukem */
476 1.1 lukem
477 1.1 lukem static int
478 1.1 lukem shells(int argc, char *argv[])
479 1.1 lukem {
480 1.1 lukem const char *sh;
481 1.1 lukem int i, rv;
482 1.1 lukem
483 1.1 lukem assert(argc > 1);
484 1.1 lukem assert(argv != NULL);
485 1.1 lukem
486 1.1 lukem setusershell();
487 1.1 lukem rv = RV_OK;
488 1.1 lukem if (argc == 2) {
489 1.1 lukem while ((sh = getusershell()) != NULL)
490 1.1 lukem printf("%s\n", sh);
491 1.1 lukem } else {
492 1.1 lukem for (i = 2; i < argc; i++) {
493 1.1 lukem setusershell();
494 1.1 lukem while ((sh = getusershell()) != NULL) {
495 1.1 lukem if (strcmp(sh, argv[i]) == 0) {
496 1.1 lukem printf("%s\n", sh);
497 1.1 lukem break;
498 1.1 lukem }
499 1.1 lukem }
500 1.1 lukem if (sh == NULL) {
501 1.1 lukem rv = RV_NOTFOUND;
502 1.1 lukem break;
503 1.1 lukem }
504 1.1 lukem }
505 1.1 lukem }
506 1.1 lukem endusershell();
507 1.1 lukem return rv;
508 1.1 lukem }
509