who.c 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173
  1. /* vi: set sw=4 ts=4: */
  2. /*----------------------------------------------------------------------
  3. * Mini who is used to display user name, login time,
  4. * idle time and host name.
  5. *
  6. * Author: Da Chen <dchen@ayrnetworks.com>
  7. *
  8. * This is a free document; you can redistribute it and/or
  9. * modify it under the terms of the GNU General Public License
  10. * as published by the Free Software Foundation:
  11. * http://www.gnu.org/copyleft/gpl.html
  12. *
  13. * Copyright (c) 2002 AYR Networks, Inc.
  14. *
  15. * Licensed under GPLv2 or later, see file LICENSE in this source tree.
  16. *
  17. *----------------------------------------------------------------------
  18. */
  19. //config:config WHO
  20. //config: bool "who"
  21. //config: default y
  22. //config: depends on FEATURE_UTMP
  23. //config: help
  24. //config: who is used to show who is logged on.
  25. //config:
  26. // procps-ng has this variation of "who":
  27. //config:config W
  28. //config: bool "w"
  29. //config: default y
  30. //config: depends on FEATURE_UTMP
  31. //config: help
  32. //config: w is used to show who is logged on.
  33. //config:
  34. //config:config USERS
  35. //config: bool "users"
  36. //config: default y
  37. //config: depends on FEATURE_UTMP
  38. //config: help
  39. //config: Print users currently logged on.
  40. // APPLET_ODDNAME:name main location suid_type help
  41. //applet:IF_USERS(APPLET_ODDNAME(users, who, BB_DIR_USR_BIN, BB_SUID_DROP, users))
  42. //applet:IF_W( APPLET_ODDNAME(w, who, BB_DIR_USR_BIN, BB_SUID_DROP, w))
  43. //applet:IF_WHO( APPLET( who, BB_DIR_USR_BIN, BB_SUID_DROP))
  44. //kbuild:lib-$(CONFIG_USERS) += who.o
  45. //kbuild:lib-$(CONFIG_W) += who.o
  46. //kbuild:lib-$(CONFIG_WHO) += who.o
  47. /* BB_AUDIT SUSv3 _NOT_ compliant -- missing options -b, -d, -l, -m, -p, -q, -r, -s, -t, -T, -u; Missing argument 'file'. */
  48. //usage:#define users_trivial_usage
  49. //usage: ""
  50. //usage:#define users_full_usage "\n\n"
  51. //usage: "Print the users currently logged on"
  52. //usage:#define w_trivial_usage
  53. //usage: ""
  54. //usage:#define w_full_usage "\n\n"
  55. //usage: "Show who is logged on"
  56. //
  57. // procps-ng 3.3.10:
  58. // "\n -h, --no-header"
  59. // "\n -u, --no-current"
  60. // Ignores the username while figuring out the current process
  61. // and cpu times. To demonstrate this, do a "su" and do a "w" and a "w -u".
  62. // "\n -s, --short"
  63. // Short format. Don't print the login time, JCPU or PCPU times.
  64. // "\n -f, --from"
  65. // Toggle printing the from (remote hostname) field.
  66. // The default is for the from field to not be printed
  67. // "\n -i, --ip-addr"
  68. // Display IP address instead of hostname for from field.
  69. // "\n -o, --old-style"
  70. // Old style output. Prints blank space for idle times less than one minute.
  71. // Example output:
  72. // 17:28:00 up 4 days, 22:41, 4 users, load average: 0.84, 0.97, 0.90
  73. // USER TTY LOGIN@ IDLE JCPU PCPU WHAT
  74. // root tty1 Thu18 4days 4:33m 0.07s /bin/sh /etc/xdg/xfce4/xinitrc -- vt
  75. // root pts/1 Mon13 3:24m 1:01 0.01s w
  76. //usage:#define who_trivial_usage
  77. //usage: "[-a]"
  78. //usage:#define who_full_usage "\n\n"
  79. //usage: "Show who is logged on\n"
  80. //usage: "\n -a Show all"
  81. //usage: "\n -H Print column headers"
  82. #include "libbb.h"
  83. static void idle_string(char *str6, time_t t)
  84. {
  85. t = time(NULL) - t;
  86. /*if (t < 60) {
  87. str6[0] = '.';
  88. str6[1] = '\0';
  89. return;
  90. }*/
  91. if (t >= 0 && t < (24 * 60 * 60)) {
  92. sprintf(str6, "%02d:%02d",
  93. (int) (t / (60 * 60)),
  94. (int) ((t % (60 * 60)) / 60));
  95. return;
  96. }
  97. strcpy(str6, "old");
  98. }
  99. int who_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
  100. int who_main(int argc UNUSED_PARAM, char **argv)
  101. {
  102. #define CNT_APPLET (ENABLE_USERS + ENABLE_W + ENABLE_WHO)
  103. int do_users = (ENABLE_USERS && (CNT_APPLET == 1 || applet_name[0] == 'u'));
  104. int do_w = (ENABLE_W && (CNT_APPLET == 1 || applet_name[1] == '\0'));
  105. int do_who = (ENABLE_WHO && (CNT_APPLET == 1 || applet_name[1] == 'h'));
  106. struct utmpx *ut;
  107. unsigned opt;
  108. const char *fmt = "%s";
  109. opt_complementary = "=0";
  110. opt = getopt32(argv, do_who ? "aH" : "");
  111. if ((opt & 2) || do_w) /* -H or we are w */
  112. puts("USER\t\tTTY\t\tIDLE\tTIME\t\t HOST");
  113. setutxent();
  114. while ((ut = getutxent()) != NULL) {
  115. if (ut->ut_user[0]
  116. && ((opt & 1) || ut->ut_type == USER_PROCESS)
  117. ) {
  118. if (!do_users) {
  119. char str6[6];
  120. char name[sizeof("/dev/") + sizeof(ut->ut_line) + 1];
  121. struct stat st;
  122. time_t seconds;
  123. str6[0] = '?';
  124. str6[1] = '\0';
  125. strcpy(name, "/dev/");
  126. safe_strncpy(ut->ut_line[0] == '/' ? name : name + sizeof("/dev/")-1,
  127. ut->ut_line,
  128. sizeof(ut->ut_line)+1
  129. );
  130. if (stat(name, &st) == 0)
  131. idle_string(str6, st.st_atime);
  132. /* manpages say ut_tv.tv_sec *is* time_t,
  133. * but some systems have it wrong */
  134. seconds = ut->ut_tv.tv_sec;
  135. /* How wide time field can be?
  136. * "Nov 10 19:33:20": 15 chars
  137. * "2010-11-10 19:33": 16 chars
  138. */
  139. printf("%-15.*s %-15.*s %-7s %-16.16s %.*s\n",
  140. (int)sizeof(ut->ut_user), ut->ut_user,
  141. (int)sizeof(ut->ut_line), ut->ut_line,
  142. str6,
  143. // TODO: with LANG=en_US.UTF-8, who from coreutils 8.25 shows
  144. // TIME col as "2017-04-06 18:47" (the default format is "Apr 6 18:47").
  145. // The former format looks saner to me. Switch to it unconditionally?
  146. ctime(&seconds) + 4,
  147. (int)sizeof(ut->ut_host), ut->ut_host
  148. );
  149. } else {
  150. printf(fmt, ut->ut_user);
  151. fmt = " %s";
  152. }
  153. }
  154. }
  155. if (do_users)
  156. bb_putchar('\n');
  157. if (ENABLE_FEATURE_CLEAN_UP)
  158. endutxent();
  159. return EXIT_SUCCESS;
  160. }