1 /*- 2 * Copyright (c) 1993, John Brezak 3 * 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 #include <sys/types.h> 35 #include <sys/param.h> 36 #include <sys/socket.h> 37 #include <netdb.h> 38 #include <stdio.h> 39 #include <strings.h> 40 #include <rpc/rpc.h> 41 #include <arpa/inet.h> 42 #include <rpcsvc/rnusers.h> 43 44 #define MAX_INT 0x7fffffff 45 #define HOST_WIDTH 20 46 #define LINE_WIDTH 15 47 char *argv0; 48 49 int longopt; 50 int allopt; 51 52 struct host_list { 53 struct host_list *next; 54 struct in_addr addr; 55 } *hosts; 56 57 int search_host(struct in_addr addr) 58 { 59 struct host_list *hp; 60 61 if (!hosts) 62 return(0); 63 64 for (hp = hosts; hp != NULL; hp = hp->next) { 65 if (hp->addr.s_addr == addr.s_addr) 66 return(1); 67 } 68 return(0); 69 } 70 71 void remember_host(struct in_addr addr) 72 { 73 struct host_list *hp; 74 75 if (!(hp = (struct host_list *)malloc(sizeof(struct host_list)))) { 76 fprintf(stderr, "%s: no memory.\n", argv0); 77 exit(1); 78 } 79 hp->addr.s_addr = addr.s_addr; 80 hp->next = hosts; 81 hosts = hp; 82 } 83 84 rusers_reply(char *replyp, struct sockaddr_in *raddrp) 85 { 86 int x, idle; 87 char date[32], idle_time[64], remote[64]; 88 struct hostent *hp; 89 utmpidlearr *up = (utmpidlearr *)replyp; 90 char *host; 91 int days, hours, minutes, seconds; 92 93 if (search_host(raddrp->sin_addr)) 94 return(0); 95 96 if (!allopt && !up->utmpidlearr_len) 97 return(0); 98 99 hp = gethostbyaddr((char *)&raddrp->sin_addr.s_addr, 100 sizeof(struct in_addr), AF_INET); 101 if (hp) 102 host = hp->h_name; 103 else 104 host = inet_ntoa(raddrp->sin_addr); 105 106 if (!longopt) 107 printf("%-*s ", HOST_WIDTH, host); 108 109 for (x = 0; x < up->utmpidlearr_len; x++) { 110 strncpy(date, 111 &(ctime((time_t *)&(up->utmpidlearr_val[x].ui_utmp.ut_time))[4]), 112 sizeof(date)-1); 113 114 idle = up->utmpidlearr_val[x].ui_idle; 115 sprintf(idle_time, " :%02d", idle); 116 if (idle == MAX_INT) 117 strcpy(idle_time, "??"); 118 else if (idle == 0) 119 strcpy(idle_time, ""); 120 else { 121 seconds = idle; 122 days = seconds/(60*60*24); 123 seconds %= (60*60*24); 124 hours = seconds/(60*60); 125 seconds %= (60*60); 126 minutes = seconds/60; 127 seconds %= 60; 128 if (idle > 60) 129 sprintf(idle_time, "%d:%02d", 130 minutes, seconds); 131 if (idle >= (60*60)) 132 sprintf(idle_time, "%d:%02d:%02d", 133 hours, minutes, seconds); 134 if (idle >= (24*60*60)) 135 sprintf(idle_time, "%d days, %d:%02d:%02d", 136 days, hours, minutes, seconds); 137 } 138 139 strncpy(remote, up->utmpidlearr_val[x].ui_utmp.ut_host, sizeof(remote)-1); 140 if (strlen(remote) != 0) 141 sprintf(remote, "(%.16s)", up->utmpidlearr_val[x].ui_utmp.ut_host); 142 143 if (longopt) 144 printf("%-8.8s %*s:%-*.*s %-12.12s %6s %.18s\n", 145 up->utmpidlearr_val[x].ui_utmp.ut_name, 146 HOST_WIDTH, host, 147 LINE_WIDTH, LINE_WIDTH, up->utmpidlearr_val[x].ui_utmp.ut_line, 148 date, 149 idle_time, 150 remote 151 ); 152 else 153 printf("%s ", 154 up->utmpidlearr_val[x].ui_utmp.ut_name); 155 } 156 if (!longopt) 157 putchar('\n'); 158 159 remember_host(raddrp->sin_addr); 160 return(0); 161 } 162 163 onehost(char *host) 164 { 165 utmpidlearr up; 166 CLIENT *rusers_clnt; 167 struct sockaddr_in addr; 168 struct hostent *hp; 169 170 hp = gethostbyname(host); 171 if (hp == NULL) { 172 fprintf(stderr, "%s: unknown host \"%s\"\n", 173 argv0, host); 174 exit(1); 175 } 176 177 rusers_clnt = clnt_create(host, RUSERSPROG, RUSERSVERS_IDLE, "udp"); 178 if (rusers_clnt == NULL) { 179 clnt_pcreateerror(argv0); 180 exit(1); 181 } 182 183 bzero((char *)&up, sizeof(up)); 184 if (clnt_call(rusers_clnt, RUSERSPROC_NAMES, xdr_void, NULL, xdr_utmpidlearr, &up, NULL) != RPC_SUCCESS) { 185 clnt_perror(rusers_clnt, argv0); 186 exit(1); 187 } 188 addr.sin_addr.s_addr = *(int *)hp->h_addr; 189 rusers_reply((char *)&up, &addr); 190 } 191 192 allhosts() 193 { 194 utmpidlearr up; 195 enum clnt_stat clnt_stat; 196 197 bzero((char *)&up, sizeof(up)); 198 clnt_stat = clnt_broadcast(RUSERSPROG, RUSERSVERS_IDLE, RUSERSPROC_NAMES, 199 xdr_void, NULL, 200 xdr_utmpidlearr, &up, rusers_reply); 201 if (clnt_stat != RPC_SUCCESS && clnt_stat != RPC_TIMEDOUT) { 202 fprintf(stderr, "%s: %s\n", argv0, clnt_sperrno(clnt_stat)); 203 exit(1); 204 } 205 } 206 207 usage() 208 { 209 fprintf(stderr, "Usage: %s [-la] [hosts ...]\n", argv0); 210 exit(1); 211 } 212 213 main(int argc, char *argv[]) 214 { 215 int ch; 216 extern int optind; 217 218 if (!(argv0 = rindex(argv[0], '/'))) 219 argv0 = argv[0]; 220 else 221 argv0++; 222 223 224 while ((ch = getopt(argc, argv, "al")) != -1) 225 switch (ch) { 226 case 'a': 227 allopt++; 228 break; 229 case 'l': 230 longopt++; 231 break; 232 default: 233 usage(); 234 /*NOTREACHED*/ 235 } 236 237 setlinebuf(stdout); 238 if (argc == optind) 239 allhosts(); 240 else { 241 for (; optind < argc; optind++) 242 (void) onehost(argv[optind]); 243 } 244 exit(0); 245 } 246