Is there anyway to pass terminal colors through a pipe?
As a simple example, ls -l --color=always | grep ii
.
When you just run the ls -l --color=always
part alone, you get the filenames color coded. But adding grep ii
removes the color coding and just has the grep
match highlighting.
Screenshot of both examples:
In the above example I would want ii.mp3
and ii.png
filenames to retain the cyan and magenta highlighting, respectively. With or without the grep
match highlighting.
Question is not specific to ls
or grep
.
If this is possible, is there a correct term/name for it? I am unable to locate anything.
Color codes will pass through pipes just like any other output.
In this case, your grep is being smarter than you want and actually parsing the incoming color codes itself.
You can try a simpler program like
head
,tail
, or evensed -n /ii/p
to see it for yourself.You can also control GNU grep’s color processing with
--color
but you may not find exactly what you seek.hmm this gives me all colors (from ls and grep)
/usr/bin/ls -l --color=always | /usr/bin/grep --color=always a
(I’m using their full paths because I usually alias
ls
andgrep
toeza
andrg
respectively)There’s a
export CLICOLOR_FORCE=1
you can try to avoid repeating--color=always
, but that didn’t work for me withls
andgrep
specifically.Edit: there’s also a
FORCE_COLOR=1
that is popular, but again, neitherls
norgrep
seem to care.TL;DR - Essentially you’re attempting to mix two types of output, a pipe with a terminal. This is pretty much not going to work as expected.
To make colour in a terminal, commands like
ls
add so-called Escape sequences, a series of bytes that your terminal knows how to interpret as colour.Whilst you might be able to force those characters though a pipe, they’re just characters, so if you only grab part of those characters, you’ll create invalid Escape sequences and all hell will break loose, exactly like what happens if you run
cat
on a binary file and the terminal display goes haywire. You can often recover using thereset
command.This is why programs like
ls
andgrep
detect if they’re running as a terminal command or a pipe command and suppress the Escape sequences when you are sending their output to anything other than a terminal.For the most part, this happens because those programs check if stdout is a pseudo-terminal (pty) and automatically disable color output because if you’re doing say
ls -l
and try to parse it, you’ll have all the ANSI escape sequences mixed in, so for safety and predictability they disable color.It is unfortunately a per-program thing. It is possible to fake it using
script
orunbuffer
according to https://stackoverflow.com/a/32981392Looks like
socat
can also be used for that: https://unix.stackexchange.com/a/157463