1 1.12 lukem /* $NetBSD: field.c,v 1.12 2009/04/11 12:10:02 lukem Exp $ */ 2 1.3 glass 3 1.1 cgd /* 4 1.3 glass * Copyright (c) 1988, 1993, 1994 5 1.3 glass * The Regents of the University of California. All rights reserved. 6 1.1 cgd * 7 1.1 cgd * Redistribution and use in source and binary forms, with or without 8 1.1 cgd * modification, are permitted provided that the following conditions 9 1.1 cgd * are met: 10 1.1 cgd * 1. Redistributions of source code must retain the above copyright 11 1.1 cgd * notice, this list of conditions and the following disclaimer. 12 1.1 cgd * 2. Redistributions in binary form must reproduce the above copyright 13 1.1 cgd * notice, this list of conditions and the following disclaimer in the 14 1.1 cgd * documentation and/or other materials provided with the distribution. 15 1.9 agc * 3. Neither the name of the University nor the names of its contributors 16 1.1 cgd * may be used to endorse or promote products derived from this software 17 1.1 cgd * without specific prior written permission. 18 1.1 cgd * 19 1.1 cgd * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 20 1.1 cgd * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 21 1.1 cgd * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 22 1.1 cgd * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 23 1.1 cgd * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 24 1.1 cgd * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 25 1.1 cgd * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 26 1.1 cgd * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 27 1.1 cgd * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 28 1.1 cgd * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 29 1.1 cgd * SUCH DAMAGE. 30 1.1 cgd */ 31 1.1 cgd 32 1.4 lukem #include <sys/cdefs.h> 33 1.1 cgd #ifndef lint 34 1.3 glass #if 0 35 1.3 glass static char sccsid[] = "@(#)field.c 8.4 (Berkeley) 4/2/94"; 36 1.3 glass #else 37 1.12 lukem __RCSID("$NetBSD: field.c,v 1.12 2009/04/11 12:10:02 lukem Exp $"); 38 1.3 glass #endif 39 1.1 cgd #endif /* not lint */ 40 1.1 cgd 41 1.1 cgd #include <sys/param.h> 42 1.3 glass 43 1.3 glass #include <ctype.h> 44 1.3 glass #include <err.h> 45 1.3 glass #include <errno.h> 46 1.3 glass #include <grp.h> 47 1.1 cgd #include <pwd.h> 48 1.3 glass #include <stdio.h> 49 1.3 glass #include <stdlib.h> 50 1.1 cgd #include <string.h> 51 1.3 glass #include <unistd.h> 52 1.3 glass 53 1.1 cgd #include "chpass.h" 54 1.1 cgd #include "pathnames.h" 55 1.1 cgd 56 1.1 cgd /* ARGSUSED */ 57 1.3 glass int 58 1.11 xtraeme p_login(const char *p, struct passwd *pw, ENTRY *ep) 59 1.1 cgd { 60 1.5 mrg 61 1.1 cgd if (!*p) { 62 1.3 glass warnx("empty login field"); 63 1.3 glass return (1); 64 1.1 cgd } 65 1.1 cgd if (*p == '-') { 66 1.3 glass warnx("login names may not begin with a hyphen"); 67 1.3 glass return (1); 68 1.1 cgd } 69 1.1 cgd if (!(pw->pw_name = strdup(p))) { 70 1.3 glass warnx("can't save entry"); 71 1.3 glass return (1); 72 1.1 cgd } 73 1.3 glass if (strchr(p, '.')) 74 1.3 glass warnx("\'.\' is dangerous in a login name"); 75 1.1 cgd for (; *p; ++p) 76 1.10 dsl if (isupper((unsigned char)*p)) { 77 1.3 glass warnx("upper-case letters are dangerous in a login name"); 78 1.1 cgd break; 79 1.1 cgd } 80 1.3 glass return (0); 81 1.1 cgd } 82 1.1 cgd 83 1.1 cgd /* ARGSUSED */ 84 1.3 glass int 85 1.11 xtraeme p_passwd(const char *p, struct passwd *pw, ENTRY *ep) 86 1.1 cgd { 87 1.5 mrg 88 1.12 lukem if (!(pw->pw_passwd = strdup(p))) { 89 1.3 glass warnx("can't save password entry"); 90 1.3 glass return (1); 91 1.1 cgd } 92 1.1 cgd 93 1.3 glass return (0); 94 1.1 cgd } 95 1.1 cgd 96 1.1 cgd /* ARGSUSED */ 97 1.3 glass int 98 1.11 xtraeme p_uid(const char *p, struct passwd *pw, ENTRY *ep) 99 1.1 cgd { 100 1.8 simonb unsigned long id; 101 1.3 glass char *np; 102 1.1 cgd 103 1.1 cgd if (!*p) { 104 1.3 glass warnx("empty uid field"); 105 1.3 glass return (1); 106 1.1 cgd } 107 1.10 dsl if (!isdigit((unsigned char)*p)) { 108 1.3 glass warnx("illegal uid"); 109 1.3 glass return (1); 110 1.1 cgd } 111 1.3 glass errno = 0; 112 1.3 glass id = strtoul(p, &np, 10); 113 1.8 simonb /* 114 1.8 simonb * We don't need to check the return value of strtoul() 115 1.8 simonb * since ULONG_MAX is greater than UID_MAX. 116 1.8 simonb */ 117 1.8 simonb if (*np || id > UID_MAX) { 118 1.3 glass warnx("illegal uid"); 119 1.3 glass return (1); 120 1.1 cgd } 121 1.8 simonb pw->pw_uid = (uid_t)id; 122 1.3 glass return (0); 123 1.1 cgd } 124 1.1 cgd 125 1.1 cgd /* ARGSUSED */ 126 1.3 glass int 127 1.11 xtraeme p_gid(const char *p, struct passwd *pw, ENTRY *ep) 128 1.1 cgd { 129 1.1 cgd struct group *gr; 130 1.8 simonb unsigned long id; 131 1.3 glass char *np; 132 1.1 cgd 133 1.1 cgd if (!*p) { 134 1.3 glass warnx("empty gid field"); 135 1.3 glass return (1); 136 1.1 cgd } 137 1.10 dsl if (!isdigit((unsigned char)*p)) { 138 1.1 cgd if (!(gr = getgrnam(p))) { 139 1.3 glass warnx("unknown group %s", p); 140 1.3 glass return (1); 141 1.1 cgd } 142 1.1 cgd pw->pw_gid = gr->gr_gid; 143 1.3 glass return (0); 144 1.1 cgd } 145 1.3 glass errno = 0; 146 1.3 glass id = strtoul(p, &np, 10); 147 1.8 simonb /* 148 1.8 simonb * We don't need to check the return value of strtoul() 149 1.8 simonb * since ULONG_MAX is greater than GID_MAX. 150 1.8 simonb */ 151 1.8 simonb if (*np || id > GID_MAX) { 152 1.3 glass warnx("illegal gid"); 153 1.3 glass return (1); 154 1.1 cgd } 155 1.8 simonb pw->pw_gid = (gid_t)id; 156 1.3 glass return (0); 157 1.1 cgd } 158 1.1 cgd 159 1.1 cgd /* ARGSUSED */ 160 1.3 glass int 161 1.11 xtraeme p_class(const char *p, struct passwd *pw, ENTRY *ep) 162 1.1 cgd { 163 1.5 mrg 164 1.12 lukem if (!(pw->pw_class = strdup(p))) { 165 1.3 glass warnx("can't save entry"); 166 1.3 glass return (1); 167 1.1 cgd } 168 1.1 cgd 169 1.3 glass return (0); 170 1.1 cgd } 171 1.1 cgd 172 1.1 cgd /* ARGSUSED */ 173 1.3 glass int 174 1.11 xtraeme p_change(const char *p, struct passwd *pw, ENTRY *ep) 175 1.1 cgd { 176 1.5 mrg 177 1.1 cgd if (!atot(p, &pw->pw_change)) 178 1.3 glass return (0); 179 1.3 glass warnx("illegal date for change field"); 180 1.3 glass return (1); 181 1.1 cgd } 182 1.1 cgd 183 1.1 cgd /* ARGSUSED */ 184 1.3 glass int 185 1.11 xtraeme p_expire(const char *p, struct passwd *pw, ENTRY *ep) 186 1.1 cgd { 187 1.5 mrg 188 1.1 cgd if (!atot(p, &pw->pw_expire)) 189 1.3 glass return (0); 190 1.3 glass warnx("illegal date for expire field"); 191 1.3 glass return (1); 192 1.1 cgd } 193 1.1 cgd 194 1.1 cgd /* ARGSUSED */ 195 1.3 glass int 196 1.11 xtraeme p_gecos(const char *p, struct passwd *pw, ENTRY *ep) 197 1.1 cgd { 198 1.5 mrg 199 1.7 kim if (!(ep->save = strdup(p))) { 200 1.3 glass warnx("can't save entry"); 201 1.3 glass return (1); 202 1.1 cgd } 203 1.3 glass return (0); 204 1.1 cgd } 205 1.1 cgd 206 1.1 cgd /* ARGSUSED */ 207 1.3 glass int 208 1.11 xtraeme p_hdir(const char *p, struct passwd *pw, ENTRY *ep) 209 1.1 cgd { 210 1.5 mrg 211 1.1 cgd if (!*p) { 212 1.3 glass warnx("empty home directory field"); 213 1.3 glass return (1); 214 1.1 cgd } 215 1.1 cgd if (!(pw->pw_dir = strdup(p))) { 216 1.3 glass warnx("can't save entry"); 217 1.3 glass return (1); 218 1.1 cgd } 219 1.3 glass return (0); 220 1.1 cgd } 221 1.1 cgd 222 1.1 cgd /* ARGSUSED */ 223 1.3 glass int 224 1.11 xtraeme p_shell(const char *p, struct passwd *pw, ENTRY *ep) 225 1.1 cgd { 226 1.6 mycroft const char *t; 227 1.1 cgd 228 1.1 cgd if (!*p) { 229 1.12 lukem if (!(pw->pw_shell = strdup(_PATH_BSHELL))) { 230 1.12 lukem warnx("can't save entry"); 231 1.12 lukem return (1); 232 1.12 lukem } 233 1.3 glass return (0); 234 1.1 cgd } 235 1.1 cgd /* only admin can change from or to "restricted" shells */ 236 1.1 cgd if (uid && pw->pw_shell && !ok_shell(pw->pw_shell)) { 237 1.3 glass warnx("%s: current shell non-standard", pw->pw_shell); 238 1.3 glass return (1); 239 1.1 cgd } 240 1.1 cgd if (!(t = ok_shell(p))) { 241 1.1 cgd if (uid) { 242 1.3 glass warnx("%s: non-standard shell", p); 243 1.3 glass return (1); 244 1.1 cgd } 245 1.1 cgd } 246 1.1 cgd else 247 1.1 cgd p = t; 248 1.1 cgd if (!(pw->pw_shell = strdup(p))) { 249 1.3 glass warnx("can't save entry"); 250 1.3 glass return (1); 251 1.1 cgd } 252 1.3 glass return (0); 253 1.1 cgd } 254