Glenn L McGrath | 18b76e6 | 2002-09-16 09:10:04 +0000 | [diff] [blame] | 1 | /* vi: set sw=4 ts=4: */ |
| 2 | /* |
| 3 | * Mini watch implementation for busybox |
| 4 | * |
| 5 | * Copyright (C) 2001 by Michael Habermann <mhabermann@gmx.de> |
Bernhard Reutner-Fischer | 73561cc | 2006-08-28 23:31:54 +0000 | [diff] [blame] | 6 | * Copyrigjt (C) Mar 16, 2003 Manuel Novoa III (mjn3@codepoet.org) |
Glenn L McGrath | 18b76e6 | 2002-09-16 09:10:04 +0000 | [diff] [blame] | 7 | * |
Denys Vlasenko | 0ef64bd | 2010-08-16 20:14:46 +0200 | [diff] [blame] | 8 | * Licensed under GPLv2 or later, see file LICENSE in this source tree. |
Glenn L McGrath | 18b76e6 | 2002-09-16 09:10:04 +0000 | [diff] [blame] | 9 | */ |
Denys Vlasenko | f8f81ed | 2016-11-23 06:23:44 +0100 | [diff] [blame] | 10 | //config:config WATCH |
Denys Vlasenko | b097a84 | 2018-12-28 03:20:17 +0100 | [diff] [blame] | 11 | //config: bool "watch (4.4 kb)" |
Denys Vlasenko | f8f81ed | 2016-11-23 06:23:44 +0100 | [diff] [blame] | 12 | //config: default y |
| 13 | //config: help |
Denys Vlasenko | 72089cf | 2017-07-21 09:50:55 +0200 | [diff] [blame] | 14 | //config: watch is used to execute a program periodically, showing |
| 15 | //config: output to the screen. |
Denys Vlasenko | f8f81ed | 2016-11-23 06:23:44 +0100 | [diff] [blame] | 16 | |
| 17 | //applet:IF_WATCH(APPLET(watch, BB_DIR_BIN, BB_SUID_DROP)) |
| 18 | |
| 19 | //kbuild:lib-$(CONFIG_WATCH) += watch.o |
| 20 | |
Pere Orga | 5bc8c00 | 2011-04-11 03:29:49 +0200 | [diff] [blame] | 21 | //usage:#define watch_trivial_usage |
| 22 | //usage: "[-n SEC] [-t] PROG ARGS" |
| 23 | //usage:#define watch_full_usage "\n\n" |
| 24 | //usage: "Run PROG periodically\n" |
Denys Vlasenko | 0ddc742 | 2019-03-12 18:56:51 +0100 | [diff] [blame^] | 25 | //usage: "\n -n SEC Loop period (default 2)" |
Pere Orga | 5bc8c00 | 2011-04-11 03:29:49 +0200 | [diff] [blame] | 26 | //usage: "\n -t Don't print header" |
| 27 | //usage: |
| 28 | //usage:#define watch_example_usage |
| 29 | //usage: "$ watch date\n" |
| 30 | //usage: "Mon Dec 17 10:31:40 GMT 2000\n" |
| 31 | //usage: "Mon Dec 17 10:31:42 GMT 2000\n" |
| 32 | //usage: "Mon Dec 17 10:31:44 GMT 2000" |
| 33 | |
Denys Vlasenko | 0c4dbd4 | 2017-09-18 16:28:43 +0200 | [diff] [blame] | 34 | /* BB_AUDIT SUSv3 N/A */ |
| 35 | /* BB_AUDIT GNU defects -- only option -n is supported. */ |
| 36 | |
Denis Vlasenko | b6adbf1 | 2007-05-26 19:00:18 +0000 | [diff] [blame] | 37 | #include "libbb.h" |
Glenn L McGrath | 18b76e6 | 2002-09-16 09:10:04 +0000 | [diff] [blame] | 38 | |
Denys Vlasenko | 8187e01 | 2017-09-13 22:48:30 +0200 | [diff] [blame] | 39 | #define ESC "\033" |
| 40 | |
Denis Vlasenko | 8d73c35 | 2006-10-20 23:48:30 +0000 | [diff] [blame] | 41 | // procps 2.0.18: |
| 42 | // watch [-d] [-n seconds] |
| 43 | // [--differences[=cumulative]] [--interval=seconds] command |
| 44 | // |
| 45 | // procps-3.2.3: |
| 46 | // watch [-dt] [-n seconds] |
| 47 | // [--differences[=cumulative]] [--interval=seconds] [--no-title] command |
| 48 | // |
| 49 | // (procps 3.x and procps 2.x are forks, not newer/older versions of the same) |
Bernhard Reutner-Fischer | 73561cc | 2006-08-28 23:31:54 +0000 | [diff] [blame] | 50 | |
Denis Vlasenko | 9b49a5e | 2007-10-11 10:05:36 +0000 | [diff] [blame] | 51 | int watch_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE; |
Denis Vlasenko | a60f84e | 2008-07-05 09:18:54 +0000 | [diff] [blame] | 52 | int watch_main(int argc UNUSED_PARAM, char **argv) |
Glenn L McGrath | 18b76e6 | 2002-09-16 09:10:04 +0000 | [diff] [blame] | 53 | { |
Denys Vlasenko | 0ddc742 | 2019-03-12 18:56:51 +0100 | [diff] [blame^] | 54 | duration_t period; |
| 55 | char *period_str = (char*) "2"; |
Denis Vlasenko | 8d73c35 | 2006-10-20 23:48:30 +0000 | [diff] [blame] | 56 | unsigned opt; |
Denis Vlasenko | 5599502 | 2008-05-18 22:28:26 +0000 | [diff] [blame] | 57 | unsigned width, new_width; |
Denis Vlasenko | 28b2991 | 2008-02-24 14:33:17 +0000 | [diff] [blame] | 58 | char *header; |
Denis Vlasenko | 8d73c35 | 2006-10-20 23:48:30 +0000 | [diff] [blame] | 59 | char *cmd; |
Glenn L McGrath | 18b76e6 | 2002-09-16 09:10:04 +0000 | [diff] [blame] | 60 | |
Denys Vlasenko | 302af9e | 2010-01-19 02:26:38 +0100 | [diff] [blame] | 61 | #if 0 // maybe ENABLE_DESKTOP? |
| 62 | // procps3 compat - "echo TEST | watch cat" doesn't show TEST: |
| 63 | close(STDIN_FILENO); |
| 64 | xopen("/dev/null", O_RDONLY); |
| 65 | #endif |
| 66 | |
Denys Vlasenko | 22542ec | 2017-08-08 21:55:02 +0200 | [diff] [blame] | 67 | // "+": stop at first non-option (procps 3.x only); -n NUM |
| 68 | // at least one param |
Denys Vlasenko | 0ddc742 | 2019-03-12 18:56:51 +0100 | [diff] [blame^] | 69 | opt = getopt32(argv, "^+" "dtn:" "\0" "-1", &period_str); |
Denis Vlasenko | 8d73c35 | 2006-10-20 23:48:30 +0000 | [diff] [blame] | 70 | argv += optind; |
Rob Landley | 399d2b5 | 2006-05-25 23:02:40 +0000 | [diff] [blame] | 71 | |
Denis Vlasenko | 28b2991 | 2008-02-24 14:33:17 +0000 | [diff] [blame] | 72 | // watch from both procps 2.x and 3.x does concatenation. Example: |
Denys Vlasenko | 302af9e | 2010-01-19 02:26:38 +0100 | [diff] [blame] | 73 | // watch ls -l "a /tmp" "2>&1" - ls won't see "a /tmp" as one param |
Denis Vlasenko | 28b2991 | 2008-02-24 14:33:17 +0000 | [diff] [blame] | 74 | cmd = *argv; |
| 75 | while (*++argv) |
| 76 | cmd = xasprintf("%s %s", cmd, *argv); // leaks cmd |
Glenn L McGrath | 18b76e6 | 2002-09-16 09:10:04 +0000 | [diff] [blame] | 77 | |
Denys Vlasenko | 0ddc742 | 2019-03-12 18:56:51 +0100 | [diff] [blame^] | 78 | period = parse_duration_str(period_str); |
Denis Vlasenko | 5599502 | 2008-05-18 22:28:26 +0000 | [diff] [blame] | 79 | width = (unsigned)-1; // make sure first time new_width != width |
Denis Vlasenko | 28b2991 | 2008-02-24 14:33:17 +0000 | [diff] [blame] | 80 | header = NULL; |
Glenn L McGrath | 18b76e6 | 2002-09-16 09:10:04 +0000 | [diff] [blame] | 81 | while (1) { |
Denys Vlasenko | d9a3e89 | 2010-05-16 23:42:13 +0200 | [diff] [blame] | 82 | /* home; clear to the end of screen */ |
Denys Vlasenko | 8187e01 | 2017-09-13 22:48:30 +0200 | [diff] [blame] | 83 | printf(ESC"[H" ESC"[J"); |
Denis Vlasenko | 8d73c35 | 2006-10-20 23:48:30 +0000 | [diff] [blame] | 84 | if (!(opt & 0x2)) { // no -t |
Denis Vlasenko | 5599502 | 2008-05-18 22:28:26 +0000 | [diff] [blame] | 85 | const unsigned time_len = sizeof("1234-67-90 23:56:89"); |
Glenn L McGrath | 18b76e6 | 2002-09-16 09:10:04 +0000 | [diff] [blame] | 86 | |
Denys Vlasenko | 302af9e | 2010-01-19 02:26:38 +0100 | [diff] [blame] | 87 | // STDERR_FILENO is procps3 compat: |
| 88 | // "watch ls 2>/dev/null" does not detect tty size |
Denys Vlasenko | 641caae | 2015-10-23 01:44:22 +0200 | [diff] [blame] | 89 | new_width = get_terminal_width(STDERR_FILENO); |
Denis Vlasenko | 28b2991 | 2008-02-24 14:33:17 +0000 | [diff] [blame] | 90 | if (new_width != width) { |
| 91 | width = new_width; |
| 92 | free(header); |
Denys Vlasenko | 0ddc742 | 2019-03-12 18:56:51 +0100 | [diff] [blame^] | 93 | header = xasprintf("Every" |
| 94 | " %"IF_FLOAT_DURATION(".1")DURATION_FMT"s:" |
| 95 | " %-*s", |
| 96 | period, |
| 97 | (int)width, cmd |
| 98 | ); |
Denis Vlasenko | 28b2991 | 2008-02-24 14:33:17 +0000 | [diff] [blame] | 99 | } |
Denys Vlasenko | 8f2cb7a | 2013-03-29 12:30:33 +0100 | [diff] [blame] | 100 | if (time_len < width) { |
| 101 | strftime_YYYYMMDDHHMMSS( |
| 102 | header + width - time_len, |
| 103 | time_len, |
| 104 | /*time_t*:*/ NULL |
| 105 | ); |
| 106 | } |
Denis Vlasenko | 28b2991 | 2008-02-24 14:33:17 +0000 | [diff] [blame] | 107 | |
Denys Vlasenko | 302af9e | 2010-01-19 02:26:38 +0100 | [diff] [blame] | 108 | // compat: empty line between header and cmd output |
| 109 | printf("%s\n\n", header); |
Denis Vlasenko | 8d73c35 | 2006-10-20 23:48:30 +0000 | [diff] [blame] | 110 | } |
Denys Vlasenko | 8131eea | 2009-11-02 14:19:51 +0100 | [diff] [blame] | 111 | fflush_all(); |
Denis Vlasenko | 8d73c35 | 2006-10-20 23:48:30 +0000 | [diff] [blame] | 112 | // TODO: 'real' watch pipes cmd's output to itself |
| 113 | // and does not allow it to overflow the screen |
| 114 | // (taking into account linewrap!) |
| 115 | system(cmd); |
Denys Vlasenko | 0ddc742 | 2019-03-12 18:56:51 +0100 | [diff] [blame^] | 116 | sleep_for_duration(period); |
Glenn L McGrath | 18b76e6 | 2002-09-16 09:10:04 +0000 | [diff] [blame] | 117 | } |
Denis Vlasenko | 703aa13 | 2006-10-23 22:43:02 +0000 | [diff] [blame] | 118 | return 0; // gcc thinks we can reach this :) |
Glenn L McGrath | 18b76e6 | 2002-09-16 09:10:04 +0000 | [diff] [blame] | 119 | } |