Properly quote bash alias definition












1















I have the following command that I am trying to put into a bash alias. The command by itself works fine, but when I try to alias it, I am getting the following errors:



The Command



find . -maxdepth 1 -mindepth 1 -type d -exec sh -c 'echo "$(find "{}" -type f | wc -l)" {}' ; | sort -nr


The Alias



alias csfiles='find . -maxdepth 1 -mindepth 1 -type d -exec sh -c 'echo "$(find "{}" -type f | wc -l)" {}' ; | sort -nr'


The Error:



-sh: alias 0: not found
-sh: alias {} ; | sort nr: not found


I think this means I am not using quotes right but I am having trouble determining the correct combo. Help?










share|improve this question




















  • 2





    It's been a while since I've worked with aliases but I think your alias is too complex. I believe alias is just used to shorten one command with arguments, i.e. alias l='ls -al'. You may need to create a function csfiles. But it has been a while so I may be mistaken.

    – Jonny Henly
    Nov 20 '18 at 17:59













  • see stackoverflow.com/questions/25941394/…

    – hek2mgl
    Nov 20 '18 at 18:01






  • 2





    I second creating a bash function to do this. You would call it the same way, with the benefit of being able to pass arguments if needed.

    – ahota
    Nov 20 '18 at 18:01






  • 2





    As written, the find command will fail for certain file names (such as those containing double quotes). Don't try to embed {} in the command; only use it as an argument to sh. sh -c '....' _ {}. Wherever you used {}, use $1.

    – chepner
    Nov 20 '18 at 18:17











  • @chepner that's a very good point!

    – hek2mgl
    Nov 20 '18 at 18:18
















1















I have the following command that I am trying to put into a bash alias. The command by itself works fine, but when I try to alias it, I am getting the following errors:



The Command



find . -maxdepth 1 -mindepth 1 -type d -exec sh -c 'echo "$(find "{}" -type f | wc -l)" {}' ; | sort -nr


The Alias



alias csfiles='find . -maxdepth 1 -mindepth 1 -type d -exec sh -c 'echo "$(find "{}" -type f | wc -l)" {}' ; | sort -nr'


The Error:



-sh: alias 0: not found
-sh: alias {} ; | sort nr: not found


I think this means I am not using quotes right but I am having trouble determining the correct combo. Help?










share|improve this question




















  • 2





    It's been a while since I've worked with aliases but I think your alias is too complex. I believe alias is just used to shorten one command with arguments, i.e. alias l='ls -al'. You may need to create a function csfiles. But it has been a while so I may be mistaken.

    – Jonny Henly
    Nov 20 '18 at 17:59













  • see stackoverflow.com/questions/25941394/…

    – hek2mgl
    Nov 20 '18 at 18:01






  • 2





    I second creating a bash function to do this. You would call it the same way, with the benefit of being able to pass arguments if needed.

    – ahota
    Nov 20 '18 at 18:01






  • 2





    As written, the find command will fail for certain file names (such as those containing double quotes). Don't try to embed {} in the command; only use it as an argument to sh. sh -c '....' _ {}. Wherever you used {}, use $1.

    – chepner
    Nov 20 '18 at 18:17











  • @chepner that's a very good point!

    – hek2mgl
    Nov 20 '18 at 18:18














1












1








1


1






I have the following command that I am trying to put into a bash alias. The command by itself works fine, but when I try to alias it, I am getting the following errors:



The Command



find . -maxdepth 1 -mindepth 1 -type d -exec sh -c 'echo "$(find "{}" -type f | wc -l)" {}' ; | sort -nr


The Alias



alias csfiles='find . -maxdepth 1 -mindepth 1 -type d -exec sh -c 'echo "$(find "{}" -type f | wc -l)" {}' ; | sort -nr'


The Error:



-sh: alias 0: not found
-sh: alias {} ; | sort nr: not found


I think this means I am not using quotes right but I am having trouble determining the correct combo. Help?










share|improve this question
















I have the following command that I am trying to put into a bash alias. The command by itself works fine, but when I try to alias it, I am getting the following errors:



The Command



find . -maxdepth 1 -mindepth 1 -type d -exec sh -c 'echo "$(find "{}" -type f | wc -l)" {}' ; | sort -nr


The Alias



alias csfiles='find . -maxdepth 1 -mindepth 1 -type d -exec sh -c 'echo "$(find "{}" -type f | wc -l)" {}' ; | sort -nr'


The Error:



-sh: alias 0: not found
-sh: alias {} ; | sort nr: not found


I think this means I am not using quotes right but I am having trouble determining the correct combo. Help?







bash alias






share|improve this question















share|improve this question













share|improve this question




share|improve this question








edited Nov 20 '18 at 18:00









hek2mgl

107k13144166




107k13144166










asked Nov 20 '18 at 17:53









zeeplezeeple

3,06263159




3,06263159








  • 2





    It's been a while since I've worked with aliases but I think your alias is too complex. I believe alias is just used to shorten one command with arguments, i.e. alias l='ls -al'. You may need to create a function csfiles. But it has been a while so I may be mistaken.

    – Jonny Henly
    Nov 20 '18 at 17:59













  • see stackoverflow.com/questions/25941394/…

    – hek2mgl
    Nov 20 '18 at 18:01






  • 2





    I second creating a bash function to do this. You would call it the same way, with the benefit of being able to pass arguments if needed.

    – ahota
    Nov 20 '18 at 18:01






  • 2





    As written, the find command will fail for certain file names (such as those containing double quotes). Don't try to embed {} in the command; only use it as an argument to sh. sh -c '....' _ {}. Wherever you used {}, use $1.

    – chepner
    Nov 20 '18 at 18:17











  • @chepner that's a very good point!

    – hek2mgl
    Nov 20 '18 at 18:18














  • 2





    It's been a while since I've worked with aliases but I think your alias is too complex. I believe alias is just used to shorten one command with arguments, i.e. alias l='ls -al'. You may need to create a function csfiles. But it has been a while so I may be mistaken.

    – Jonny Henly
    Nov 20 '18 at 17:59













  • see stackoverflow.com/questions/25941394/…

    – hek2mgl
    Nov 20 '18 at 18:01






  • 2





    I second creating a bash function to do this. You would call it the same way, with the benefit of being able to pass arguments if needed.

    – ahota
    Nov 20 '18 at 18:01






  • 2





    As written, the find command will fail for certain file names (such as those containing double quotes). Don't try to embed {} in the command; only use it as an argument to sh. sh -c '....' _ {}. Wherever you used {}, use $1.

    – chepner
    Nov 20 '18 at 18:17











  • @chepner that's a very good point!

    – hek2mgl
    Nov 20 '18 at 18:18








2




2





It's been a while since I've worked with aliases but I think your alias is too complex. I believe alias is just used to shorten one command with arguments, i.e. alias l='ls -al'. You may need to create a function csfiles. But it has been a while so I may be mistaken.

– Jonny Henly
Nov 20 '18 at 17:59







It's been a while since I've worked with aliases but I think your alias is too complex. I believe alias is just used to shorten one command with arguments, i.e. alias l='ls -al'. You may need to create a function csfiles. But it has been a while so I may be mistaken.

– Jonny Henly
Nov 20 '18 at 17:59















see stackoverflow.com/questions/25941394/…

– hek2mgl
Nov 20 '18 at 18:01





see stackoverflow.com/questions/25941394/…

– hek2mgl
Nov 20 '18 at 18:01




2




2





I second creating a bash function to do this. You would call it the same way, with the benefit of being able to pass arguments if needed.

– ahota
Nov 20 '18 at 18:01





I second creating a bash function to do this. You would call it the same way, with the benefit of being able to pass arguments if needed.

– ahota
Nov 20 '18 at 18:01




2




2





As written, the find command will fail for certain file names (such as those containing double quotes). Don't try to embed {} in the command; only use it as an argument to sh. sh -c '....' _ {}. Wherever you used {}, use $1.

– chepner
Nov 20 '18 at 18:17





As written, the find command will fail for certain file names (such as those containing double quotes). Don't try to embed {} in the command; only use it as an argument to sh. sh -c '....' _ {}. Wherever you used {}, use $1.

– chepner
Nov 20 '18 at 18:17













@chepner that's a very good point!

– hek2mgl
Nov 20 '18 at 18:18





@chepner that's a very good point!

– hek2mgl
Nov 20 '18 at 18:18












2 Answers
2






active

oldest

votes


















4














Your outer find doesn't do anything you couldn't do with a simple glob. This eliminates a layer of quotes (along with the sh process for each directory found).



# Ignoring the issue of assuming no file name contains a newline
for d in ./*/; do
echo "$(find "$d" -type f | wc -l) $d"
done


Just define a shell function to eliminate the second layer imposed on the argument to alias.



csfiles () {
for d in ./*/; do
echo "$(find "$d" -type f | wc -l) $d"
done
}


The remaining call(s) to find can also be replaced with a for loop, eliminating the problematic assumption of one line per file name:



csfiles () {
for d in ./*/; do
echo "$(for f in "$d"/*; do [ -f "$f" ] && echo; done | wc -l) $d"
done
}


You could keep find if it supports the -printf primary, because you don't care about the actual names of the files, just that you get exactly one line of output per file.



csfiles () {
for d in ./*/; do
echo "$(find "$d" -type f -printf . | wc -l) $d"
done
}





share|improve this answer

































    1














    You can use double quotes around the definition, like this:



    alias foo="find . -maxdepth 1 -mindepth 1 -type d -exec sh -c 'echo "$(find "{}" -type f | wc -l)" {}' ; | sort -nr"


    Every literal " inside the definition gets escaped: ".



    Note: You also need to escape the inner command substitution to prevent it from getting expanded upon alias definition time. Like this ... $(...)





    As a follow up on chepners comment, you should pass the filename to the inner find command as an argument. Otherwise you will run into problems if one of your folders has a name with a " in it:



    alias foo="find . -maxdepth 1 -mindepth 1 -type d -exec bash -c 'echo "$(find "${1}" -type f | wc -l) "${1}" "' -- "{}" ; | sort -nr"





    share|improve this answer


























    • Good call with escaping the inner command substitution, +1.

      – Jonny Henly
      Nov 20 '18 at 18:17











    Your Answer






    StackExchange.ifUsing("editor", function () {
    StackExchange.using("externalEditor", function () {
    StackExchange.using("snippets", function () {
    StackExchange.snippets.init();
    });
    });
    }, "code-snippets");

    StackExchange.ready(function() {
    var channelOptions = {
    tags: "".split(" "),
    id: "1"
    };
    initTagRenderer("".split(" "), "".split(" "), channelOptions);

    StackExchange.using("externalEditor", function() {
    // Have to fire editor after snippets, if snippets enabled
    if (StackExchange.settings.snippets.snippetsEnabled) {
    StackExchange.using("snippets", function() {
    createEditor();
    });
    }
    else {
    createEditor();
    }
    });

    function createEditor() {
    StackExchange.prepareEditor({
    heartbeatType: 'answer',
    autoActivateHeartbeat: false,
    convertImagesToLinks: true,
    noModals: true,
    showLowRepImageUploadWarning: true,
    reputationToPostImages: 10,
    bindNavPrevention: true,
    postfix: "",
    imageUploader: {
    brandingHtml: "Powered by u003ca class="icon-imgur-white" href="https://imgur.com/"u003eu003c/au003e",
    contentPolicyHtml: "User contributions licensed under u003ca href="https://creativecommons.org/licenses/by-sa/3.0/"u003ecc by-sa 3.0 with attribution requiredu003c/au003e u003ca href="https://stackoverflow.com/legal/content-policy"u003e(content policy)u003c/au003e",
    allowUrls: true
    },
    onDemand: true,
    discardSelector: ".discard-answer"
    ,immediatelyShowMarkdownHelp:true
    });


    }
    });














    draft saved

    draft discarded


















    StackExchange.ready(
    function () {
    StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fstackoverflow.com%2fquestions%2f53398792%2fproperly-quote-bash-alias-definition%23new-answer', 'question_page');
    }
    );

    Post as a guest















    Required, but never shown

























    2 Answers
    2






    active

    oldest

    votes








    2 Answers
    2






    active

    oldest

    votes









    active

    oldest

    votes






    active

    oldest

    votes









    4














    Your outer find doesn't do anything you couldn't do with a simple glob. This eliminates a layer of quotes (along with the sh process for each directory found).



    # Ignoring the issue of assuming no file name contains a newline
    for d in ./*/; do
    echo "$(find "$d" -type f | wc -l) $d"
    done


    Just define a shell function to eliminate the second layer imposed on the argument to alias.



    csfiles () {
    for d in ./*/; do
    echo "$(find "$d" -type f | wc -l) $d"
    done
    }


    The remaining call(s) to find can also be replaced with a for loop, eliminating the problematic assumption of one line per file name:



    csfiles () {
    for d in ./*/; do
    echo "$(for f in "$d"/*; do [ -f "$f" ] && echo; done | wc -l) $d"
    done
    }


    You could keep find if it supports the -printf primary, because you don't care about the actual names of the files, just that you get exactly one line of output per file.



    csfiles () {
    for d in ./*/; do
    echo "$(find "$d" -type f -printf . | wc -l) $d"
    done
    }





    share|improve this answer






























      4














      Your outer find doesn't do anything you couldn't do with a simple glob. This eliminates a layer of quotes (along with the sh process for each directory found).



      # Ignoring the issue of assuming no file name contains a newline
      for d in ./*/; do
      echo "$(find "$d" -type f | wc -l) $d"
      done


      Just define a shell function to eliminate the second layer imposed on the argument to alias.



      csfiles () {
      for d in ./*/; do
      echo "$(find "$d" -type f | wc -l) $d"
      done
      }


      The remaining call(s) to find can also be replaced with a for loop, eliminating the problematic assumption of one line per file name:



      csfiles () {
      for d in ./*/; do
      echo "$(for f in "$d"/*; do [ -f "$f" ] && echo; done | wc -l) $d"
      done
      }


      You could keep find if it supports the -printf primary, because you don't care about the actual names of the files, just that you get exactly one line of output per file.



      csfiles () {
      for d in ./*/; do
      echo "$(find "$d" -type f -printf . | wc -l) $d"
      done
      }





      share|improve this answer




























        4












        4








        4







        Your outer find doesn't do anything you couldn't do with a simple glob. This eliminates a layer of quotes (along with the sh process for each directory found).



        # Ignoring the issue of assuming no file name contains a newline
        for d in ./*/; do
        echo "$(find "$d" -type f | wc -l) $d"
        done


        Just define a shell function to eliminate the second layer imposed on the argument to alias.



        csfiles () {
        for d in ./*/; do
        echo "$(find "$d" -type f | wc -l) $d"
        done
        }


        The remaining call(s) to find can also be replaced with a for loop, eliminating the problematic assumption of one line per file name:



        csfiles () {
        for d in ./*/; do
        echo "$(for f in "$d"/*; do [ -f "$f" ] && echo; done | wc -l) $d"
        done
        }


        You could keep find if it supports the -printf primary, because you don't care about the actual names of the files, just that you get exactly one line of output per file.



        csfiles () {
        for d in ./*/; do
        echo "$(find "$d" -type f -printf . | wc -l) $d"
        done
        }





        share|improve this answer















        Your outer find doesn't do anything you couldn't do with a simple glob. This eliminates a layer of quotes (along with the sh process for each directory found).



        # Ignoring the issue of assuming no file name contains a newline
        for d in ./*/; do
        echo "$(find "$d" -type f | wc -l) $d"
        done


        Just define a shell function to eliminate the second layer imposed on the argument to alias.



        csfiles () {
        for d in ./*/; do
        echo "$(find "$d" -type f | wc -l) $d"
        done
        }


        The remaining call(s) to find can also be replaced with a for loop, eliminating the problematic assumption of one line per file name:



        csfiles () {
        for d in ./*/; do
        echo "$(for f in "$d"/*; do [ -f "$f" ] && echo; done | wc -l) $d"
        done
        }


        You could keep find if it supports the -printf primary, because you don't care about the actual names of the files, just that you get exactly one line of output per file.



        csfiles () {
        for d in ./*/; do
        echo "$(find "$d" -type f -printf . | wc -l) $d"
        done
        }






        share|improve this answer














        share|improve this answer



        share|improve this answer








        edited Nov 20 '18 at 18:44

























        answered Nov 20 '18 at 18:31









        chepnerchepner

        248k33234328




        248k33234328

























            1














            You can use double quotes around the definition, like this:



            alias foo="find . -maxdepth 1 -mindepth 1 -type d -exec sh -c 'echo "$(find "{}" -type f | wc -l)" {}' ; | sort -nr"


            Every literal " inside the definition gets escaped: ".



            Note: You also need to escape the inner command substitution to prevent it from getting expanded upon alias definition time. Like this ... $(...)





            As a follow up on chepners comment, you should pass the filename to the inner find command as an argument. Otherwise you will run into problems if one of your folders has a name with a " in it:



            alias foo="find . -maxdepth 1 -mindepth 1 -type d -exec bash -c 'echo "$(find "${1}" -type f | wc -l) "${1}" "' -- "{}" ; | sort -nr"





            share|improve this answer


























            • Good call with escaping the inner command substitution, +1.

              – Jonny Henly
              Nov 20 '18 at 18:17
















            1














            You can use double quotes around the definition, like this:



            alias foo="find . -maxdepth 1 -mindepth 1 -type d -exec sh -c 'echo "$(find "{}" -type f | wc -l)" {}' ; | sort -nr"


            Every literal " inside the definition gets escaped: ".



            Note: You also need to escape the inner command substitution to prevent it from getting expanded upon alias definition time. Like this ... $(...)





            As a follow up on chepners comment, you should pass the filename to the inner find command as an argument. Otherwise you will run into problems if one of your folders has a name with a " in it:



            alias foo="find . -maxdepth 1 -mindepth 1 -type d -exec bash -c 'echo "$(find "${1}" -type f | wc -l) "${1}" "' -- "{}" ; | sort -nr"





            share|improve this answer


























            • Good call with escaping the inner command substitution, +1.

              – Jonny Henly
              Nov 20 '18 at 18:17














            1












            1








            1







            You can use double quotes around the definition, like this:



            alias foo="find . -maxdepth 1 -mindepth 1 -type d -exec sh -c 'echo "$(find "{}" -type f | wc -l)" {}' ; | sort -nr"


            Every literal " inside the definition gets escaped: ".



            Note: You also need to escape the inner command substitution to prevent it from getting expanded upon alias definition time. Like this ... $(...)





            As a follow up on chepners comment, you should pass the filename to the inner find command as an argument. Otherwise you will run into problems if one of your folders has a name with a " in it:



            alias foo="find . -maxdepth 1 -mindepth 1 -type d -exec bash -c 'echo "$(find "${1}" -type f | wc -l) "${1}" "' -- "{}" ; | sort -nr"





            share|improve this answer















            You can use double quotes around the definition, like this:



            alias foo="find . -maxdepth 1 -mindepth 1 -type d -exec sh -c 'echo "$(find "{}" -type f | wc -l)" {}' ; | sort -nr"


            Every literal " inside the definition gets escaped: ".



            Note: You also need to escape the inner command substitution to prevent it from getting expanded upon alias definition time. Like this ... $(...)





            As a follow up on chepners comment, you should pass the filename to the inner find command as an argument. Otherwise you will run into problems if one of your folders has a name with a " in it:



            alias foo="find . -maxdepth 1 -mindepth 1 -type d -exec bash -c 'echo "$(find "${1}" -type f | wc -l) "${1}" "' -- "{}" ; | sort -nr"






            share|improve this answer














            share|improve this answer



            share|improve this answer








            edited Nov 20 '18 at 18:26

























            answered Nov 20 '18 at 18:08









            hek2mglhek2mgl

            107k13144166




            107k13144166













            • Good call with escaping the inner command substitution, +1.

              – Jonny Henly
              Nov 20 '18 at 18:17



















            • Good call with escaping the inner command substitution, +1.

              – Jonny Henly
              Nov 20 '18 at 18:17

















            Good call with escaping the inner command substitution, +1.

            – Jonny Henly
            Nov 20 '18 at 18:17





            Good call with escaping the inner command substitution, +1.

            – Jonny Henly
            Nov 20 '18 at 18:17


















            draft saved

            draft discarded




















































            Thanks for contributing an answer to Stack Overflow!


            • Please be sure to answer the question. Provide details and share your research!

            But avoid



            • Asking for help, clarification, or responding to other answers.

            • Making statements based on opinion; back them up with references or personal experience.


            To learn more, see our tips on writing great answers.




            draft saved


            draft discarded














            StackExchange.ready(
            function () {
            StackExchange.openid.initPostLogin('.new-post-login', 'https%3a%2f%2fstackoverflow.com%2fquestions%2f53398792%2fproperly-quote-bash-alias-definition%23new-answer', 'question_page');
            }
            );

            Post as a guest















            Required, but never shown





















































            Required, but never shown














            Required, but never shown












            Required, but never shown







            Required, but never shown

































            Required, but never shown














            Required, but never shown












            Required, but never shown







            Required, but never shown







            Popular posts from this blog

            MongoDB - Not Authorized To Execute Command

            Npm cannot find a required file even through it is in the searched directory

            in spring boot 2.1 many test slices are not allowed anymore due to multiple @BootstrapWith