Initial import from FreeBSD RELENG_4:
[games.git] / usr.sbin / rpc.yppasswdd / pw_copy.c
1 /*-
2  * Copyright (c) 1990, 1993, 1994
3  *      The Regents of the University of California.  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  * 3. All advertising materials mentioning features or use of this software
14  *    must display the following acknowledgement:
15  *      This product includes software developed by the University of
16  *      California, Berkeley and its contributors.
17  * 4. Neither the name of the University nor the names of its contributors
18  *    may be used to endorse or promote products derived from this software
19  *    without specific prior written permission.
20  *
21  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
22  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
25  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
26  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
27  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
28  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
29  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
30  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31  * SUCH DAMAGE.
32  */
33
34 #ifndef lint
35 #if 0
36 static char sccsid[] = "@(#)pw_copy.c   8.4 (Berkeley) 4/2/94";
37 #endif
38 static const char rcsid[] =
39   "$FreeBSD: src/usr.sbin/rpc.yppasswdd/pw_copy.c,v 1.4.2.1 2000/07/12 11:09:40 davidn Exp $";
40 #endif /* not lint */
41
42 /*
43  * This module is used to copy the master password file, replacing a single
44  * record, by chpass(1) and passwd(1).
45  */
46
47 #include <err.h>
48 #include <pwd.h>
49 #include <stdio.h>
50 #include <string.h>
51
52 #include <pw_util.h>
53 #include "yppasswdd_extern.h"
54
55 int
56 pw_copy(ffd, tfd, pw)
57         int ffd, tfd;
58         struct passwd *pw;
59 {
60         FILE *from, *to;
61         int done;
62         char *p, buf[8192];
63         char uidstr[20];
64         char gidstr[20];
65         char chgstr[20];
66         char expstr[20];
67
68         snprintf(uidstr, sizeof(uidstr), "%d", pw->pw_uid);
69         snprintf(gidstr, sizeof(gidstr), "%d", pw->pw_gid);
70         snprintf(chgstr, sizeof(chgstr), "%ld", pw->pw_change);
71         snprintf(expstr, sizeof(expstr), "%ld", pw->pw_expire);
72
73         if (!(from = fdopen(ffd, "r"))) {
74                 pw_error(passfile, 1, 1);
75                 return(-1);
76         }
77         if (!(to = fdopen(tfd, "w"))) {
78                 pw_error(tempname, 1, 1);
79                 return(-1);
80         }
81         for (done = 0; fgets(buf, sizeof(buf), from);) {
82                 if (!strchr(buf, '\n')) {
83                         yp_error("%s: line too long", passfile);
84                         pw_error(NULL, 0, 1);
85                         goto err;
86                 }
87                 if (done) {
88                         (void)fprintf(to, "%s", buf);
89                         if (ferror(to))
90                                 goto err;
91                         continue;
92                 }
93                 /*
94                  * Just copy comments and blank lines
95                  */
96                 p = buf + strspn(buf, " \t\n");
97                 if (*p == '\0' || *p == '#') {
98                         (void)fprintf(to, "%s", buf);
99                         if (ferror(to))
100                                 goto err;
101                         continue;
102                 }
103                 if (!(p = strchr(buf, ':'))) {
104                         yp_error("%s: corrupted entry", passfile);
105                         pw_error(NULL, 0, 1);
106                         goto err;
107                 }
108                 *p = '\0';
109                 if (strcmp(buf, pw->pw_name)) {
110                         *p = ':';
111                         (void)fprintf(to, "%s", buf);
112                         if (ferror(to))
113                                 goto err;
114                         continue;
115                 }
116                 (void)fprintf(to, "%s:%s:%s:%s:%s:%s:%s:%s:%s:%s\n",
117                     pw->pw_name, pw->pw_passwd,
118                     pw->pw_fields & _PWF_UID ? uidstr : "",
119                     pw->pw_fields & _PWF_GID ? gidstr : "",
120                     pw->pw_class,
121                     pw->pw_fields & _PWF_CHANGE ? chgstr : "",
122                     pw->pw_fields & _PWF_EXPIRE ? expstr : "",
123                     pw->pw_gecos, pw->pw_dir, pw->pw_shell);
124                 done = 1;
125                 if (ferror(to))
126                         goto err;
127         }
128         if (!done) {
129                 if (allow_additions) {
130                         (void)fprintf(to, "%s:%s:%s:%s:%s:%s:%s:%s:%s:%s\n",
131                         pw->pw_name, pw->pw_passwd,
132                         pw->pw_fields & _PWF_UID ? uidstr : "",
133                         pw->pw_fields & _PWF_GID ? gidstr : "",
134                         pw->pw_class,
135                         pw->pw_fields & _PWF_CHANGE ? chgstr : "",
136                         pw->pw_fields & _PWF_EXPIRE ? expstr : "",
137                         pw->pw_gecos, pw->pw_dir, pw->pw_shell);
138                 } else {
139                         yp_error("user \"%s\" not found in %s -- \
140 NIS maps and password file possibly out of sync", pw->pw_name, passfile);
141                         goto err;
142                 }
143         }
144         if (ferror(to)) {
145 err:            pw_error(NULL, 1, 1);
146                 (void)fclose(to);
147                 (void)fclose(from);
148                 return(-1);
149         }
150         (void)fclose(to);
151         (void)fclose(from);
152         return(0);
153 }