Prevent Watch Command from Breaking Colors in Printf

colorsprintf

I'm altering scripts at work that monitor log files to single out certain items and colorize them. The final output is a list of 6-digit numbers in several columns. I've been able to add a ~ to the start of the number, and it doesn't break anything in the final output, but it's rather ugly and hard to read. However, when I try to colorize the number, either in preceding bash scripts or the final processing of the data in a perl script, the color text gets clobbered.

For example, the original number, which I will put into $num:

123456

I input this in perl (or, much the same thing in bash):

"\e[1;34m" . $num . "\e[0m"

And here's what I get:

[1;34m123456[0m 

I've also tried this, an escaped version, but get the same thing:

"\x1B[1;34m" . $num . "\x1B[0m"

If this makes a difference, here's the real stack of scripts (it's a mess):

  • Perl called inside shell script, which then pipes to grep/cut/etc.
  • shell script with more cut/etc., then pipes to .pl
  • shell script watches that output

Maybe watch doesn't like color then?

I don't know perl too well, so I guess it wouldn't hurt to show what I'm seeing around the print lines:

my $format = ("%-8s") x scalar(@{$a[0]});

printf("$format\n", @$_)

Best Answer

I encountered the 'color-stealing' issues with watch, it's because watch uses a simple /bin/sh output style, which clobbers any color, aliases, shortcuts... the whole shebang!

So I cobbled this together, placed in my .bashrc, allows me to use all my own aliases, shortcuts and whatnot:

Usage:
    watcher 20 'somecommand | apipe | grep'

The number is the delay between updates, in seconds, the command can include anything you can quote.

###### 'watch' workalike, lets me use my aliases and such
function watcher() { WATCHERTIME=$1 ; WATCHERFILE=/tmp/watcher$$ ; shift ; while true ; do WATCHERHEIGHT=$(($LINES - 5)) ; ( eval $* ) | tail -n ${WATCHERHEIGHT} > ${WATCHERFILE} 2>/dev/null; clear ; /bin/echo -n "Every ${WATCHERTIME} seconds - " ; date ; /bin/echo ; cat ${WATCHERFILE} ; \rm -f ${WATCHERFILE} ; /bin/echo ; /bin/echo "==" ; sleep ${WATCHERTIME} ; done ; }

To break it down:

function watcher()
{ 
  WATCHERTIME=$1
  WATCHERFILE=/tmp/watcher$$
  shift
  while true; do
    WATCHERHEIGHT=$(($LINES - 5))
    ( eval $* ) | tail -n ${WATCHERHEIGHT} > ${WATCHERFILE} 2>/dev/null
    clear
    /bin/echo -n "Every ${WATCHERTIME} seconds - "
    date
    /bin/echo
    cat ${WATCHERFILE}
    \rm -f ${WATCHERFILE}
    /bin/echo
    /bin/echo "=="
    sleep ${WATCHERTIME}
  done
}

It determines the current screen height in lines, subtracts out enough for its own output, then repeatedly executes the command given, then clears the screen, displays the output and waits for the next loop. It displays a short '==' at the bottom to signify that's where the output ended. Sometime's it's helpful to know this.

I did it this way so there was smallest possible delay when displaying. if you don't capture output and then display, you get a long pause, then the output... icky.

Since it doesn't clobber colors, you get everything you're used to getting. Enjoy!