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
    3
    ·
    4 days ago

    Great minds, lol. I have almost the exact same command set up as a little script. Mine has an extra modification for my use case, and I named mine iter, but foreach is a good name for it too.

      • kittenroar@beehaw.org
        link
        fedilink
        English
        arrow-up
        2
        ·
        edit-2
        3 hours 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
          1
          ·
          edit-2
          3 hours 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