Saturday 8 February 2020

bash - How to pipe command output to other commands?


Example:


ls | echo prints nothing ( a blank line, actually ). I'd expect it to print a list of files.


ls | grep 'foo', on the other hand, works as expected ( prints files with 'foo' in their name ).


What I do in these situations is something like: ls | while read OUT; do echo $OUT; done but this is rather cumbersome.


Why does piping work with some commands, but not with others ? How can I circumvent this issue ?



Answer



There is a distinction between command line arguments and standard input. A pipe will connect standard output of one process to standard input of another. So


ls | echo

Connects standard output of ls to standard input of echo. Fine right? Well, echo ignores standard input and will dump its command line arguments - which are none in this case to - its own stdout. The output: nothing at all.


There are a few solutions in this case. One is to use a command that reads stdin and dumps to stdout, such as cat.


ls | cat

Will 'work', depending on what your definition of work is.


But what about the general case. What you really want is to convert stdout of one command to command line args of another. As others have said, xargs is the canonical helper tool in this case, reading its command line args for a command from its stdin, and constructing commands to run.


ls | xargs echo

You could also convert this some, using the substitution command $()


echo $(ls)

Would also do what you want.


Both of these tools are pretty core to shell scripting, you should learn both.


For completeness, as you indicate in the question, the other base way to convert stdin to command line args is the shell's builtin read command. It converts "words" (words as defined by the IFS variable) to a temp variable, which you can use in any command runs.


No comments:

Post a Comment

How can I VLOOKUP in multiple Excel documents?

I am trying to VLOOKUP reference data with around 400 seperate Excel files. Is it possible to do this in a quick way rather than doing it m...