I created a simple alias for xargs, with the intend to pipe it when needed. It will simply run a command for each line of it. My question to you is, is this useful or are there better ways of doing this? This is just a little bit of brainstorming basically. Maybe I have a knot in my head.

# Pipe each line and execute a command. The "{}" will be replaced by the line.
# Example:
#   find . -maxdepth 2 -type f -name 'M*' | foreach grep "USB" {}
alias foreach='xargs -d "\n" -I{}'

For commands that already operate on every line from stdin, this won’t be much useful. But in other cases, it might be. A more simplified usage example (and a useless one) would be:

find . -maxdepth 1 | foreach echo "File" {}

It’s important to use the {} as a placeholder for the “current line” that is processed. What do you think about the usefulness? Have you any idea how to use it?

  • kittenroar@beehaw.org
    link
    fedilink
    English
    arrow-up
    2
    ·
    edit-2
    1 day ago

    Here’s the code:

    #!/bin/bash
    cmd=$@
    if echo $cmd | grep '/$'
    then
        xargs -rd '\n' -I {} $cmd{}
    else
        xargs -rd '\n' -I {} $cmd {}
    fi
    

    Usage is like:

    ls *zip | iter shasum
    

    or

    ls *zip | iter shasum ../zipfile_archive/
    

    The second one would get the shasum of zip files that have the same name as ones in the cwd

    This assumes, of course, that the input files have sane names, ie, no spaces, etc

    • thingsiplay@beehaw.orgOP
      link
      fedilink
      arrow-up
      2
      ·
      edit-2
      1 day ago

      Thanks for posting. I find the echo part and extra use of variable is a little bit flaky. Here is a modified version. But I am not 100% sure if its doing what your script is doing.

      I skipped the extra variable and echo and grep, by comparing its content with ${*}, which is similar to ${@}, but won’t separate each argument and create a single string instead. The =~ /$ is a regex comparison, which Bash supports native. Then I am using ${@} for the call, which separates each argument. Maybe this could be done with ${*} instead. I’m not sure which of them is the correct one for this case. At least it seems filenames with spaces work. Otherwise, not claiming it would be better. Just giving some food for thoughts.

      #!/usr/bin/bash
      
      if [[ "${*}" =~ /$ ]]; then
          xargs -rd '\n' -I {} "${@}"{}
      else
          xargs -rd '\n' -I {} "${@}" {}
      fi