* Copyright (c) 1989 The Regents of the University of California.
* This code is derived from software contributed to Berkeley by
* Tony Nardo of the Johns Hopkins University/Applied Physics Lab.
* Redistribution and use in source and binary forms are permitted
* provided that the above copyright notice and this paragraph are
* duplicated in all such forms and that any documentation,
* advertising materials, and other materials related to such
* distribution and use acknowledge that the software was developed
* by the University of California, Berkeley. The name of the
* University may not be used to endorse or promote products derived
* from this software without specific prior written permission.
* THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR
* IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
"@(#) Copyright (c) 1989 The Regents of the University of California.\n\
static char sccsid
[] = "@(#)finger.c 5.13 (Berkeley) %G%";
* Finger prints out information about users. It is not portable since
* certain fields (e.g. the full user name, office, and phone numbers) are
* extracted from the gecos field of the passwd file which other UNIXes
* may not have or may use for other things.
* There are currently two output formats; the short format is one line
* per user and displays login name, tty, login time, real name, idle time,
* and office location/phone number. The long format gives the same
* information (in a more legible format) as well as home directory, shell,
* mail info, and .plan/.project files.
int entries
, lflag
, sflag
, mflag
, pplan
;
while ((ch
= getopt(argc
, argv
, "lmps")) != EOF
)
lflag
= 1; /* long format */
mflag
= 1; /* force exact match of names */
pplan
= 1; /* don't show .plan/.project */
sflag
= 1; /* short format */
"usage: finger [-lmps] [login ...]\n");
* Assign explicit "small" format if no names given and -l
* not selected. Force the -s BEFORE we get names so proper
* screening will be done.
sflag
= 1; /* if -l not explicit, force -s */
(void)printf("No one logged on.\n");
* Assign explicit "large" format if names given and -s not
* explicitly stated. Force the -l AFTER we get names so any
* remote finger attempts specified won't be mishandled.
lflag
= 1; /* if -s not explicit, force -l */
char name
[UT_NAMESIZE
+ 1], *strdup(), *malloc();
if ((fd
= open(_PATH_UTMP
, O_RDONLY
, 0)) < 0) {
(void)fprintf(stderr
, "finger: can't read %s.\n", _PATH_UTMP
);
name
[UT_NAMESIZE
] = NULL
;
while(read(fd
, (char *)&user
, sizeof(user
)) == sizeof(user
)) {
bcopy(user
.ut_name
, name
, UT_NAMESIZE
);
if (!(pw
= getpwnam(name
)))
if (!(pn
= (PERSON
*)malloc((u_int
)sizeof(PERSON
)))) {
(void)fprintf(stderr
, "finger: out of space.\n");
find_idle_and_ttywrite(pn
);
#define ARGIGNORE (char *)0x01
register PERSON
*nethead
, *p
, *pn
;
register struct passwd
*pw
;
register char **a1
, **a2
;
char **sargv
, **arglist
, *malloc(), *rindex(), *strcpy();
/* suppress duplicates while it's still easy */
for (a1
= argv
; *a1
; ++a1
)
for (a2
= a1
+ 1; *a2
; ++a2
)
if (!strcasecmp(*a1
, *a2
)) {
/* pull out all network requests */
for (sargv
= argv
, dolocal
= 0, nethead
= NULL
; *argv
; ++argv
) {
if (!index(*argv
, '@')) {
if (!(pn
= (PERSON
*)malloc((u_int
)sizeof(PERSON
)))) {
(void)fprintf(stderr
, "finger: out of space.\n");
* traverse the list of possible login names and check the login name
* and real name against the name specified by the user. Possible
* speedup would be to use getpwnam(3) if mflag set -- maybe not
* worthwhile, given that the default is the mflag off.
nelem
= argv
- sargv
+ 1;
(char **)malloc((u_int
)(nelem
* sizeof(char *))))) {
(void)fprintf(stderr
, "finger: out of space.\n");
bcopy((char *)sargv
, (char *)arglist
, nelem
* sizeof(char *));
while (pw
= getpwent()) {
for (argv
= sargv
; *argv
; ++argv
) {
if (strcasecmp(pw
->pw_name
, *argv
) &&
(mflag
|| !match(pw
, *argv
)))
if (!(pn
= (PERSON
*)malloc((u_int
)sizeof(PERSON
)))) {
"finger: out of space.\n");
arglist
[argv
- sargv
] = ARGIGNORE
;
/* don't break, may be listed multiple times */
for (; *arglist
; ++arglist
)
if (*arglist
!= ARGIGNORE
)
"finger: %s: no such user.\n", *arglist
);
/* handle network requests */
net
: for (pn
= nethead
; pn
; pn
= pn
->next
) {
* Scan thru the list of users currently logged in, saving
* appropriate data whenever a match occurs.
if ((fd
= open(_PATH_UTMP
, O_RDONLY
, 0)) < 0) {
(void)fprintf( stderr
, "finger: can't read %s.\n", _PATH_UTMP
);
while (read(fd
, (char *)&user
, sizeof(user
)) == sizeof(user
)) {
for (pn
= head
; pn
; pn
= pn
->next
) {
if (strncasecmp(pn
->name
, user
.ut_name
, UT_NAMESIZE
))
if (pn
->info
== LOGGEDIN
) {
(PERSON
*)malloc((u_int
)sizeof(PERSON
)))) {
"finger: out of space.\n");
(void)strcpy(p
->tty
, pn
->tty
);
/* link in so finds `real' entry first! */
find_idle_and_ttywrite(pn
);
/* don't break, may be listed multiple times... */
for (pn
= head
; pn
; pn
= pn
->next
)