1092 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			1092 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			C
		
	
	
		
			Executable File
		
	
	
	
	
/*
 | 
						|
 * Copyright (c) 1980 Regents of the University of California.
 | 
						|
 * All rights reserved.  The Berkeley software License Agreement
 | 
						|
 * specifies the terms and conditions for redistribution.
 | 
						|
 */
 | 
						|
 | 
						|
#ifndef lint
 | 
						|
static char sccsid[] = "@(#)finger.c 1.1 87/12/21 SMI"; /* from 5.8 3/13/86 */
 | 
						|
#endif /* not lint */
 | 
						|
 | 
						|
/*
 | 
						|
 * This is a finger program.  It prints out useful information about users
 | 
						|
 * by digging it up from various system files.
 | 
						|
 *
 | 
						|
 * There are three output formats, all of which give login name, teletype
 | 
						|
 * line number, and login time.  The short output format is reminiscent
 | 
						|
 * of finger on ITS, and gives one line of information per user containing
 | 
						|
 * in addition to the minimum basic requirements (MBR), the full name of
 | 
						|
 * the user, his idle time and location.  The
 | 
						|
 * quick style output is UNIX who-like, giving only name, teletype and
 | 
						|
 * login time.  Finally, the long style output give the same information
 | 
						|
 * as the short (in more legible format), the home directory and shell
 | 
						|
 * of the user, and, if it exits, a copy of the file .plan in the users
 | 
						|
 * home directory.  Finger may be called with or without a list of people
 | 
						|
 * to finger -- if no list is given, all the people currently logged in
 | 
						|
 * are fingered.
 | 
						|
 *
 | 
						|
 * The program is validly called by one of the following:
 | 
						|
 *
 | 
						|
 *	finger			{short form list of users}
 | 
						|
 *	finger -l		{long form list of users}
 | 
						|
 *	finger -b		{briefer long form list of users}
 | 
						|
 *	finger -q		{quick list of users}
 | 
						|
 *	finger -i		{quick list of users with idle times}
 | 
						|
 *	finger namelist		{long format list of specified users}
 | 
						|
 *	finger -s namelist	{short format list of specified users}
 | 
						|
 *	finger -w namelist	{narrow short format list of specified users}
 | 
						|
 *
 | 
						|
 * where 'namelist' is a list of users login names.
 | 
						|
 * The other options can all be given after one '-', or each can have its
 | 
						|
 * own '-'.  The -f option disables the printing of headers for short and
 | 
						|
 * quick outputs.  The -b option briefens long format outputs.  The -p
 | 
						|
 * option turns off plans for long format outputs.
 | 
						|
 */
 | 
						|
 | 
						|
#include <sys/types.h>
 | 
						|
#include <ctype.h>
 | 
						|
#include <errno.h>
 | 
						|
#include <fcntl.h>
 | 
						|
#include <pwd.h>
 | 
						|
#include <stdio.h>
 | 
						|
#include <stdlib.h>
 | 
						|
#include <string.h>
 | 
						|
#include <time.h>
 | 
						|
#include <unistd.h>
 | 
						|
#include <utmp.h>
 | 
						|
#include <sys/ioctl.h>
 | 
						|
#include <sys/stat.h>
 | 
						|
#include <net/gen/in.h>
 | 
						|
#include <net/gen/inet.h>
 | 
						|
#include <net/gen/netdb.h>
 | 
						|
#include <net/gen/socket.h>
 | 
						|
#include <net/gen/tcp.h>
 | 
						|
#include <net/gen/tcp_hdr.h>
 | 
						|
#include <net/gen/tcp_io.h>
 | 
						|
#include <net/hton.h>
 | 
						|
#include <net/netlib.h>
 | 
						|
 | 
						|
#define NONOTHING	1		/* don't say "No plan", or "No mail" */
 | 
						|
 | 
						|
#define NONET	0
 | 
						|
 | 
						|
#define ASTERISK	'*'		/* ignore this in real name */
 | 
						|
#define COMMA		','		/* separator in pw_gecos field */
 | 
						|
#define COMMAND		'-'		/* command line flag char */
 | 
						|
#define SAMENAME	'&'		/* repeat login name in real name */
 | 
						|
#define TALKABLE	0220		/* tty is writable if this mode */
 | 
						|
 | 
						|
struct utmp user;
 | 
						|
#define NMAX sizeof(user.ut_name)
 | 
						|
#define LMAX sizeof(user.ut_line)
 | 
						|
#define HMAX sizeof(user.ut_host)
 | 
						|
 | 
						|
struct person {			/* one for each person fingered */
 | 
						|
	char *name;			/* name */
 | 
						|
	char tty[LMAX+1];		/* null terminated tty line */
 | 
						|
	char host[HMAX+1];		/* null terminated remote host name */
 | 
						|
	long loginat;			/* time of (last) login */
 | 
						|
	long idletime;			/* how long idle (if logged in) */
 | 
						|
	char *realname;			/* pointer to full name */
 | 
						|
	struct passwd *pwd;		/* structure of /etc/passwd stuff */
 | 
						|
	char loggedin;			/* person is logged in */
 | 
						|
	char writable;			/* tty is writable */
 | 
						|
	char original;			/* this is not a duplicate entry */
 | 
						|
	struct person *link;		/* link to next person */
 | 
						|
	char *where;			/* terminal location */
 | 
						|
	char hostt[HMAX+1];		/* login host */
 | 
						|
};
 | 
						|
 | 
						|
char LASTLOG[] = "/usr/adm/lastlog";	/* last login info */
 | 
						|
char USERLOG[] = "/etc/utmp";		/* who is logged in */
 | 
						|
char PLAN[] = "/.plan";			/* what plan file is */
 | 
						|
char PROJ[] = "/.project";		/* what project file */
 | 
						|
	
 | 
						|
int unbrief = 1;			/* -b option default */
 | 
						|
int header = 1;				/* -f option default */
 | 
						|
int hack = 1;				/* -h option default */
 | 
						|
int idle = 0;				/* -i option default */
 | 
						|
int large = 0;				/* -l option default */
 | 
						|
int match = 1;				/* -m option default */
 | 
						|
int plan = 1;				/* -p option default */
 | 
						|
int unquick = 1;			/* -q option default */
 | 
						|
int small = 0;				/* -s option default */
 | 
						|
int wide = 1;				/* -w option default */
 | 
						|
 | 
						|
int unshort;
 | 
						|
int lf;					/* LASTLOG file descriptor */
 | 
						|
struct person *person1;			/* list of people */
 | 
						|
long tloc;				/* current time */
 | 
						|
 | 
						|
#if !_MINIX
 | 
						|
char *strcpy();
 | 
						|
char *ctime();
 | 
						|
#endif
 | 
						|
 | 
						|
char *prog_name;
 | 
						|
 | 
						|
int main (int argc, char *argv[]);
 | 
						|
static void doall(void);
 | 
						|
static void donames(char **args);
 | 
						|
static void print(void);
 | 
						|
static void fwopen(void);
 | 
						|
static void decode(struct person *pers);
 | 
						|
static void fwclose(void);
 | 
						|
static int netfinger (char *name);
 | 
						|
static int matchcmp (char *gname, char *login, char *given);
 | 
						|
static void quickprint (struct person *pers);
 | 
						|
static void shortprint (struct person *pers);
 | 
						|
static void personprint (struct person *pers);
 | 
						|
static int AlreadyPrinted(int uid);
 | 
						|
static int AnyMail (char *name);
 | 
						|
static struct passwd *pwdcopy(struct passwd *pfrom);
 | 
						|
static void findidle (struct person *pers);
 | 
						|
static int ltimeprint (char *dt, long *before, char *after);
 | 
						|
static void stimeprint (long *dt);
 | 
						|
static void findwhen (struct person *pers);
 | 
						|
static int namecmp (char *name1, char *name2);
 | 
						|
 | 
						|
main(argc, argv)
 | 
						|
	int argc;
 | 
						|
	register char **argv;
 | 
						|
{
 | 
						|
	FILE *fp;
 | 
						|
	register char *s;
 | 
						|
 | 
						|
	prog_name= argv[0];
 | 
						|
 | 
						|
	/* parse command line for (optional) arguments */
 | 
						|
	while (*++argv && **argv == COMMAND)
 | 
						|
		for (s = *argv + 1; *s; s++)
 | 
						|
			switch (*s) {
 | 
						|
			case 'b':
 | 
						|
				unbrief = 0;
 | 
						|
				break;
 | 
						|
			case 'f':
 | 
						|
				header = 0;
 | 
						|
				break;
 | 
						|
			case 'h':
 | 
						|
				hack = 0;
 | 
						|
				break;
 | 
						|
			case 'i':
 | 
						|
				idle = 1;
 | 
						|
				unquick = 0;
 | 
						|
				break;
 | 
						|
			case 'l':
 | 
						|
				large = 1;
 | 
						|
				break;
 | 
						|
			case 'm':
 | 
						|
				match = 0;
 | 
						|
				break;
 | 
						|
			case 'p':
 | 
						|
				plan = 0;
 | 
						|
				break;
 | 
						|
			case 'q':
 | 
						|
				unquick = 0;
 | 
						|
				break;
 | 
						|
			case 's':
 | 
						|
				small = 1;
 | 
						|
				break;
 | 
						|
			case 'w':
 | 
						|
				wide = 0;
 | 
						|
				break;
 | 
						|
			default:
 | 
						|
				fprintf(stderr, "Usage: finger [-bfhilmpqsw] [login1 [login2 ...] ]\n");
 | 
						|
				exit(1);
 | 
						|
			}
 | 
						|
	if (unquick || idle)
 | 
						|
		time(&tloc);
 | 
						|
	/*
 | 
						|
	 * *argv == 0 means no names given
 | 
						|
	 */
 | 
						|
	if (*argv == 0)
 | 
						|
		doall();
 | 
						|
	else
 | 
						|
		donames(argv);
 | 
						|
	if (person1)
 | 
						|
		print();
 | 
						|
	exit(0);
 | 
						|
}
 | 
						|
 | 
						|
static void doall()
 | 
						|
{
 | 
						|
	register struct person *p;
 | 
						|
	register struct passwd *pw;
 | 
						|
	int uf;
 | 
						|
	char name[NMAX + 1];
 | 
						|
 | 
						|
	unshort = large;
 | 
						|
	if ((uf = open(USERLOG, 0)) < 0) {
 | 
						|
		fprintf(stderr, "finger: error opening %s\n", USERLOG);
 | 
						|
		exit(2);
 | 
						|
	}
 | 
						|
	if (unquick) {
 | 
						|
		setpwent();
 | 
						|
		fwopen();
 | 
						|
	}
 | 
						|
	while (read(uf, (char *)&user, sizeof user) == sizeof user) {
 | 
						|
		if (user.ut_name[0] == 0)
 | 
						|
			continue;
 | 
						|
		if (person1 == 0)
 | 
						|
			p = person1 = (struct person *) malloc(sizeof *p);
 | 
						|
		else {
 | 
						|
			p->link = (struct person *) malloc(sizeof *p);
 | 
						|
			p = p->link;
 | 
						|
		}
 | 
						|
		bcopy(user.ut_name, name, NMAX);
 | 
						|
		name[NMAX] = 0;
 | 
						|
		bcopy(user.ut_line, p->tty, LMAX);
 | 
						|
		p->tty[LMAX] = 0;
 | 
						|
		bcopy(user.ut_host, p->host, HMAX);
 | 
						|
		p->host[HMAX] = 0;
 | 
						|
		p->loginat = user.ut_time;
 | 
						|
		p->pwd = 0;
 | 
						|
		p->loggedin = 1;
 | 
						|
		p->where = NULL;
 | 
						|
		if (unquick && (pw = getpwnam(name))) {
 | 
						|
			p->pwd = pwdcopy(pw);
 | 
						|
			decode(p);
 | 
						|
			p->name = p->pwd->pw_name;
 | 
						|
		} else
 | 
						|
			p->name = strcpy(malloc(strlen(name) + 1), name);
 | 
						|
	}
 | 
						|
	if (unquick) {
 | 
						|
		fwclose();
 | 
						|
		endpwent();
 | 
						|
	}
 | 
						|
	close(uf);
 | 
						|
	if (person1 == 0) {
 | 
						|
		printf("No one logged on\n");
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	p->link = 0;
 | 
						|
}
 | 
						|
 | 
						|
static void donames(argv)
 | 
						|
	char **argv;
 | 
						|
{
 | 
						|
	register struct person *p;
 | 
						|
	register struct passwd *pw;
 | 
						|
	int uf;
 | 
						|
 | 
						|
	/*
 | 
						|
	 * get names from command line and check to see if they're
 | 
						|
	 * logged in
 | 
						|
	 */
 | 
						|
	unshort = !small;
 | 
						|
	for (; *argv != 0; argv++) {
 | 
						|
		if (netfinger(*argv))
 | 
						|
			continue;
 | 
						|
		if (person1 == 0)
 | 
						|
			p = person1 = (struct person *) malloc(sizeof *p);
 | 
						|
		else {
 | 
						|
			p->link = (struct person *) malloc(sizeof *p);
 | 
						|
			p = p->link;
 | 
						|
		}
 | 
						|
		p->name = *argv;
 | 
						|
		p->loggedin = 0;
 | 
						|
		p->original = 1;
 | 
						|
		p->pwd = 0;
 | 
						|
	}
 | 
						|
	if (person1 == 0)
 | 
						|
		return;
 | 
						|
	p->link = 0;
 | 
						|
	/*
 | 
						|
	 * if we are doing it, read /etc/passwd for the useful info
 | 
						|
	 */
 | 
						|
	if (unquick) {
 | 
						|
		setpwent();
 | 
						|
		if (!match) {
 | 
						|
			for (p = person1; p != 0; p = p->link)
 | 
						|
				if (pw = getpwnam(p->name))
 | 
						|
					p->pwd = pwdcopy(pw);
 | 
						|
		} else while ((pw = getpwent()) != 0) {
 | 
						|
			for (p = person1; p != 0; p = p->link) {
 | 
						|
				if (!p->original)
 | 
						|
					continue;
 | 
						|
				if (strcmp(p->name, pw->pw_name) != 0 &&
 | 
						|
				    !matchcmp(pw->pw_gecos, pw->pw_name, p->name))
 | 
						|
					continue;
 | 
						|
				if (p->pwd == 0)
 | 
						|
					p->pwd = pwdcopy(pw);
 | 
						|
				else {
 | 
						|
					struct person *new;
 | 
						|
					/*
 | 
						|
					 * handle multiple login names, insert
 | 
						|
					 * new "duplicate" entry behind
 | 
						|
					 */
 | 
						|
					new = (struct person *)
 | 
						|
						malloc(sizeof *new);
 | 
						|
					new->pwd = pwdcopy(pw);
 | 
						|
					new->name = p->name;
 | 
						|
					new->original = 1;
 | 
						|
					new->loggedin = 0;
 | 
						|
					new->link = p->link;
 | 
						|
					p->original = 0;
 | 
						|
					p->link = new;
 | 
						|
					p = new;
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
		endpwent();
 | 
						|
	}
 | 
						|
	/* Now get login information */
 | 
						|
	if ((uf = open(USERLOG, 0)) < 0) {
 | 
						|
		fprintf(stderr, "finger: error opening %s\n", USERLOG);
 | 
						|
		exit(2);
 | 
						|
	}
 | 
						|
	while (read(uf, (char *)&user, sizeof user) == sizeof user) {
 | 
						|
		if (*user.ut_name == 0)
 | 
						|
			continue;
 | 
						|
		for (p = person1; p != 0; p = p->link) {
 | 
						|
			if (p->loggedin == 2)
 | 
						|
				continue;
 | 
						|
			if (strncmp(p->pwd ? p->pwd->pw_name : p->name,
 | 
						|
				    user.ut_name, NMAX) != 0)
 | 
						|
				continue;
 | 
						|
			if (p->loggedin == 0) {
 | 
						|
				bcopy(user.ut_line, p->tty, LMAX);
 | 
						|
				p->tty[LMAX] = 0;
 | 
						|
				bcopy(user.ut_host, p->host, HMAX);
 | 
						|
				p->host[HMAX] = 0;
 | 
						|
				p->loginat = user.ut_time;
 | 
						|
				p->loggedin = 1;
 | 
						|
			} else {	/* p->loggedin == 1 */
 | 
						|
				struct person *new;
 | 
						|
				new = (struct person *) malloc(sizeof *new);
 | 
						|
				new->name = p->name;
 | 
						|
				bcopy(user.ut_line, new->tty, LMAX);
 | 
						|
				new->tty[LMAX] = 0;
 | 
						|
				bcopy(user.ut_host, new->host, HMAX);
 | 
						|
				new->host[HMAX] = 0;
 | 
						|
				new->loginat = user.ut_time;
 | 
						|
				new->pwd = p->pwd;
 | 
						|
				new->loggedin = 1;
 | 
						|
				new->original = 0;
 | 
						|
				new->link = p->link;
 | 
						|
				p->loggedin = 2;
 | 
						|
				p->link = new;
 | 
						|
				p = new;
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
	close(uf);
 | 
						|
	if (unquick) {
 | 
						|
		fwopen();
 | 
						|
		for (p = person1; p != 0; p = p->link)
 | 
						|
			decode(p);
 | 
						|
		fwclose();
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void print()
 | 
						|
{
 | 
						|
	register FILE *fp;
 | 
						|
	register struct person *p;
 | 
						|
	register char *s;
 | 
						|
	register c;
 | 
						|
 | 
						|
	/*
 | 
						|
	 * print out what we got
 | 
						|
	 */
 | 
						|
	if (header) {
 | 
						|
		if (unquick) {
 | 
						|
			if (!unshort)
 | 
						|
				if (wide)
 | 
						|
					printf("Login       Name              TTY Idle    When            Where\n");
 | 
						|
				else
 | 
						|
					printf("Login    TTY Idle    When            Where\n");
 | 
						|
		} else {
 | 
						|
			printf("Login      TTY            When");
 | 
						|
			if (idle)
 | 
						|
				printf("             Idle");
 | 
						|
			putchar('\n');
 | 
						|
		}
 | 
						|
	}
 | 
						|
	for (p = person1; p != 0; p = p->link) {
 | 
						|
		if (!unquick) {
 | 
						|
			quickprint(p);
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
		if (!unshort) {
 | 
						|
			shortprint(p);
 | 
						|
			continue;
 | 
						|
		}
 | 
						|
		personprint(p);
 | 
						|
		if (p->pwd != 0 && !AlreadyPrinted(p->pwd->pw_uid)) {
 | 
						|
			AnyMail(p->pwd->pw_name);
 | 
						|
			if (hack) {
 | 
						|
				s = malloc(strlen(p->pwd->pw_dir) +
 | 
						|
					sizeof PROJ);
 | 
						|
				strcpy(s, p->pwd->pw_dir);
 | 
						|
				strcat(s, PROJ);
 | 
						|
				if ((fp = fopen(s, "r")) != 0) {
 | 
						|
					printf("Project: ");
 | 
						|
					while ((c = getc(fp)) != EOF) {
 | 
						|
						if (c == '\n')
 | 
						|
							break;
 | 
						|
						if (isprint(c) || isspace(c))
 | 
						|
							putchar(c);
 | 
						|
						else
 | 
						|
							putchar(c ^ 100);
 | 
						|
					}
 | 
						|
					fclose(fp);
 | 
						|
					putchar('\n');
 | 
						|
				}
 | 
						|
				free(s);
 | 
						|
			}
 | 
						|
			if (plan) {
 | 
						|
				s = malloc(strlen(p->pwd->pw_dir) +
 | 
						|
					sizeof PLAN);
 | 
						|
				strcpy(s, p->pwd->pw_dir);
 | 
						|
				strcat(s, PLAN);
 | 
						|
				if ((fp = fopen(s, "r")) == 0) {
 | 
						|
					if (!NONOTHING) printf("No Plan.\n");
 | 
						|
				} else {
 | 
						|
					printf("Plan:\n");
 | 
						|
					while ((c = getc(fp)) != EOF)
 | 
						|
						if (isprint(c) || isspace(c))
 | 
						|
							putchar(c);
 | 
						|
						else
 | 
						|
							putchar(c ^ 100);
 | 
						|
					fclose(fp);
 | 
						|
				}
 | 
						|
				free(s);
 | 
						|
			}
 | 
						|
		}
 | 
						|
		if (p->link != 0)
 | 
						|
			putchar('\n');
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * Duplicate a pwd entry.
 | 
						|
 * Note: Only the useful things (what the program currently uses) are copied.
 | 
						|
 */
 | 
						|
static struct passwd *
 | 
						|
pwdcopy(pfrom)
 | 
						|
	register struct passwd *pfrom;
 | 
						|
{
 | 
						|
	register struct passwd *pto;
 | 
						|
 | 
						|
	pto = (struct passwd *) malloc(sizeof *pto);
 | 
						|
#define savestr(s) strcpy(malloc(strlen(s) + 1), s)
 | 
						|
	pto->pw_name = savestr(pfrom->pw_name);
 | 
						|
	pto->pw_uid = pfrom->pw_uid;
 | 
						|
	pto->pw_gecos = savestr(pfrom->pw_gecos);
 | 
						|
	pto->pw_dir = savestr(pfrom->pw_dir);
 | 
						|
	pto->pw_shell = savestr(pfrom->pw_shell);
 | 
						|
#undef savestr
 | 
						|
	return pto;
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * print out information on quick format giving just name, tty, login time
 | 
						|
 * and idle time if idle is set.
 | 
						|
 */
 | 
						|
static void quickprint(pers)
 | 
						|
	register struct person *pers;
 | 
						|
{
 | 
						|
	printf("%-*.*s  ", NMAX, NMAX, pers->name);
 | 
						|
	if (pers->loggedin) {
 | 
						|
		if (idle) {
 | 
						|
			findidle(pers);
 | 
						|
			printf("%c%-*s %-16.16s", pers->writable ? ' ' : '*',
 | 
						|
				LMAX, pers->tty, ctime(&pers->loginat));
 | 
						|
			ltimeprint("   ", &pers->idletime, "");
 | 
						|
		} else
 | 
						|
			printf(" %-*s %-16.16s", LMAX,
 | 
						|
				pers->tty, ctime(&pers->loginat));
 | 
						|
		putchar('\n');
 | 
						|
	} else
 | 
						|
		printf("          Not Logged In\n");
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * print out information in short format, giving login name, full name,
 | 
						|
 * tty, idle time, login time, and host.
 | 
						|
 */
 | 
						|
static void shortprint(pers)
 | 
						|
	register struct person *pers;
 | 
						|
{
 | 
						|
	char *p;
 | 
						|
	char dialup;
 | 
						|
 | 
						|
	if (pers->pwd == 0) {
 | 
						|
		printf("%-15s       ???\n", pers->name);
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	printf("%-*s", NMAX, pers->pwd->pw_name);
 | 
						|
	dialup = 0;
 | 
						|
	if (wide) {
 | 
						|
		if (pers->realname)
 | 
						|
			printf(" %-20.20s", pers->realname);
 | 
						|
		else
 | 
						|
			printf("        ???          ");
 | 
						|
	}
 | 
						|
	putchar(' ');
 | 
						|
	if (pers->loggedin && !pers->writable)
 | 
						|
		putchar('*');
 | 
						|
	else
 | 
						|
		putchar(' ');
 | 
						|
	if (*pers->tty) {
 | 
						|
		if (pers->tty[0] == 't' && pers->tty[1] == 't' &&
 | 
						|
		    pers->tty[2] == 'y') {
 | 
						|
			if (pers->tty[3] == 'd' && pers->loggedin)
 | 
						|
				dialup = 1;
 | 
						|
			printf("%-2.2s ", pers->tty + 3);
 | 
						|
		} else
 | 
						|
			printf("%-2.2s ", pers->tty);
 | 
						|
	} else
 | 
						|
		printf("   ");
 | 
						|
	p = ctime(&pers->loginat);
 | 
						|
	if (pers->loggedin) {
 | 
						|
		stimeprint(&pers->idletime);
 | 
						|
		printf(" %3.3s %-5.5s ", p, p + 11);
 | 
						|
	} else if (pers->loginat == 0)
 | 
						|
		printf(" < .  .  .  . >");
 | 
						|
	else if (tloc - pers->loginat >= 180L * 24 * 60 * 60)
 | 
						|
		printf(" <%-6.6s, %-4.4s>", p + 4, p + 20);
 | 
						|
	else
 | 
						|
		printf(" <%-12.12s>", p + 4);
 | 
						|
	if (pers->host[0])
 | 
						|
		printf(" %-20.20s", pers->host);
 | 
						|
	putchar('\n');
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * print out a person in long format giving all possible information.
 | 
						|
 * directory and shell are inhibited if unbrief is clear.
 | 
						|
 */
 | 
						|
static void
 | 
						|
personprint(pers)
 | 
						|
	register struct person *pers;
 | 
						|
{
 | 
						|
	if (pers->pwd == 0) {
 | 
						|
		printf("Login name: %-10s\t\t\tIn real life: ???\n",
 | 
						|
			pers->name);
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	printf("Login name: %-10s", pers->pwd->pw_name);
 | 
						|
	if (pers->loggedin && !pers->writable)
 | 
						|
		printf("	(messages off)	");
 | 
						|
	else
 | 
						|
		printf("			");
 | 
						|
	if (pers->realname)
 | 
						|
		printf("In real life: %s", pers->realname);
 | 
						|
	if (unbrief) {
 | 
						|
		printf("\nDirectory: %-25s", pers->pwd->pw_dir);
 | 
						|
		if (*pers->pwd->pw_shell)
 | 
						|
			printf("\tShell: %-s", pers->pwd->pw_shell);
 | 
						|
	}
 | 
						|
	if (pers->loggedin) {
 | 
						|
		register char *ep = ctime(&pers->loginat);
 | 
						|
		if (*pers->host) {
 | 
						|
			printf("\nOn since %15.15s on %s from %s",
 | 
						|
				&ep[4], pers->tty, pers->host);
 | 
						|
			ltimeprint("\n", &pers->idletime, " Idle Time");
 | 
						|
		} else {
 | 
						|
			printf("\nOn since %15.15s on %-*s",
 | 
						|
				&ep[4], LMAX, pers->tty);
 | 
						|
			ltimeprint("\t", &pers->idletime, " Idle Time");
 | 
						|
		}
 | 
						|
	} else if (pers->loginat == 0) {
 | 
						|
		if (lf >= 0) printf("\nNever logged in.");
 | 
						|
	} else if (tloc - pers->loginat > 180L * 24 * 60 * 60) {
 | 
						|
		register char *ep = ctime(&pers->loginat);
 | 
						|
		printf("\nLast login %10.10s, %4.4s on %s",
 | 
						|
			ep, ep+20, pers->tty);
 | 
						|
		if (*pers->host)
 | 
						|
			printf(" from %s", pers->host);
 | 
						|
	} else {
 | 
						|
		register char *ep = ctime(&pers->loginat);
 | 
						|
		printf("\nLast login %16.16s on %s", ep, pers->tty);
 | 
						|
		if (*pers->host)
 | 
						|
			printf(" from %s", pers->host);
 | 
						|
	}
 | 
						|
	putchar('\n');
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
/*
 | 
						|
 * decode the information in the gecos field of /etc/passwd
 | 
						|
 */
 | 
						|
static void
 | 
						|
decode(pers)
 | 
						|
	register struct person *pers;
 | 
						|
{
 | 
						|
	char buffer[256];
 | 
						|
	register char *bp, *gp, *lp;
 | 
						|
	int alldigits;
 | 
						|
	int hasspace;
 | 
						|
	int len;
 | 
						|
 | 
						|
	pers->realname = 0;
 | 
						|
	if (pers->pwd == 0)
 | 
						|
		return;
 | 
						|
	gp = pers->pwd->pw_gecos;
 | 
						|
	bp = buffer;
 | 
						|
	if (*gp == ASTERISK)
 | 
						|
		gp++;
 | 
						|
	while (*gp && *gp != COMMA) 			/* name */
 | 
						|
		if (*gp == SAMENAME) {
 | 
						|
			lp = pers->pwd->pw_name;
 | 
						|
			if (islower(*lp))
 | 
						|
				*bp++ = toupper(*lp++);
 | 
						|
			while (*bp++ = *lp++)
 | 
						|
				;
 | 
						|
			bp--;
 | 
						|
			gp++;
 | 
						|
		} else
 | 
						|
			*bp++ = *gp++;
 | 
						|
	*bp++ = 0;
 | 
						|
	if ((len = bp - buffer) > 1)
 | 
						|
		pers->realname = strcpy(malloc(len), buffer);
 | 
						|
	if (pers->loggedin)
 | 
						|
		findidle(pers);
 | 
						|
	else
 | 
						|
		findwhen(pers);
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * find the last log in of a user by checking the LASTLOG file.
 | 
						|
 * the entry is indexed by the uid, so this can only be done if
 | 
						|
 * the uid is known (which it isn't in quick mode)
 | 
						|
 */
 | 
						|
 | 
						|
static void
 | 
						|
fwopen()
 | 
						|
{
 | 
						|
	if ((lf = open(LASTLOG, 0)) < 0) {
 | 
						|
		if (errno == ENOENT) return;
 | 
						|
		fprintf(stderr, "finger: %s open error\n", LASTLOG);
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void
 | 
						|
findwhen(pers)
 | 
						|
	register struct person *pers;
 | 
						|
{
 | 
						|
	struct utmp ll;
 | 
						|
#define ll_line ut_line
 | 
						|
#define ll_host ut_host
 | 
						|
#define ll_time ut_time
 | 
						|
 | 
						|
	int i;
 | 
						|
 | 
						|
	if (lf >= 0) {
 | 
						|
		lseek(lf, (long)pers->pwd->pw_uid * sizeof ll, 0);
 | 
						|
		if ((i = read(lf, (char *)&ll, sizeof ll)) == sizeof ll) {
 | 
						|
			bcopy(ll.ll_line, pers->tty, LMAX);
 | 
						|
			pers->tty[LMAX] = 0;
 | 
						|
			bcopy(ll.ll_host, pers->host, HMAX);
 | 
						|
			pers->host[HMAX] = 0;
 | 
						|
			pers->loginat = ll.ll_time;
 | 
						|
		} else {
 | 
						|
			if (i != 0)
 | 
						|
				fprintf(stderr, "finger: %s read error\n",
 | 
						|
					LASTLOG);
 | 
						|
			pers->tty[0] = 0;
 | 
						|
			pers->host[0] = 0;
 | 
						|
			pers->loginat = 0L;
 | 
						|
		}
 | 
						|
	} else {
 | 
						|
		pers->tty[0] = 0;
 | 
						|
		pers->host[0] = 0;
 | 
						|
		pers->loginat = 0L;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
static void fwclose()
 | 
						|
{
 | 
						|
	if (lf >= 0)
 | 
						|
		close(lf);
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * find the idle time of a user by doing a stat on /dev/tty??,
 | 
						|
 * where tty?? has been gotten from USERLOG, supposedly.
 | 
						|
 */
 | 
						|
static void
 | 
						|
findidle(pers)
 | 
						|
	register struct person *pers;
 | 
						|
{
 | 
						|
	struct stat ttystatus;
 | 
						|
	static char buffer[20] = "/dev/";
 | 
						|
	long t;
 | 
						|
#define TTYLEN 5
 | 
						|
 | 
						|
	strcpy(buffer + TTYLEN, pers->tty);
 | 
						|
	buffer[TTYLEN+LMAX] = 0;
 | 
						|
	if (stat(buffer, &ttystatus) < 0) {
 | 
						|
		fprintf(stderr, "finger: Can't stat %s\n", buffer);
 | 
						|
		exit(4);
 | 
						|
	}
 | 
						|
	time(&t);
 | 
						|
	if (t < ttystatus.st_atime)
 | 
						|
		pers->idletime = 0L;
 | 
						|
	else
 | 
						|
		pers->idletime = t - ttystatus.st_atime;
 | 
						|
	pers->writable = (ttystatus.st_mode & TALKABLE) == TALKABLE;
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * print idle time in short format; this program always prints 4 characters;
 | 
						|
 * if the idle time is zero, it prints 4 blanks.
 | 
						|
 */
 | 
						|
static void
 | 
						|
stimeprint(dt)
 | 
						|
	long *dt;
 | 
						|
{
 | 
						|
	register struct tm *delta;
 | 
						|
 | 
						|
	delta = gmtime(dt);
 | 
						|
	if (delta->tm_yday == 0)
 | 
						|
		if (delta->tm_hour == 0)
 | 
						|
			if (delta->tm_min == 0)
 | 
						|
				printf("    ");
 | 
						|
			else
 | 
						|
				printf("  %2d", delta->tm_min);
 | 
						|
		else
 | 
						|
			if (delta->tm_hour >= 10)
 | 
						|
				printf("%3d:", delta->tm_hour);
 | 
						|
			else
 | 
						|
				printf("%1d:%02d",
 | 
						|
					delta->tm_hour, delta->tm_min);
 | 
						|
	else
 | 
						|
		printf("%3dd", delta->tm_yday);
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * print idle time in long format with care being taken not to pluralize
 | 
						|
 * 1 minutes or 1 hours or 1 days.
 | 
						|
 * print "prefix" first.
 | 
						|
 */
 | 
						|
static int
 | 
						|
ltimeprint(before, dt, after)
 | 
						|
	long *dt;
 | 
						|
	char *before, *after;
 | 
						|
{
 | 
						|
	register struct tm *delta;
 | 
						|
 | 
						|
	delta = gmtime(dt);
 | 
						|
	if (delta->tm_yday == 0 && delta->tm_hour == 0 && delta->tm_min == 0 &&
 | 
						|
	    delta->tm_sec <= 10)
 | 
						|
		return (0);
 | 
						|
	printf("%s", before);
 | 
						|
	if (delta->tm_yday >= 10)
 | 
						|
		printf("%d days", delta->tm_yday);
 | 
						|
	else if (delta->tm_yday > 0)
 | 
						|
		printf("%d day%s %d hour%s",
 | 
						|
			delta->tm_yday, delta->tm_yday == 1 ? "" : "s",
 | 
						|
			delta->tm_hour, delta->tm_hour == 1 ? "" : "s");
 | 
						|
	else
 | 
						|
		if (delta->tm_hour >= 10)
 | 
						|
			printf("%d hours", delta->tm_hour);
 | 
						|
		else if (delta->tm_hour > 0)
 | 
						|
			printf("%d hour%s %d minute%s",
 | 
						|
				delta->tm_hour, delta->tm_hour == 1 ? "" : "s",
 | 
						|
				delta->tm_min, delta->tm_min == 1 ? "" : "s");
 | 
						|
		else
 | 
						|
			if (delta->tm_min >= 10)
 | 
						|
				printf("%2d minutes", delta->tm_min);
 | 
						|
			else if (delta->tm_min == 0)
 | 
						|
				printf("%2d seconds", delta->tm_sec);
 | 
						|
			else
 | 
						|
				printf("%d minute%s %d second%s",
 | 
						|
					delta->tm_min,
 | 
						|
					delta->tm_min == 1 ? "" : "s",
 | 
						|
					delta->tm_sec,
 | 
						|
					delta->tm_sec == 1 ? "" : "s");
 | 
						|
	printf("%s", after);
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
matchcmp(gname, login, given)
 | 
						|
	register char *gname;
 | 
						|
	char *login;
 | 
						|
	char *given;
 | 
						|
{
 | 
						|
	char buffer[100];
 | 
						|
	register char *bp, *lp;
 | 
						|
	register c;
 | 
						|
 | 
						|
	if (*gname == ASTERISK)
 | 
						|
		gname++;
 | 
						|
	lp = 0;
 | 
						|
	bp = buffer;
 | 
						|
	for (;;)
 | 
						|
		switch (c = *gname++) {
 | 
						|
		case SAMENAME:
 | 
						|
			for (lp = login; bp < buffer + sizeof buffer
 | 
						|
					 && (*bp++ = *lp++);)
 | 
						|
				;
 | 
						|
			bp--;
 | 
						|
			break;
 | 
						|
		case ' ':
 | 
						|
		case COMMA:
 | 
						|
		case '\0':
 | 
						|
			*bp = 0;
 | 
						|
			if (namecmp(buffer, given))
 | 
						|
				return (1);
 | 
						|
			if (c == COMMA || c == 0)
 | 
						|
				return (0);
 | 
						|
			bp = buffer;
 | 
						|
			break;
 | 
						|
		default:
 | 
						|
			if (bp < buffer + sizeof buffer)
 | 
						|
				*bp++ = c;
 | 
						|
		}
 | 
						|
	/*NOTREACHED*/
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
namecmp(name1, name2)
 | 
						|
	register char *name1, *name2;
 | 
						|
{
 | 
						|
	register c1, c2;
 | 
						|
 | 
						|
	for (;;) {
 | 
						|
		c1 = *name1++;
 | 
						|
		if (islower(c1))
 | 
						|
			c1 = toupper(c1);
 | 
						|
		c2 = *name2++;
 | 
						|
		if (islower(c2))
 | 
						|
			c2 = toupper(c2);
 | 
						|
		if (c1 != c2)
 | 
						|
			break;
 | 
						|
		if (c1 == 0)
 | 
						|
			return (1);
 | 
						|
	}
 | 
						|
	if (!c1) {
 | 
						|
		for (name2--; isdigit(*name2); name2++)
 | 
						|
			;
 | 
						|
		if (*name2 == 0)
 | 
						|
			return (1);
 | 
						|
	} else if (!c2) {
 | 
						|
		for (name1--; isdigit(*name1); name1++)
 | 
						|
			;
 | 
						|
		if (*name2 == 0)
 | 
						|
			return (1);
 | 
						|
	}
 | 
						|
	return (0);
 | 
						|
}
 | 
						|
 | 
						|
#if NONET
 | 
						|
static int
 | 
						|
netfinger(name)
 | 
						|
char *name;
 | 
						|
{
 | 
						|
	return 0;
 | 
						|
}
 | 
						|
#else
 | 
						|
static int
 | 
						|
netfinger(name)
 | 
						|
	char *name;
 | 
						|
{
 | 
						|
	char *host;
 | 
						|
	char fname[100];
 | 
						|
	struct hostent *hp;
 | 
						|
	struct servent *sp;
 | 
						|
	int s, result;
 | 
						|
#if !_MINIX
 | 
						|
	char *rindex();
 | 
						|
#endif
 | 
						|
	register FILE *f;
 | 
						|
	register int c;
 | 
						|
	register int lastc;
 | 
						|
	nwio_tcpconf_t tcpconf;
 | 
						|
	nwio_tcpcl_t tcpconnopt;
 | 
						|
	char *tcp_device;
 | 
						|
 | 
						|
	if (name == NULL)
 | 
						|
		return (0);
 | 
						|
	host = rindex(name, '@');
 | 
						|
	if (host == NULL)
 | 
						|
		return (0);
 | 
						|
	*host++ = 0;
 | 
						|
	hp = gethostbyname(host);
 | 
						|
	if (hp == NULL) {
 | 
						|
		static struct hostent def;
 | 
						|
		static ipaddr_t defaddr;
 | 
						|
		static char namebuf[128];
 | 
						|
 | 
						|
		defaddr = inet_addr(host);
 | 
						|
		if (defaddr == -1) {
 | 
						|
			printf("unknown host: %s\n", host);
 | 
						|
			return (1);
 | 
						|
		}
 | 
						|
		strcpy(namebuf, host);
 | 
						|
		def.h_name = namebuf;
 | 
						|
		def.h_addr = (char *)&defaddr;
 | 
						|
		def.h_length = sizeof (ipaddr_t);
 | 
						|
		def.h_addrtype = AF_INET;
 | 
						|
		def.h_aliases = 0;
 | 
						|
		hp = &def;
 | 
						|
	}
 | 
						|
	printf("[%s] ", hp->h_name);
 | 
						|
	fflush(stdout);
 | 
						|
 | 
						|
	tcp_device= getenv("TCP_DEVICE");
 | 
						|
	if (tcp_device == NULL)
 | 
						|
		tcp_device= TCP_DEVICE;
 | 
						|
	s= open (tcp_device, O_RDWR);
 | 
						|
	if (s == -1)
 | 
						|
	{
 | 
						|
		fprintf(stderr, "%s: unable to open %s (%s)\n",
 | 
						|
			prog_name, tcp_device, strerror(errno));
 | 
						|
		exit(1);
 | 
						|
	}
 | 
						|
	tcpconf.nwtc_flags= NWTC_LP_SEL | NWTC_SET_RA | NWTC_SET_RP;
 | 
						|
	tcpconf.nwtc_remaddr= *(ipaddr_t *)hp->h_addr;
 | 
						|
	tcpconf.nwtc_remport= htons(TCPPORT_FINGER);
 | 
						|
 | 
						|
	result= ioctl (s, NWIOSTCPCONF, &tcpconf);
 | 
						|
	if (result<0)
 | 
						|
	{
 | 
						|
		fprintf(stderr, "%s\n", strerror(errno));
 | 
						|
		exit(1);
 | 
						|
	}
 | 
						|
 | 
						|
	tcpconnopt.nwtcl_flags= 0;
 | 
						|
 | 
						|
	do
 | 
						|
	{
 | 
						|
		result= ioctl (s, NWIOTCPCONN, &tcpconnopt);
 | 
						|
		if (result<0 && errno== EAGAIN)
 | 
						|
		{
 | 
						|
			fprintf(stderr, "got EAGAIN error, sleeping 2s\n");
 | 
						|
			sleep(2);
 | 
						|
		}
 | 
						|
	} while (result<0 && errno == EAGAIN);
 | 
						|
	if (result<0)
 | 
						|
	{
 | 
						|
		fprintf(stderr, "%s\n", strerror(errno));
 | 
						|
		exit(1);
 | 
						|
	}
 | 
						|
	printf("\r\n");
 | 
						|
	if (large) write(s, "/W ", 3);
 | 
						|
	write(s, name, strlen(name));
 | 
						|
	write(s, "\r\n", 2);
 | 
						|
	f = fdopen(s, "r");
 | 
						|
	while ((c = getc(f)) != EOF) {
 | 
						|
/*
 | 
						|
		switch(c) {
 | 
						|
		case 0210:
 | 
						|
		case 0211:
 | 
						|
		case 0212:
 | 
						|
		case 0214:
 | 
						|
			c -= 0200;
 | 
						|
			break;
 | 
						|
		case 0215:
 | 
						|
			c = '\n';
 | 
						|
			break;
 | 
						|
		}
 | 
						|
*/
 | 
						|
		c &= ~0200;
 | 
						|
		if (c == '\r')
 | 
						|
		{
 | 
						|
			c= getc(f) & ~0200;
 | 
						|
			if (c == '\012')
 | 
						|
			{
 | 
						|
				lastc= c;
 | 
						|
				putchar('\n');
 | 
						|
				continue;
 | 
						|
			}
 | 
						|
			else
 | 
						|
				putchar('\r');
 | 
						|
		}
 | 
						|
		lastc = c;
 | 
						|
		if (isprint(c) || isspace(c))
 | 
						|
			putchar(c);
 | 
						|
		else
 | 
						|
			putchar(c ^ 100);
 | 
						|
	}
 | 
						|
	if (lastc != '\n')
 | 
						|
		putchar('\n');
 | 
						|
	(void)fclose(f);
 | 
						|
	return (1);
 | 
						|
}
 | 
						|
#endif
 | 
						|
 | 
						|
/*
 | 
						|
 *	AnyMail - takes a username (string pointer thereto), and
 | 
						|
 *	prints on standard output whether there is any unread mail,
 | 
						|
 *	and if so, how old it is.	(JCM@Shasta 15 March 80)
 | 
						|
 */
 | 
						|
#define preamble "/usr/spool/mail/"	/* mailboxes are there */
 | 
						|
static int
 | 
						|
AnyMail(name)
 | 
						|
char *name;
 | 
						|
{
 | 
						|
	struct stat buf;		/* space for file status buffer */
 | 
						|
	char *mbxdir = preamble; 	/* string with path preamble */
 | 
						|
	char *mbxpath;			/* space for entire pathname */
 | 
						|
 | 
						|
#if !_MINIX
 | 
						|
	char *ctime();			/* convert longword time to ascii */
 | 
						|
#endif
 | 
						|
	char *timestr;
 | 
						|
 | 
						|
	mbxpath = malloc(strlen(name) + strlen(preamble) + 1);
 | 
						|
 | 
						|
	strcpy(mbxpath, mbxdir);	/* copy preamble into path name */
 | 
						|
	strcat(mbxpath, name);		/* concatenate user name to path */
 | 
						|
 | 
						|
	if (stat(mbxpath, &buf) == -1 || buf.st_size == 0) {
 | 
						|
	    /* Mailbox is empty or nonexistent */
 | 
						|
	    if (!NONOTHING) printf("No unread mail\n");
 | 
						|
        } else {
 | 
						|
	    if (buf.st_mtime == buf.st_atime) {
 | 
						|
		/* There is something in the mailbox, but we can't really
 | 
						|
		 *   be sure whether it is mail held there by the user
 | 
						|
		 *   or a (single) new message that was placed in a newly
 | 
						|
		 *   recreated mailbox, so we punt and call it "unread mail."
 | 
						|
		 */
 | 
						|
		printf("Unread mail since ");
 | 
						|
	        printf(ctime(&buf.st_mtime));
 | 
						|
	    } else {
 | 
						|
		/* New mail has definitely arrived since the last time
 | 
						|
		 *   mail was read.  mtime is the time the most recent
 | 
						|
		 *   message arrived; atime is either the time the oldest
 | 
						|
		 *   unread message arrived, or the last time the mail
 | 
						|
		 *   was read.
 | 
						|
		 */
 | 
						|
		printf("New mail received ");
 | 
						|
		timestr = ctime(&buf.st_mtime);	/* time last modified */
 | 
						|
		timestr[24] = '\0';		/* suppress newline (ugh) */
 | 
						|
		printf(timestr);
 | 
						|
		printf(";\n  unread since ");
 | 
						|
	        printf(ctime(&buf.st_atime));	/* time last accessed */
 | 
						|
	    }
 | 
						|
	}
 | 
						|
	
 | 
						|
	free(mbxpath);
 | 
						|
}
 | 
						|
 | 
						|
/*
 | 
						|
 * return true iff we've already printed project/plan for this uid;
 | 
						|
 * if not, enter this uid into table (so this function has a side-effect.)
 | 
						|
 */
 | 
						|
#define	PPMAX	200		/* assume no more than 200 logged-in users */
 | 
						|
int	PlanPrinted[PPMAX+1];
 | 
						|
int	PPIndex = 0;		/* index of next unused table entry */
 | 
						|
 | 
						|
static int
 | 
						|
AlreadyPrinted(uid)
 | 
						|
int uid;
 | 
						|
{
 | 
						|
	int i = 0;
 | 
						|
	
 | 
						|
	while (i++ < PPIndex) {
 | 
						|
	    if (PlanPrinted[i] == uid)
 | 
						|
		return(1);
 | 
						|
	}
 | 
						|
	if (i < PPMAX) {
 | 
						|
	    PlanPrinted[i] = uid;
 | 
						|
	    PPIndex++;
 | 
						|
	}
 | 
						|
	return(0);
 | 
						|
}
 |