Properly quote bash alias definition
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
add a comment |
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
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 functioncsfiles
. 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, thefind
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 tosh
.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
add a comment |
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
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
bash alias
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 functioncsfiles
. 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, thefind
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 tosh
.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
add a comment |
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 functioncsfiles
. 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, thefind
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 tosh
.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
add a comment |
2 Answers
2
active
oldest
votes
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
}
add a comment |
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"
Good call with escaping the inner command substitution, +1.
– Jonny Henly
Nov 20 '18 at 18:17
add a comment |
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
});
}
});
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
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
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
}
add a comment |
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
}
add a comment |
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
}
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
}
edited Nov 20 '18 at 18:44
answered Nov 20 '18 at 18:31
chepnerchepner
248k33234328
248k33234328
add a comment |
add a comment |
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"
Good call with escaping the inner command substitution, +1.
– Jonny Henly
Nov 20 '18 at 18:17
add a comment |
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"
Good call with escaping the inner command substitution, +1.
– Jonny Henly
Nov 20 '18 at 18:17
add a comment |
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"
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"
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
add a comment |
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
add a comment |
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.
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
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
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Post as a guest
Required, but never shown
Sign up or log in
StackExchange.ready(function () {
StackExchange.helpers.onClickDraftSave('#login-link');
});
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
Sign up using Google
Sign up using Facebook
Sign up using Email and Password
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
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 functioncsfiles
. 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 tosh
.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