de7a9611c514d6dc6c36ac0888633050da41c22e
[dragonfly.git] / usr.sbin / pw / pw_vpw.c
1 /*-
2  * Copyright (C) 1996
3  *      David L. Nugent.  All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  *
14  * THIS SOFTWARE IS PROVIDED BY DAVID L. NUGENT AND CONTRIBUTORS ``AS IS'' AND
15  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17  * ARE DISCLAIMED.  IN NO EVENT SHALL DAVID L. NUGENT OR CONTRIBUTORS BE LIABLE
18  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24  * SUCH DAMAGE.
25  *
26  * $FreeBSD: src/usr.sbin/pw/pw_vpw.c,v 1.3 2000/01/15 00:20:21 davidn Exp $
27  * $DragonFly: src/usr.sbin/pw/pw_vpw.c,v 1.4 2005/12/05 02:40:27 swildner Exp $
28  */
29
30 #include <stdio.h>
31 #include <string.h>
32 #include <stdlib.h>
33 #include <sys/param.h>
34
35 #include "pwupd.h"
36
37 static FILE * pwd_fp = NULL;
38
39 void
40 vendpwent(void)
41 {
42         if (pwd_fp != NULL) {
43                 fclose(pwd_fp);
44                 pwd_fp = NULL;
45         }
46 }
47
48 void
49 vsetpwent(void)
50 {
51         vendpwent();
52 }
53
54 static struct passwd *
55 vnextpwent(char const * nam, uid_t uid, int doclose)
56 {
57         struct passwd * pw = NULL;
58         static char pwtmp[1024];
59
60         strncpy(pwtmp, getpwpath(_MASTERPASSWD), sizeof pwtmp);
61         pwtmp[sizeof pwtmp - 1] = '\0';
62
63         if (pwd_fp != NULL || (pwd_fp = fopen(pwtmp, "r")) != NULL) {
64                 int done = 0;
65
66                 static struct passwd pwd;
67
68                 while (!done && fgets(pwtmp, sizeof pwtmp, pwd_fp) != NULL)
69                 {
70                         int i, quickout = 0;
71                         char * q;
72                         char * p = strchr(pwtmp, '\n');
73
74                         if (p == NULL) {
75                                 while (fgets(pwtmp, sizeof pwtmp, pwd_fp) != NULL && strchr(pwtmp, '\n')==NULL)
76                                         ; /* Skip long lines */
77                                 continue;
78                         }
79
80                         /* skip comments & empty lines */
81                         if (*pwtmp =='\n' || *pwtmp == '#')
82                                 continue;
83
84                         i = 0;
85                         q = p = pwtmp;
86                         bzero(&pwd, sizeof pwd);
87                         while (!quickout && (p = strsep(&q, ":\n")) != NULL) {
88                                 switch (i++)
89                                 {
90                                 case 0:   /* username */
91                                         pwd.pw_name = p;
92                                         if (nam) {
93                                                 if (strcmp(nam, p) == 0)
94                                                         done = 1;
95                                                 else
96                                                         quickout = 1;
97                                         }
98                                         break;
99                                 case 1:   /* password */
100                                         pwd.pw_passwd = p;
101                                         break;
102                                 case 2:   /* uid */
103                                         pwd.pw_uid = atoi(p);
104                                         if (uid != (uid_t)-1) {
105                                                 if (uid == pwd.pw_uid)
106                                                         done = 1;
107                                                 else
108                                                         quickout = 1;
109                                         }
110                                         break;
111                                 case 3:   /* gid */
112                                         pwd.pw_gid = atoi(p);
113                                         break;
114                                 case 4:   /* class */
115                                         if (nam == NULL && uid == (uid_t)-1)
116                                                 done = 1;
117                                         pwd.pw_class = p;
118                                         break;
119                                 case 5:   /* change */
120                                         pwd.pw_change = (time_t)atol(p);
121                                         break;
122                                 case 6:   /* expire */
123                                         pwd.pw_expire = (time_t)atol(p);
124                                         break;
125                                 case 7:   /* gecos */
126                                         pwd.pw_gecos = p;
127                                         break;
128                                 case 8:   /* directory */
129                                         pwd.pw_dir = p;
130                                         break;
131                                 case 9:   /* shell */
132                                         pwd.pw_shell = p;
133                                         break;
134                                 }
135                         }
136                 }
137                 if (doclose)
138                         vendpwent();
139                 if (done && pwd.pw_name) {
140                         pw = &pwd;
141
142                         #define CKNULL(s)   s = s ? s : ""
143                         CKNULL(pwd.pw_passwd);
144                         CKNULL(pwd.pw_class);
145                         CKNULL(pwd.pw_gecos);
146                         CKNULL(pwd.pw_dir);
147                         CKNULL(pwd.pw_shell);
148                 }
149         }
150         return pw;
151 }
152
153 struct passwd *
154 vgetpwent(void)
155 {
156   return vnextpwent(NULL, -1, 0);
157 }
158
159 struct passwd *
160 vgetpwuid(uid_t uid)
161 {
162   return vnextpwent(NULL, uid, 1);
163 }
164
165 struct passwd *
166 vgetpwnam(const char * nam)
167 {
168   return vnextpwent(nam, -1, 1);
169 }
170
171 int
172 vpwdb(char *arg, ...)
173 {
174   arg=arg;
175   return 0;
176 }
177
178
179
180 static FILE * grp_fp = NULL;
181
182 void
183 vendgrent(void)
184 {
185         if (grp_fp != NULL) {
186                 fclose(grp_fp);
187                 grp_fp = NULL;
188         }
189 }
190
191 void
192 vsetgrent(void)
193 {
194         vendgrent();
195 }
196
197 static struct group *
198 vnextgrent(char const * nam, gid_t gid, int doclose)
199 {
200         struct group * gr = NULL;
201
202         static char * grtmp = NULL;
203         static int grlen = 0;
204         static char ** mems = NULL;
205         static int memlen = 0;
206
207         extendline(&grtmp, &grlen, MAXPATHLEN);
208         strncpy(grtmp, getgrpath(_GROUP), MAXPATHLEN);
209         grtmp[MAXPATHLEN - 1] = '\0';
210
211         if (grp_fp != NULL || (grp_fp = fopen(grtmp, "r")) != NULL) {
212                 int done = 0;
213
214                 static struct group grp;
215
216                 while (!done && fgets(grtmp, grlen, grp_fp) != NULL)
217                 {
218                         int i, quickout = 0;
219                         int mno = 0;
220                         char * q, * p;
221                         char * sep = ":\n";
222
223                         if ((p = strchr(grtmp, '\n')) == NULL) {
224                                 int l;
225                                 extendline(&grtmp, &grlen, grlen + PWBUFSZ);
226                                 l = strlen(grtmp);
227                                 if (fgets(grtmp + l, grlen - l, grp_fp) == NULL)
228                                   break;        /* No newline terminator on last line */
229                         }
230                         /* Skip comments and empty lines */
231                         if (*grtmp == '\n' || *grtmp == '#')
232                                 continue;
233                         i = 0;
234                         q = p = grtmp;
235                         bzero(&grp, sizeof grp);
236                         extendarray(&mems, &memlen, 200);
237                         while (!quickout && (p = strsep(&q, sep)) != NULL) {
238                                 switch (i++)
239                                 {
240                                 case 0:   /* groupname */
241                                         grp.gr_name = p;
242                                         if (nam) {
243                                                 if (strcmp(nam, p) == 0)
244                                                         done = 1;
245                                                 else
246                                                         quickout = 1;
247                                         }
248                                         break;
249                                 case 1:   /* password */
250                                         grp.gr_passwd = p;
251                                         break;
252                                 case 2:   /* gid */
253                                         grp.gr_gid = atoi(p);
254                                         if (gid != (gid_t)-1) {
255                                                 if (gid == (gid_t)grp.gr_gid)
256                                                         done = 1;
257                                                 else
258                                                         quickout = 1;
259                                         } else if (nam == NULL)
260                                                 done = 1;
261                                         break;
262                                 case 3:
263                                         q = p;
264                                         sep = ",\n";
265                                         break;
266                                 default:
267                                         if (*p) {
268                                                 extendarray(&mems, &memlen, mno + 2);
269                                                 mems[mno++] = p;
270                                         }
271                                         break;
272                                 }
273                         }
274                         grp.gr_mem = mems;
275                         mems[mno] = NULL;
276                 }
277                 if (doclose)
278                         vendgrent();
279                 if (done && grp.gr_name) {
280                         gr = &grp;
281
282                         CKNULL(grp.gr_passwd);
283                 }
284         }
285         return gr;
286 }
287
288 struct group *
289 vgetgrent(void)
290 {
291   return vnextgrent(NULL, -1, 0);
292 }
293
294
295 struct group *
296 vgetgrgid(gid_t gid)
297 {
298   return vnextgrent(NULL, gid, 1);
299 }
300
301 struct group *
302 vgetgrnam(const char * nam)
303 {
304   return vnextgrent(nam, -1, 1);
305 }
306
307 int
308 vgrdb(char *arg, ...)
309 {
310   arg=arg;
311   return 0;
312 }
313