jfh@rpp386.cactus.org (John F Haugh II) (01/10/91)
I finally got a copy of the freed BSD sources and was tinkering with chpass. It is a slightly more user-friendly version of the chfn and chsh commands. So, I ported it to SCO Xenix and this is what I got. It should work just fine on any System V machine. The chpass.1 manpage has not been modified at all and may not be usable. Feel free to fix it and mail the fixes back to me. It requires that you have the "mkpasswd" command. If you don't have it, feel free to rip out that part of the code or pick up the version which I posted here some time back from your nearest alt.sources archive site. I will be modifying this to work with the password libraries I've been tinkering with and will add it to that collection of commands at that time. On thing I hope to fix is not having to rebuild the entire password DBM file each time an entry is changed. This will greatly cut down on the amount of time the password file is locked. As always, unshar and enjoy! -- #! /bin/sh # This is a shell archive, meaning: # 1. Remove everything above the #! /bin/sh line. # 2. Save the resulting text in a file. # 3. Execute the file with /bin/sh (not csh) to create: # chpass.1 # Makefile # chpass.c # chpass.h # field.c # pathnames.h # util.c # This archive created: Thu Jan 10 09:00:12 1991 # By: John F Haugh II (River Parishes Programming, Austin TX) export PATH; PATH=/bin:/usr/bin:$PATH echo shar: "extracting 'chpass.1'" '(5439 characters)' if test -f 'chpass.1' then echo shar: "will not over-write existing file 'chpass.1'" else sed 's/^X//' << \SHAR_EOF > 'chpass.1' X.\" Copyright (c) 1988, 1990 The Regents of the University of California. X.\" All rights reserved. X.\" X.\" Redistribution and use in source and binary forms are permitted provided X.\" that: (1) source distributions retain this entire copyright notice and X.\" comment, and (2) distributions including binaries display the following X.\" acknowledgement: ``This product includes software developed by the X.\" University of California, Berkeley and its contributors'' in the X.\" documentation or other materials provided with the distribution and in X.\" all advertising materials mentioning features or use of this software. X.\" Neither the name of the University nor the names of its contributors may X.\" be used to endorse or promote products derived from this software without X.\" specific prior written permission. X.\" THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED X.\" WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF X.\" MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. X.\" X.\" @(#)chpass.1 5.9 (Berkeley) 7/24/90 X.\" X.Dd July 24, 1990 X.Dt CHPASS 1 X.Os BSD 4.4 X.Sh NAME X.Nm chpass X.Nd add or change user database information X.Sh SYNOPSIS Xchpass X.Op Fl a Ar list X.Op Fl s Ar shell X.Op user X.Sh DESCRIPTION X.Nm Chpass Xallows editing of the user database information associated Xwith X.Ar user Xor, by default, the current user. XThe information is formatted and supplied to an editor for changes. X.Pp XOnly the information that the user is allowed to change is displayed. X.Pp XThe options are as follows: X.Tw Ds X.Tp Fl a XThe super-user is allowed to directly supply a user database Xentry, in the format specified by X.Xr passwd 5 , Xas an argument. XThis argument must be a colon (``:'') separated list of all the Xuser database fields, although they may be empty. X.Tp Fl s XThe X.Fl s Xoption attempts to change the user's shell to X.Ar newsh . X.Tp X.Pp XPossible display items are as follows: X.Pp X.Dw Home\ Directory: X.Dp Login: Xuser's login name X.Dp Password: Xuser's encrypted password X.Dp Uid: Xuser's id X.Dp Gid: Xuser's login group id X.Dp Change: Xpassword change time X.Dp Expire: Xaccount expiration time X.Dp Class: Xuser's general classification X.Dp Home Directory: Xuser's home directory X.Dp Shell: Xuser's login shell X.Dp Full Name: Xuser's real name X.Dp Location: Xuser's normal location X.Dp Home Phone: Xuser's home phone X.Dp Office Phone: Xuser's office phone X.Dp X.Pp XThe X.Ar login Xfield is the user name used to access the computer account. X.Pp XThe X.Ar password Xfield contains the encrypted form of the user's password. X.Pp XThe X.Ar uid Xfield is the number associated with the X.Ar login Xfield. XBoth of these fields should be unique across the system (and often Xacross a group of systems) as they control file access. X.Pp XWhile it is possible to have multiple entries with identical login names Xand/or identical user id's, it is usually a mistake to do so. Routines Xthat manipulate these files will often return only one of the multiple Xentries, and that one by random selection. X.Pp XThe X.Ar group Xfield is the group that the user will be placed in at login. XSince this system supports multiple groups (see X.Xr groups 1 ) Xthis field currently has little special meaning. XThis field may be filled in with either a number or a group name (see X.Xr group 5 ) . X.Pp XThe X.Ar change Xfield is the date by which the password must be changed. X.Pp XThe X.Ar expire Xfield is the date on which the account expires. X.Pp XBoth the X.Ar change Xand X.Ar expire Xfields should be entered in the form ``month day year'' where X.Ar month Xis the month name (the first three characters are sufficient), X.Ar day Xis the day of the month, and X.Ar year Xis the year. X.bp XThe X.Ar class Xfield is currently unused. In the near future it will be a key to Xa X.Xr termcap 5 Xstyle database of user attributes. X.Pp XThe user's X.Ar home directory Xis the full UNIX path name where the user Xwill be placed at login. X.Pp XThe X.Ar shell Xfield is the command interpreter the user prefers. XIf the X.Ar shell Xfield is empty, the Bourne shell, X.Pa /bin/sh , Xis assumed. XWhen altering a login shell, and not the super-user, the user Xmay not change from a non-standard shell or to a non-standard Xshell. XNon-standard is defined as a shell not found in X.Pa /etc/shells . X.Pp XThe last four fields are for storing the user's X.Ar full name , office location , Xand X.Ar home Xand X.Ar work telephone Xnumbers. X.Pp XOnce the information has been verified, X.Nm chpass Xuses X.Xr mkpasswd 8 Xto update the user database. This is run in the background, and, Xat very large sites could take several minutes. Until this update Xis completed, the password file is unavailable for other updates Xand the new information will not be available to programs. X.Sh ENVIRONMENT XThe X.Xr vi 1 Xeditor will be used unless the environment variable EDITOR is set to Xan alternate editor. XWhen the editor terminates, the information is re-read and used to Xupdate the user database itself. XOnly the user, or the super-user, may edit the information associated Xwith the user. X.Sh FILES X.Dw /etc/master.passwd X.Di L X.Dp Pa /etc/master.passwd XThe user database X.Dp Pa /etc/shells XThe list of approved shells X.Dp X.Sh SEE ALSO X.Xr login 1 , X.Xr finger 1 , X.Xr getusershell 3 , X.Xr passwd 5 , X.Xr mkpasswd 8 , X.Xr vipw 8 X.Pp XRobert Morris and Ken Thompson, X.Ar UNIX Password security X.Sh HISTORY XFirst release 4.3 Reno BSD. X.Sh BUGS XUser information should (and eventually will) be stored elsewhere. SHAR_EOF if test 5439 -ne "`wc -c < 'chpass.1'`" then echo shar: "error transmitting 'chpass.1'" '(should have been 5439 characters)' fi fi echo shar: "extracting 'Makefile'" '(302 characters)' if test -f 'Makefile' then echo shar: "will not over-write existing file 'Makefile'" else sed 's/^X//' << \SHAR_EOF > 'Makefile' X# @(#)Makefile 5.3 (Berkeley) 5/11/90 X XPROG= chpass XSRCS= chpass.c field.c util.c XBINOWN= root XBINMODE=4555 XLINKS= ${BINDIR}/chpass ${BINDIR}/chfn ${BINDIR}/chpass ${BINDIR}/chsh XMLINKS= chpass.1 chfn.1 chpass.1 chsh.1 XCFLAGS= -g X Xall: $(PROG) X X$(PROG): $(SRCS:.c=.o) X cc $(CFLAGS) -o $@ $(SRCS:.c=.o) SHAR_EOF if test 302 -ne "`wc -c < 'Makefile'`" then echo shar: "error transmitting 'Makefile'" '(should have been 302 characters)' fi fi echo shar: "extracting 'chpass.c'" '(11797 characters)' if test -f 'chpass.c' then echo shar: "will not over-write existing file 'chpass.c'" else sed 's/^X//' << \SHAR_EOF > 'chpass.c' X/*- X * Copyright (c) 1988 The Regents of the University of California. X * All rights reserved. X * X * Redistribution and use in source and binary forms are permitted provided X * that: (1) source distributions retain this entire copyright notice and X * comment, and (2) distributions including binaries display the following X * acknowledgement: ``This product includes software developed by the X * University of California, Berkeley and its contributors'' in the X * documentation or other materials provided with the distribution and in X * all advertising materials mentioning features or use of this software. X * Neither the name of the University nor the names of its contributors may X * be used to endorse or promote products derived from this software without X * specific prior written permission. X * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED X * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF X * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. X */ X X/* X * Modified 1/10/91, John F Haugh II X * X * Made changes needed to get chpass to run on System V. X */ X X#ifndef lint Xchar copyright[] = X"@(#) Copyright (c) 1988 The Regents of the University of California.\n\ X All rights reserved.\n"; X#endif /* not lint */ X X#ifndef lint Xstatic char sccsid[] = "@(#)chpass.c 5.15 (Berkeley) 6/29/90"; X#endif /* not lint */ X X#include <sys/types.h> X#include <sys/param.h> X#include <sys/file.h> X#include <sys/stat.h> X#include <sys/signal.h> X#include <fcntl.h> X#include <time.h> X#include <pwd.h> X#include <errno.h> X#include <stdio.h> X#include <ctype.h> X#include <string.h> X#include "chpass.h" X#include "pathnames.h" X Xchar e1[] = ": "; Xchar e2[] = ":,"; X Xint p_gecos(), p_gid(), p_hdir(); Xint p_login(), p_passwd(), p_shell(), p_uid(); X Xstruct entry list[] = { X { "Login", p_login, 1, 5, e1, }, X { "Password", p_passwd, 1, 8, e1, }, X { "Uid", p_uid, 1, 3, e1, }, X { "Gid", p_gid, 1, 3, e1, }, X#define E_NAME 4 X { "Full Name", p_gecos, 0, 9, e2, }, X#define E_BPHONE 5 X { "Office Phone", p_gecos, 0, 12, e2, }, X#define E_HPHONE 6 X { "Home Phone", p_gecos, 0, 10, e2, }, X#define E_LOCATE 7 X { "Location", p_gecos, 0, 8, e2, }, X { "Home directory", p_hdir, 1, 14, e1, }, X#define E_SHELL 9 X { "Shell", p_shell, 0, 5, e1, }, X { NULL, 0, }, X}; X Xuid_t uid; X Xstruct flock lock = { F_WRLCK, 0, 0, 0, 0, 0 }; Xextern char *sys_errlist[]; X Xmain(argc, argv) X int argc; X char **argv; X{ X extern int errno, optind; X extern char *optarg; X register char *p; X struct passwd lpw, *pw; X FILE *temp_fp; X int aflag, ch, fd; X char *fend, *newsh, *passwd, *temp, *tend; X char from[BUFSIZ], to[BUFSIZ]; X char *getusershell(); X X uid = getuid(); X aflag = 0; X newsh = NULL; X while ((ch = getopt(argc, argv, "a:s:")) != EOF) X switch(ch) { X case 'a': X if (uid) X baduser(); X loadpw(optarg, pw = &lpw); X aflag = 1; X break; X case 's': X newsh = optarg; X /* protect p_field -- it thinks NULL is /bin/sh */ X if (!*newsh) X usage(); X break; X case '?': X default: X usage(); X } X argc -= optind; X argv += optind; X X if (!aflag) X switch(argc) { X case 0: X if (!(pw = getpwuid(uid))) { X (void)fprintf(stderr, X "chpass: unknown user: uid %u\n", uid); X exit(1); X } X break; X case 1: X if (!(pw = getpwnam(*argv))) { X (void)fprintf(stderr, X "chpass: unknown user %s.\n", *argv); X exit(1); X } X if (uid && uid != pw->pw_uid) X baduser(); X break; X default: X usage(); X } X X (void)signal(SIGHUP, SIG_IGN); X (void)signal(SIGINT, SIG_IGN); X (void)signal(SIGQUIT, SIG_IGN); X X (void)ulimit(2, (1L<<20)); X (void)umask(0); X X temp = _PATH_PTMP; X if ((fd = open(temp, O_WRONLY|O_CREAT|O_EXCL, 0644)) < 0) { X if (errno == EEXIST) { X (void)fprintf(stderr, X "chpass: password file busy -- try again later.\n"); X exit(1); X } X (void)fprintf(stderr, "chpass: %s: %s; ", X temp, sys_errlist[errno]); X goto bad; X } X if (!(temp_fp = fdopen(fd, "w"))) { X (void)fprintf(stderr, "chpass: can't write %s; ", temp); X goto bad; X } X X if (newsh) { X if (p_shell(newsh, pw, (struct entry *)NULL)) X goto bad; X } X else if (!aflag && !info(pw)) X goto bad; X X /* root should have a 0 uid and a reasonable shell */ X if (!strcmp(pw->pw_name, "root")) { X if (pw->pw_uid) { X (void)fprintf(stderr, "chpass: root uid should be 0."); X exit(1); X } X setusershell(); X for (;;) X if (!(p = getusershell())) { X (void)fprintf(stderr, X "chpass: warning, unknown root shell."); X break; X } X else if (!strcmp(pw->pw_shell, p)) X break; X } X X passwd = _PATH_PASSWD; X if (!freopen(passwd, "r", stdin)) { X (void)fprintf(stderr, "chpass: can't read %s; ", passwd); X goto bad; X } X if (!copy(pw, temp_fp)) X goto bad; X X (void)fclose(temp_fp); X (void)fclose(stdin); X X switch(fork()) { X case 0: X break; X case -1: X (void)fprintf(stderr, "chpass: can't fork; "); X goto bad; X /* NOTREACHED */ X default: X exit(0); X /* NOTREACHED */ X } X X if (makedb(temp)) { X (void)fprintf(stderr, "chpass: mkpasswd failed; "); Xbad: (void)fprintf(stderr, "%s unchanged.\n", _PATH_PASSWD); X (void)unlink(temp); X exit(1); X } X X /* X * possible race; have to rename four files, and someone could slip X * in between them. LOCK_EX and rename the ``passwd.dir'' file first X * so that getpwent(3) can't slip in; the lock should never fail and X * it's unclear what to do if it does. Rename ``ptmp'' last so that X * passwd/vipw/chpass can't slip in. X */ X (void)nice(-19); X fend = strcpy(from, temp) + strlen(temp); X tend = strcpy(to, _PATH_PASSWD) + strlen(_PATH_PASSWD); X memcpy(fend, ".dir", 5); X memcpy(tend, ".dir", 5); X if ((fd = open(from, O_RDONLY, 0)) >= 0) X fcntl (fd, F_SETLK, &lock); X X /* here we go... */ X (void)rename(from, to); X memcpy(fend, ".pag", 5); X memcpy(tend, ".pag", 5); X (void)rename(from, to); X memcpy(fend, ".orig", 6); X (void)rename(from, _PATH_PASSWD); X (void)rename(temp, passwd); X /* done! */ X exit(0); X} X Xrename (from, to) Xchar *from; Xchar *to; X{ X struct stat s1, s2; X X if (stat (from, &s1)) X return -1; X X if (stat (to, &s2)) { X if (link (from, to)) X return -1; X X return unlink(from); X } X if (s1.st_dev != s2.st_dev) { X errno = EXDEV; X return -1; X } X if (unlink (to)) X return -1; X X if (link (from, to)) X return -1; X X return unlink (from); X} X Xinfo(pw) X struct passwd *pw; X{ X struct stat begin, end; X FILE *fp; X int fd, rval; X char *tfile; X X tfile = _PATH_TMP; X if ((fd = open(mktemp(tfile), O_RDWR|O_CREAT, 0600)) == -1 X || !(fp = fdopen(fd, "w+"))) { X (void)fprintf(stderr, "chpass: no temporary file"); X return(0); X } X X /* X * if print doesn't print out a shell field, make it restricted. X * Not particularly pretty, but print is the routine that checks X * to see if the user can change their shell. X */ X if (!print(fp, pw)) X list[E_SHELL].restricted = 1; X (void)fflush(fp); X X /* X * give the file to the real user; setuid permissions X * are discarded in edit() X */ X (void)chown(tfile, getuid(), getgid()); X X for (rval = 0;;) { X (void)fstat(fd, &begin); X if (edit(tfile)) { X (void)fprintf(stderr, "chpass: edit failed; "); X break; X } X (void)fstat(fd, &end); X if (begin.st_mtime == end.st_mtime) { X (void)fprintf(stderr, "chpass: no changes made; "); X break; X } X (void)rewind(fp); X if (check(fp, pw)) { X rval = 1; X break; X } X (void)fflush(stderr); X if (prompt()) X break; X } X (void)fclose(fp); X (void)unlink(tfile); X return(rval); X} X Xcheck(fp, pw) X FILE *fp; X struct passwd *pw; X{ X register struct entry *ep; X register char *p; X static char buf[1024]; X X while (fgets(buf, sizeof(buf), fp)) { X if (!buf[0] || buf[0] == '#') X continue; X if (!(p = strchr(buf, '\n'))) { X (void)fprintf(stderr, "chpass: line too long.\n"); X return(0); X } X *p = '\0'; X for (ep = list;; ++ep) { X if (!ep->prompt) { X (void)fprintf(stderr, X "chpass: unrecognized field.\n"); X return(0); X } X if (!strncmp(buf, ep->prompt, ep->len)) { Xprintf ("found %s\n", buf); X if (ep->restricted && uid) { X (void)fprintf(stderr, X "chpass: you may not change the %s field.\n", X ep->prompt); X return(0); X } X if (!(p = strchr(buf, ':'))) { X (void)fprintf(stderr, X "chpass: line corrupted.\n"); X return(0); X } X while (isspace(*++p)); X if (ep->except && strpbrk(p, ep->except)) { X (void)fprintf(stderr, X "chpass: illegal character in the \"%s\" field.\n", X ep->prompt); X return(0); X } X if ((*ep->func)(p, pw, ep)) X return(0); X break; X } X } X } X /* X * special checks... X * X * there has to be a limit on the size of the gecos fields, X * otherwise getpwent(3) can choke. X * ``if I swallow anything evil, put your fingers down my throat...'' X * -- The Who X */ X if (strlen(list[E_NAME].save) + strlen(list[E_BPHONE].save) + X strlen(list[E_HPHONE].save) + strlen(list[E_LOCATE].save) X > 512) { X (void)fprintf(stderr, "chpass: gecos field too large.\n"); X exit(1); X } X (void)sprintf(pw->pw_gecos = buf, "%s,%s,%s,%s", X list[E_NAME].save, list[E_LOCATE].save, list[E_BPHONE].save, X list[E_HPHONE].save); X return(1); X} X Xcopy(pw, fp) X struct passwd *pw; X FILE *fp; X{ X register int done; X register char *p; X char buf[1024]; X X for (done = 0; fgets(buf, sizeof(buf), stdin);) { X /* skip lines that are too big */ X if (!strchr(buf, '\n')) { X (void)fprintf(stderr, "chpass: line too long; "); X return(0); X } X if (done) { X (void)fprintf(fp, "%s", buf); X continue; X } X if (!(p = strchr(buf, ':'))) { X (void)fprintf(stderr, "chpass: corrupted entry; "); X return(0); X } X *p = '\0'; X if (strcmp(buf, pw->pw_name)) { X *p = ':'; X (void)fprintf(fp, "%s", buf); X continue; X } X (void)fprintf(fp, "%s:%s:%d:%d:%s:%s:%s\n", X pw->pw_name, pw->pw_passwd, pw->pw_uid, pw->pw_gid, X pw->pw_gecos, pw->pw_dir, pw->pw_shell); X done = 1; X } X if (!done) X (void)fprintf(fp, "%s:%s:%d:%d:%s:%s:%s\n", X pw->pw_name, pw->pw_passwd, pw->pw_uid, pw->pw_gid, X pw->pw_gecos, pw->pw_dir, pw->pw_shell); X return(1); X} X Xmakedb(file) X char *file; X{ X int status, pid, w; X X if (!(pid = fork())) { X execl(_PATH_MKPASSWD, "mkpasswd", "-p", file, NULL); X (void)fprintf(stderr, "chpass: can't find \"mkpasswd\".\n"); X _exit(127); X } X while ((w = wait(&status)) != pid && w != -1); X return(w == -1 || status); X} X Xedit(file) X char *file; X{ X int status, pid, w; X char *p, *editor, *getenv(); X X if (editor = getenv("EDITOR")) { X if (p = strrchr(editor, '/')) X ++p; X else X p = editor; X } X else X p = editor = "vi"; X if (!(pid = fork())) { X (void)setgid(getgid()); X (void)setuid(getuid()); X execlp(editor, p, file, NULL); X (void)fprintf(stderr, "chpass: can't find \"%s\".\n", editor); X _exit(127); X } X while ((w = wait(&status)) != pid && w != -1); X return(w == -1 || status); X} X Xloadpw(arg, pw) X char *arg; X register struct passwd *pw; X{ X register char *cp; X char *bp = arg; X long atol(); X char *strtok(); X X pw->pw_name = strtok(bp, ":"); X pw->pw_passwd = strtok((char *) 0, ":"); X if (!(cp = strtok((char *) 0, ":"))) X goto bad; X pw->pw_uid = atoi(cp); X if (!(cp = strtok((char *) 0, ":"))) X goto bad; X pw->pw_gid = atoi(cp); X pw->pw_gecos = strtok((char *) 0, ":"); X pw->pw_dir = strtok((char *) 0, ":"); X pw->pw_shell = strtok((char *) 0, ":"); X if (!pw->pw_shell || strtok((char *) 0, ":")) { Xbad: (void)fprintf(stderr, "chpass: bad password list.\n"); X exit(1); X } X} X Xprompt() X{ X register int c; X X for (;;) { X (void)printf("re-edit the password file? [y]: "); X (void)fflush(stdout); X c = getchar(); X if (c != EOF && c != (int)'\n') X while (getchar() != (int)'\n'); X return(c == (int)'n'); X } X /* NOTREACHED */ X} X Xbaduser() X{ X (void)fprintf(stderr, "chpass: %s\n", sys_errlist[EACCES]); X exit(1); X} X Xusage() X{ X (void)fprintf(stderr, "usage: chpass [-a list] [-s shell] [user]\n"); X exit(1); X} SHAR_EOF if test 11797 -ne "`wc -c < 'chpass.c'`" then echo shar: "error transmitting 'chpass.c'" '(should have been 11797 characters)' fi fi echo shar: "extracting 'chpass.h'" '(1156 characters)' if test -f 'chpass.h' then echo shar: "will not over-write existing file 'chpass.h'" else sed 's/^X//' << \SHAR_EOF > 'chpass.h' X/* X * Copyright (c) 1988 The Regents of the University of California. X * All rights reserved. X * X * Redistribution and use in source and binary forms are permitted X * provided that: (1) source distributions retain this entire copyright X * notice and comment, and (2) distributions including binaries display X * the following acknowledgement: ``This product includes software X * developed by the University of California, Berkeley and its contributors'' X * in the documentation or other materials provided with the distribution X * and in all advertising materials mentioning features or use of this X * software. Neither the name of the University nor the names of its X * contributors may be used to endorse or promote products derived X * from this software without specific prior written permission. X * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR X * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED X * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. X * X * @(#)chpass.h 5.2 (Berkeley) 6/1/90 X */ X Xstruct entry { X char *prompt; X int (*func)(), restricted, len; X char *except, *save; X}; X Xextern uid_t uid; SHAR_EOF if test 1156 -ne "`wc -c < 'chpass.h'`" then echo shar: "error transmitting 'chpass.h'" '(should have been 1156 characters)' fi fi echo shar: "extracting 'field.c'" '(4654 characters)' if test -f 'field.c' then echo shar: "will not over-write existing file 'field.c'" else sed 's/^X//' << \SHAR_EOF > 'field.c' X/* X * Copyright (c) 1988 The Regents of the University of California. X * All rights reserved. X * X * Redistribution and use in source and binary forms are permitted X * provided that: (1) source distributions retain this entire copyright X * notice and comment, and (2) distributions including binaries display X * the following acknowledgement: ``This product includes software X * developed by the University of California, Berkeley and its contributors'' X * in the documentation or other materials provided with the distribution X * and in all advertising materials mentioning features or use of this X * software. Neither the name of the University nor the names of its X * contributors may be used to endorse or promote products derived X * from this software without specific prior written permission. X * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR X * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED X * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. X */ X X/* X * Modified 1/10/91, John F Haugh II X * X * Made changes needed to get chpass to run on System V. X */ X X#ifndef lint Xstatic char sccsid[] = "@(#)field.c 5.12 (Berkeley) 6/1/90"; X#endif /* not lint */ X X#include <sys/types.h> X#include <sys/param.h> X#include <pwd.h> X#include <grp.h> X#include <string.h> X#include <stdio.h> X#include <ctype.h> X#include "chpass.h" X#include "pathnames.h" X X#ifndef USHRT_MAX X#define USHRT_MAX (0xffff) X#endif X X/* ARGSUSED */ Xp_login(p, pw, ep) X char *p; X struct passwd *pw; X struct entry *ep; X{ X if (!*p) { X (void)fprintf(stderr, "chpass: empty login field.\n"); X return(1); X } X if (*p == '-') { X (void)fprintf(stderr, X "chpass: login names may not begin with a hyphen.\n"); X return(1); X } X if (!(pw->pw_name = strdup(p))) { X (void)fprintf(stderr, "chpass: can't save entry.\n"); X return(1); X } X if (strchr(p, '.')) X (void)fprintf(stderr, X "chpass: \'.\' is dangerous in a login name.\n"); X for (; *p; ++p) X if (isupper(*p)) { X (void)fprintf(stderr, X "chpass: upper-case letters are dangerous in a login name.\n"); X break; X } X return(0); X} X X/* ARGSUSED */ Xp_passwd(p, pw, ep) X char *p; X struct passwd *pw; X struct entry *ep; X{ X if (!*p) X pw->pw_passwd = ""; /* "NOLOGIN"; */ X else if (!(pw->pw_passwd = strdup(p))) { X (void)fprintf(stderr, "chpass: can't save password entry.\n"); X return(1); X } X X return(0); X} X X/* ARGSUSED */ Xp_uid(p, pw, ep) X register char *p; X struct passwd *pw; X struct entry *ep; X{ X int id; X X if (!*p) { X (void)fprintf(stderr, "chpass: empty uid field.\n"); X return(1); X } X if (!isdigit(*p)) { X (void)fprintf(stderr, "chpass: illegal uid.\n"); X return(1); X } X id = atoi(p); X if ((unsigned int)id > USHRT_MAX) { X (void)fprintf(stderr, "chpass: %d > max uid value (%d).\n", X id, USHRT_MAX); X return(1); X } X pw->pw_uid = id; X return(0); X} X X/* ARGSUSED */ Xp_gid(p, pw, ep) X register char *p; X struct passwd *pw; X struct entry *ep; X{ X struct group *gr; X int id; X X if (!*p) { X (void)fprintf(stderr, "chpass: empty gid field.\n"); X return(1); X } X if (!isdigit(*p)) { X if (!(gr = getgrnam(p))) { X (void)fprintf(stderr, X "chpass: unknown group %s.\n", p); X return(1); X } X pw->pw_gid = gr->gr_gid; X return(0); X } X id = atoi(p); X if ((unsigned int)id > USHRT_MAX) { X (void)fprintf(stderr, "chpass: %d > max gid value (%d).\n", X id, USHRT_MAX); X return(1); X } X pw->pw_gid = id; X return(0); X} X X/* ARGSUSED */ Xp_gecos(p, pw, ep) X char *p; X struct passwd *pw; X struct entry *ep; X{ Xprintf ("adding %s to %s\n", p, ep->prompt); X if (!*p) X ep->save = ""; X else if (!(ep->save = strdup(p))) { X (void)fprintf(stderr, "chpass: can't save entry.\n"); X return(1); X } X return(0); X} X X/* ARGSUSED */ Xp_hdir(p, pw, ep) X char *p; X struct passwd *pw; X struct entry *ep; X{ X if (!*p) { X (void)fprintf(stderr, "chpass: empty home directory field.\n"); X return(1); X } X if (!(pw->pw_dir = strdup(p))) { X (void)fprintf(stderr, "chpass: can't save entry.\n"); X return(1); X } X return(0); X} X X/* ARGSUSED */ Xp_shell(p, pw, ep) X register char *p; X struct passwd *pw; X struct entry *ep; X{ X char *t, *ok_shell(); X X if (!*p) { X pw->pw_shell = _PATH_BSHELL; X return(0); X } X /* only admin can change from or to "restricted" shells */ X if (uid && pw->pw_shell && !ok_shell(pw->pw_shell)) { X (void)fprintf(stderr, X "chpass: %s: current shell non-standard.\n", pw->pw_shell); X return(1); X } X if (!(t = ok_shell(p))) { X if (uid) { X (void)fprintf(stderr, X "chpass: %s: non-standard shell.\n", p); X return(1); X } X } X else X p = t; X if (!(pw->pw_shell = strdup(p))) { X (void)fprintf(stderr, "chpass: can't save entry.\n"); X return(1); X } X return(0); X} SHAR_EOF if test 4654 -ne "`wc -c < 'field.c'`" then echo shar: "error transmitting 'field.c'" '(should have been 4654 characters)' fi fi echo shar: "extracting 'pathnames.h'" '(1367 characters)' if test -f 'pathnames.h' then echo shar: "will not over-write existing file 'pathnames.h'" else sed 's/^X//' << \SHAR_EOF > 'pathnames.h' X/* X * Copyright (c) 1989 The Regents of the University of California. X * All rights reserved. X * X * Redistribution and use in source and binary forms are permitted X * provided that: (1) source distributions retain this entire copyright X * notice and comment, and (2) distributions including binaries display X * the following acknowledgement: ``This product includes software X * developed by the University of California, Berkeley and its contributors'' X * in the documentation or other materials provided with the distribution X * and in all advertising materials mentioning features or use of this X * software. Neither the name of the University nor the names of its X * contributors may be used to endorse or promote products derived X * from this software without specific prior written permission. X * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR X * IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED X * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. X * X * @(#)pathnames.h 5.3 (Berkeley) 6/1/90 X */ X X/* X * Modified 1/10/91, John F Haugh II X * X * Made changes needed to get chpass to run on System V. X */ X X#define _PATH_TMP "/tmp/passwd.XXXXXX" X#define _PATH_PTMP "/etc/ptmp" X#define _PATH_PASSWD "/etc/passwd" X#define _PATH_MKPASSWD "/etc/mkpasswd" X#define _PATH_BSHELL "/bin/sh" X#define _PATH_SHELLS "/etc/shells" SHAR_EOF if test 1367 -ne "`wc -c < 'pathnames.h'`" then echo shar: "error transmitting 'pathnames.h'" '(should have been 1367 characters)' fi fi echo shar: "extracting 'util.c'" '(3810 characters)' if test -f 'util.c' then echo shar: "will not over-write existing file 'util.c'" else sed 's/^X//' << \SHAR_EOF > 'util.c' X/*- X * Copyright (c) 1988 The Regents of the University of California. X * All rights reserved. X * X * Redistribution and use in source and binary forms are permitted provided X * that: (1) source distributions retain this entire copyright notice and X * comment, and (2) distributions including binaries display the following X * acknowledgement: ``This product includes software developed by the X * University of California, Berkeley and its contributors'' in the X * documentation or other materials provided with the distribution and in X * all advertising materials mentioning features or use of this software. X * Neither the name of the University nor the names of its contributors may X * be used to endorse or promote products derived from this software without X * specific prior written permission. X * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED X * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF X * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. X */ X X/* X * Modified 1/10/91, John F Haugh II X * X * Made changes needed to get chpass to run on System V. X */ X X#ifndef lint Xstatic char sccsid[] = "@(#)util.c 5.13 (Berkeley) 6/29/90"; X#endif /* not lint */ X X#include <sys/types.h> X#include <time.h> X#include <pwd.h> X#include <stdio.h> X#include <string.h> X#include <ctype.h> X#include "chpass.h" X#include "pathnames.h" X Xstatic int dmsize[] = X { -1, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 }; Xstatic char *months[] = X { "January", "February", "March", "April", "May", "June", X "July", "August", "September", "October", "November", X "December", NULL }; X X#define TM_YEAR_BASE (1900) X#define EPOCH_YEAR (1969) X#define DAYSPERLYEAR (366) X#define DAYSPERNYEAR (365) X#define HOURSPERDAY (24) X#define MINSPERHOUR (60) X#define SECSPERMIN (60) X Xstatic FILE *shell_fp; X Xint Xsetusershell () X{ X if (shell_fp) { X rewind (shell_fp); X return 0; X } X return (shell_fp = fopen (_PATH_SHELLS, "r")) == 0 ? -1:0; X} X Xchar * Xgetusershell () X{ X static char shell[BUFSIZ]; X char *cp; X X if (! shell_fp) X return 0; X X if (fgets (shell, sizeof shell, shell_fp) == 0) X return 0; X X if (cp = strrchr (shell, '\n')) X *cp = '\0'; X X return shell; X} X X/* X * print -- X * print out the file for the user to edit; strange side-effect: X * return if the user is allowed to modify their shell. X */ Xprint(fp, pw) X FILE *fp; X struct passwd *pw; X{ X register char *p; X int shellval; X char *bp; X char *getusershell(), *ok_shell(), *ttoa(); X X shellval = 1; X (void)fprintf(fp, "#Changing user database information for %s.\n", X pw->pw_name); X if (!uid) { X (void)fprintf(fp, "Login: %s\n", pw->pw_name); X (void)fprintf(fp, "Password: %s\n", pw->pw_passwd); X (void)fprintf(fp, "Uid [#]: %d\n", pw->pw_uid); X (void)fprintf(fp, "Gid [# or name]: %d\n", pw->pw_gid); X (void)fprintf(fp, "Home directory: %s\n", pw->pw_dir); X (void)fprintf(fp, "Shell: %s\n", X *pw->pw_shell ? pw->pw_shell : _PATH_BSHELL); X } X /* only admin can change "restricted" shells */ X else if (ok_shell(pw->pw_shell)) X (void)fprintf(fp, "Shell: %s\n", X *pw->pw_shell ? pw->pw_shell : _PATH_BSHELL); X else X shellval = 0; X bp = pw->pw_gecos; X p = strtok(bp, ","); X (void)fprintf(fp, "Full Name: %s\n", p ? p : ""); X p = strtok((char *) 0, ","); X (void)fprintf(fp, "Location: %s\n", p ? p : ""); X p = strtok((char *) 0, ","); X (void)fprintf(fp, "Office Phone: %s\n", p ? p : ""); X p = strtok((char *) 0, ","); X (void)fprintf(fp, "Home Phone: %s\n", p ? p : ""); X return(shellval); X} X Xchar * Xok_shell(name) X register char *name; X{ X register char *p, *sh; X char *getusershell(); X X setusershell(); X while (sh = getusershell()) { X if (!strcmp(name, sh)) X return(name); X /* allow just shell name, but use "real" path */ X if ((p = strrchr(sh, '/')) && !strcmp(name, p + 1)) X return(sh); X } X return(NULL); X} SHAR_EOF if test 3810 -ne "`wc -c < 'util.c'`" then echo shar: "error transmitting 'util.c'" '(should have been 3810 characters)' fi fi exit 0 # End of shell archive -- John F. Haugh II UUCP: ...!cs.utexas.edu!rpp386!jfh Ma Bell: (512) 832-8832 Domain: jfh@rpp386.cactus.org "While you are here, your wives and girlfriends are dating handsome American movie and TV stars. Stars like Tom Selleck, Bruce Willis, and Bart Simpson."