getent.c revision 1.2 1 1.2 lukem /* $NetBSD: getent.c,v 1.2 2004/11/26 04:52:45 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.2 lukem __RCSID("$NetBSD: getent.c,v 1.2 2004/11/26 04:52:45 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.1 lukem static int shells(int, char *[]);
70 1.1 lukem
71 1.1 lukem enum {
72 1.1 lukem RV_OK = 0,
73 1.1 lukem RV_USAGE = 1,
74 1.1 lukem RV_NOTFOUND = 2,
75 1.1 lukem RV_NOENUM = 3,
76 1.1 lukem };
77 1.1 lukem
78 1.1 lukem int
79 1.1 lukem main(int argc, char *argv[])
80 1.1 lukem {
81 1.1 lukem static struct {
82 1.1 lukem const char *name;
83 1.1 lukem int (*callback)(int, char *[]);
84 1.1 lukem } *curdb, dbs[] = {
85 1.1 lukem { "group", group, },
86 1.1 lukem { "hosts", hosts, },
87 1.2 lukem { "networks", networks, },
88 1.1 lukem { "passwd", passwd, },
89 1.1 lukem { "shells", shells, },
90 1.1 lukem
91 1.1 lukem { NULL, NULL, },
92 1.1 lukem };
93 1.1 lukem
94 1.1 lukem setprogname(argv[0]);
95 1.1 lukem
96 1.1 lukem if (argc < 2)
97 1.1 lukem usage();
98 1.1 lukem for (curdb = dbs; curdb->name != NULL; curdb++) {
99 1.1 lukem if (strcmp(curdb->name, argv[1]) == 0) {
100 1.1 lukem exit(curdb->callback(argc, argv));
101 1.1 lukem break;
102 1.1 lukem }
103 1.1 lukem }
104 1.1 lukem fprintf(stderr, "Unknown database: %s\n", argv[1]);
105 1.1 lukem usage();
106 1.1 lukem /* NOTREACHED */
107 1.1 lukem return RV_USAGE;
108 1.1 lukem }
109 1.1 lukem
110 1.1 lukem static int
111 1.1 lukem usage(void)
112 1.1 lukem {
113 1.1 lukem
114 1.1 lukem fprintf(stderr, "Usage: %s database [key ...]\n",
115 1.1 lukem getprogname());
116 1.1 lukem exit(RV_USAGE);
117 1.1 lukem /* NOTREACHED */
118 1.1 lukem }
119 1.1 lukem
120 1.1 lukem static int
121 1.1 lukem parsenum(const char *word, unsigned long *result)
122 1.1 lukem {
123 1.1 lukem unsigned long num;
124 1.1 lukem char *ep;
125 1.1 lukem
126 1.1 lukem assert(word != NULL);
127 1.1 lukem assert(result != NULL);
128 1.1 lukem
129 1.1 lukem if (!isdigit((unsigned char)word[0]))
130 1.1 lukem return 0;
131 1.1 lukem errno = 0;
132 1.1 lukem num = strtoul(word, &ep, 10);
133 1.1 lukem if (num == ULONG_MAX && errno == ERANGE)
134 1.1 lukem return 0;
135 1.1 lukem if (*ep != '\0')
136 1.1 lukem return 0;
137 1.1 lukem *result = num;
138 1.1 lukem return 1;
139 1.1 lukem }
140 1.1 lukem
141 1.1 lukem
142 1.1 lukem /*
143 1.1 lukem * group
144 1.1 lukem */
145 1.1 lukem
146 1.1 lukem static void
147 1.1 lukem groupprint(const struct group *gr)
148 1.1 lukem {
149 1.1 lukem char prefix;
150 1.1 lukem int i;
151 1.1 lukem
152 1.1 lukem assert(gr != NULL);
153 1.1 lukem printf("%s:%s:%u",
154 1.1 lukem gr->gr_name, gr->gr_passwd, gr->gr_gid);
155 1.1 lukem prefix = ':';
156 1.1 lukem for (i = 0; gr->gr_mem[i] != NULL; i++) {
157 1.1 lukem printf("%c%s", prefix, gr->gr_mem[i]);
158 1.1 lukem prefix = ',';
159 1.1 lukem }
160 1.1 lukem printf("\n");
161 1.1 lukem }
162 1.1 lukem
163 1.1 lukem static int
164 1.1 lukem group(int argc, char *argv[])
165 1.1 lukem {
166 1.1 lukem struct group *gr;
167 1.1 lukem unsigned long id;
168 1.1 lukem int i, rv;
169 1.1 lukem
170 1.1 lukem assert(argc > 1);
171 1.1 lukem assert(argv != NULL);
172 1.1 lukem
173 1.1 lukem setgroupent(1);
174 1.1 lukem rv = RV_OK;
175 1.1 lukem if (argc == 2) {
176 1.1 lukem while ((gr = getgrent()) != NULL)
177 1.1 lukem groupprint(gr);
178 1.1 lukem } else {
179 1.1 lukem for (i = 2; i < argc; i++) {
180 1.1 lukem if (parsenum(argv[i], &id))
181 1.1 lukem gr = getgrgid((gid_t)id);
182 1.1 lukem else
183 1.1 lukem gr = getgrnam(argv[i]);
184 1.1 lukem if (gr != NULL)
185 1.1 lukem groupprint(gr);
186 1.1 lukem else {
187 1.1 lukem rv = RV_NOTFOUND;
188 1.1 lukem break;
189 1.1 lukem }
190 1.1 lukem }
191 1.1 lukem }
192 1.1 lukem endgrent();
193 1.1 lukem return rv;
194 1.1 lukem }
195 1.1 lukem
196 1.1 lukem /*
197 1.1 lukem * hosts
198 1.1 lukem */
199 1.1 lukem
200 1.1 lukem static void
201 1.1 lukem hostsprint(const struct hostent *he)
202 1.1 lukem {
203 1.1 lukem char buf[INET6_ADDRSTRLEN];
204 1.1 lukem int i;
205 1.1 lukem
206 1.1 lukem assert(he != NULL);
207 1.1 lukem if (inet_ntop(he->h_addrtype, he->h_addr, buf, sizeof(buf)) == NULL)
208 1.1 lukem strlcpy(buf, "# unknown", sizeof(buf));
209 1.1 lukem printf("%s\t%s",
210 1.1 lukem buf, he->h_name);
211 1.1 lukem for (i = 0; he->h_aliases[i] != NULL; i++) {
212 1.1 lukem printf(" %s", he->h_aliases[i]);
213 1.1 lukem }
214 1.1 lukem printf("\n");
215 1.1 lukem }
216 1.1 lukem
217 1.1 lukem static int
218 1.1 lukem hosts(int argc, char *argv[])
219 1.1 lukem {
220 1.1 lukem struct hostent *he;
221 1.1 lukem char addr[IN6ADDRSZ];
222 1.1 lukem int i, rv;
223 1.1 lukem
224 1.1 lukem assert(argc > 1);
225 1.1 lukem assert(argv != NULL);
226 1.1 lukem
227 1.1 lukem sethostent(1);
228 1.1 lukem rv = RV_OK;
229 1.1 lukem if (argc == 2) {
230 1.1 lukem while ((he = gethostent()) != NULL)
231 1.1 lukem hostsprint(he);
232 1.1 lukem } else {
233 1.1 lukem for (i = 2; i < argc; i++) {
234 1.1 lukem if (inet_pton(AF_INET6, argv[i], (void *)addr) > 0)
235 1.1 lukem he = gethostbyaddr(addr, IN6ADDRSZ, AF_INET6);
236 1.1 lukem else if (inet_pton(AF_INET, argv[i], (void *)addr) > 0)
237 1.1 lukem he = gethostbyaddr(addr, INADDRSZ, AF_INET);
238 1.1 lukem else
239 1.1 lukem he = gethostbyname(argv[i]);
240 1.1 lukem if (he != NULL)
241 1.1 lukem hostsprint(he);
242 1.1 lukem else {
243 1.1 lukem rv = RV_NOTFOUND;
244 1.1 lukem break;
245 1.1 lukem }
246 1.1 lukem }
247 1.1 lukem }
248 1.1 lukem endhostent();
249 1.1 lukem return rv;
250 1.1 lukem }
251 1.1 lukem
252 1.2 lukem
253 1.2 lukem /*
254 1.2 lukem * networks
255 1.2 lukem */
256 1.2 lukem
257 1.2 lukem static void
258 1.2 lukem networksprint(const struct netent *ne)
259 1.2 lukem {
260 1.2 lukem char buf[INET6_ADDRSTRLEN];
261 1.2 lukem struct in_addr ianet;
262 1.2 lukem int i;
263 1.2 lukem char prefix;
264 1.2 lukem
265 1.2 lukem assert(ne != NULL);
266 1.2 lukem ianet = inet_makeaddr(ne->n_net, 0);
267 1.2 lukem if (inet_ntop(ne->n_addrtype, &ianet, buf, sizeof(buf)) == NULL)
268 1.2 lukem strlcpy(buf, "# unknown", sizeof(buf));
269 1.2 lukem printf("%s\t%s",
270 1.2 lukem ne->n_name, buf);
271 1.2 lukem prefix = '\t';
272 1.2 lukem for (i = 0; ne->n_aliases[i] != NULL; i++) {
273 1.2 lukem printf("%c%s", prefix, ne->n_aliases[i]);
274 1.2 lukem prefix = ' ';
275 1.2 lukem }
276 1.2 lukem printf("\n");
277 1.2 lukem }
278 1.2 lukem
279 1.2 lukem static int
280 1.2 lukem networks(int argc, char *argv[])
281 1.2 lukem {
282 1.2 lukem struct netent *ne;
283 1.2 lukem in_addr_t net;
284 1.2 lukem int i, rv;
285 1.2 lukem
286 1.2 lukem assert(argc > 1);
287 1.2 lukem assert(argv != NULL);
288 1.2 lukem
289 1.2 lukem setnetent(1);
290 1.2 lukem rv = RV_OK;
291 1.2 lukem if (argc == 2) {
292 1.2 lukem while ((ne = getnetent()) != NULL)
293 1.2 lukem networksprint(ne);
294 1.2 lukem } else {
295 1.2 lukem for (i = 2; i < argc; i++) {
296 1.2 lukem net = inet_network(argv[i]);
297 1.2 lukem if (net != INADDR_NONE)
298 1.2 lukem ne = getnetbyaddr(net, AF_INET);
299 1.2 lukem else
300 1.2 lukem ne = getnetbyname(argv[i]);
301 1.2 lukem if (ne != NULL)
302 1.2 lukem networksprint(ne);
303 1.2 lukem else {
304 1.2 lukem rv = RV_NOTFOUND;
305 1.2 lukem break;
306 1.2 lukem }
307 1.2 lukem }
308 1.2 lukem }
309 1.2 lukem endnetent();
310 1.2 lukem return rv;
311 1.2 lukem }
312 1.2 lukem
313 1.1 lukem /*
314 1.1 lukem * passwd
315 1.1 lukem */
316 1.1 lukem
317 1.1 lukem static void
318 1.1 lukem passwdprint(const struct passwd *pw)
319 1.1 lukem {
320 1.1 lukem
321 1.1 lukem assert(pw != NULL);
322 1.1 lukem printf("%s:%s:%u:%u:%s:%s:%s\n",
323 1.1 lukem pw->pw_name, pw->pw_passwd, pw->pw_uid, pw->pw_gid,
324 1.1 lukem pw->pw_gecos, pw->pw_dir, pw->pw_shell);
325 1.1 lukem }
326 1.1 lukem
327 1.1 lukem static int
328 1.1 lukem passwd(int argc, char *argv[])
329 1.1 lukem {
330 1.1 lukem struct passwd *pw;
331 1.1 lukem unsigned long id;
332 1.1 lukem int i, rv;
333 1.1 lukem
334 1.1 lukem assert(argc > 1);
335 1.1 lukem assert(argv != NULL);
336 1.1 lukem
337 1.1 lukem setpassent(1);
338 1.1 lukem rv = RV_OK;
339 1.1 lukem if (argc == 2) {
340 1.1 lukem while ((pw = getpwent()) != NULL)
341 1.1 lukem passwdprint(pw);
342 1.1 lukem } else {
343 1.1 lukem for (i = 2; i < argc; i++) {
344 1.1 lukem if (parsenum(argv[i], &id))
345 1.1 lukem pw = getpwuid((uid_t)id);
346 1.1 lukem else
347 1.1 lukem pw = getpwnam(argv[i]);
348 1.1 lukem if (pw != NULL)
349 1.1 lukem passwdprint(pw);
350 1.1 lukem else {
351 1.1 lukem rv = RV_NOTFOUND;
352 1.1 lukem break;
353 1.1 lukem }
354 1.1 lukem }
355 1.1 lukem }
356 1.1 lukem endpwent();
357 1.1 lukem return rv;
358 1.1 lukem }
359 1.1 lukem
360 1.1 lukem /*
361 1.1 lukem * shells
362 1.1 lukem */
363 1.1 lukem
364 1.1 lukem static int
365 1.1 lukem shells(int argc, char *argv[])
366 1.1 lukem {
367 1.1 lukem const char *sh;
368 1.1 lukem int i, rv;
369 1.1 lukem
370 1.1 lukem assert(argc > 1);
371 1.1 lukem assert(argv != NULL);
372 1.1 lukem
373 1.1 lukem setusershell();
374 1.1 lukem rv = RV_OK;
375 1.1 lukem if (argc == 2) {
376 1.1 lukem while ((sh = getusershell()) != NULL)
377 1.1 lukem printf("%s\n", sh);
378 1.1 lukem } else {
379 1.1 lukem for (i = 2; i < argc; i++) {
380 1.1 lukem setusershell();
381 1.1 lukem while ((sh = getusershell()) != NULL) {
382 1.1 lukem if (strcmp(sh, argv[i]) == 0) {
383 1.1 lukem printf("%s\n", sh);
384 1.1 lukem break;
385 1.1 lukem }
386 1.1 lukem }
387 1.1 lukem if (sh == NULL) {
388 1.1 lukem rv = RV_NOTFOUND;
389 1.1 lukem break;
390 1.1 lukem }
391 1.1 lukem }
392 1.1 lukem }
393 1.1 lukem endusershell();
394 1.1 lukem return rv;
395 1.1 lukem }
396