blob: b4955258eb1996dc804b0722f34ebffe7def233e [file] [log] [blame]
Glenn L McGrath18b76e62002-09-16 09:10:04 +00001/* 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-Fischer73561cc2006-08-28 23:31:54 +00006 * Copyrigjt (C) Mar 16, 2003 Manuel Novoa III (mjn3@codepoet.org)
Glenn L McGrath18b76e62002-09-16 09:10:04 +00007 *
Rob Landley399d2b52006-05-25 23:02:40 +00008 * Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
Glenn L McGrath18b76e62002-09-16 09:10:04 +00009 */
10
Manuel Novoa III cad53642003-03-19 09:13:01 +000011/* BB_AUDIT SUSv3 N/A */
12/* BB_AUDIT GNU defects -- only option -n is supported. */
13
Denis Vlasenkob6adbf12007-05-26 19:00:18 +000014#include "libbb.h"
Glenn L McGrath18b76e62002-09-16 09:10:04 +000015
Denis Vlasenko8d73c352006-10-20 23:48:30 +000016// procps 2.0.18:
17// watch [-d] [-n seconds]
18// [--differences[=cumulative]] [--interval=seconds] command
19//
20// procps-3.2.3:
21// watch [-dt] [-n seconds]
22// [--differences[=cumulative]] [--interval=seconds] [--no-title] command
23//
24// (procps 3.x and procps 2.x are forks, not newer/older versions of the same)
Bernhard Reutner-Fischer73561cc2006-08-28 23:31:54 +000025
Denis Vlasenko9b49a5e2007-10-11 10:05:36 +000026int watch_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
Rob Landleydfba7412006-03-06 20:47:33 +000027int watch_main(int argc, char **argv)
Glenn L McGrath18b76e62002-09-16 09:10:04 +000028{
Denis Vlasenko8d73c352006-10-20 23:48:30 +000029 unsigned opt;
Manuel Novoa III cad53642003-03-19 09:13:01 +000030 unsigned period = 2;
Denis Vlasenko28b29912008-02-24 14:33:17 +000031 int width, new_width;
32 char *header;
Denis Vlasenko8d73c352006-10-20 23:48:30 +000033 char *cmd;
Glenn L McGrath18b76e62002-09-16 09:10:04 +000034
Denis Vlasenko28b29912008-02-24 14:33:17 +000035 opt_complementary = "-1:n+"; // at least one param; -n NUM
36 // "+": stop at first non-option (procps 3.x only)
37 opt = getopt32(argv, "+dtn:", &period);
Denis Vlasenko8d73c352006-10-20 23:48:30 +000038 argv += optind;
Rob Landley399d2b52006-05-25 23:02:40 +000039
Denis Vlasenko28b29912008-02-24 14:33:17 +000040 // watch from both procps 2.x and 3.x does concatenation. Example:
41 // watch ls -l "a /tmp" "2>&1" -- ls won't see "a /tmp" as one param
42 cmd = *argv;
43 while (*++argv)
44 cmd = xasprintf("%s %s", cmd, *argv); // leaks cmd
Glenn L McGrath18b76e62002-09-16 09:10:04 +000045
Denis Vlasenko28b29912008-02-24 14:33:17 +000046 width = -1; // make sure first time new_width != width
47 header = NULL;
Glenn L McGrath18b76e62002-09-16 09:10:04 +000048 while (1) {
Denis Vlasenko8d73c352006-10-20 23:48:30 +000049 printf("\033[H\033[J");
50 if (!(opt & 0x2)) { // no -t
Denis Vlasenko28b29912008-02-24 14:33:17 +000051 const int time_len = sizeof("1234-67-90 23:56:89");
Denis Vlasenko8d73c352006-10-20 23:48:30 +000052 time_t t;
Glenn L McGrath18b76e62002-09-16 09:10:04 +000053
Denis Vlasenko28b29912008-02-24 14:33:17 +000054 get_terminal_width_height(STDIN_FILENO, &new_width, NULL);
55 if (new_width != width) {
56 width = new_width;
57 free(header);
58 header = xasprintf("Every %us: %-*s", period, width, cmd);
59 }
Denis Vlasenko8d73c352006-10-20 23:48:30 +000060 time(&t);
Denis Vlasenko28b29912008-02-24 14:33:17 +000061 if (time_len < width)
62 strftime(header + width - time_len, time_len,
63 "%Y-%m-%d %H:%M:%S", localtime(&t));
64
Denis Vlasenko8d73c352006-10-20 23:48:30 +000065 puts(header);
66 }
67 fflush(stdout);
68 // TODO: 'real' watch pipes cmd's output to itself
69 // and does not allow it to overflow the screen
70 // (taking into account linewrap!)
71 system(cmd);
Rob Landley399d2b52006-05-25 23:02:40 +000072 sleep(period);
Glenn L McGrath18b76e62002-09-16 09:10:04 +000073 }
Denis Vlasenko703aa132006-10-23 22:43:02 +000074 return 0; // gcc thinks we can reach this :)
Glenn L McGrath18b76e62002-09-16 09:10:04 +000075}