30

This question concerns a bash script that is run in automator osx. I am using automator actions to get and filter a bunch of file references from the finder. Then I append to that list the name of the parent folder, also via an automator action. Automator then feeds these arguments to an action called "run shell script". I am not sure exactly how automator invokes the script but the argument list looks like this when echoed with: echo "$@"

/Volumes/G-Raid/Online/WAV_TEST/Testbok 50/01/01000 43-001.wav /Volumes/G-Raid/Online/WAV_TEST/Testbok 50/02/02000 43-002.wav /Volumes/G-Raid/Online/WAV_TEST/Testbok 50/03/03000 43-003.wav /Volumes/G-Raid/Online/WAV_TEST/Testbok 50

In this case path to 3 files and a folder.

In the shell script I launch an application called ripcheckc* with the args passed from automator minus the last argument(the folder) in the list.

I use this to remove the last argument:

_args=( "$@" )
unset _args[${#_args[@]}-1]

And this is echo $_args:

/Volumes/G-Raid/Online/WAV_TEST/Testbok 50/01/01000 43-001.wav /Volumes/G-Raid/Online/WAV_TEST/Testbok 50/02/02000 43-002.wav /Volumes/G-Raid/Online/WAV_TEST/Testbok 50/03/03000 43-003.wav

Same as before but without the folder.

Now, if I run ripcheckc with "$@" as argument it works (but fails later on because of that last path in the argument list) If I use ${_args[@]} the application will just abort silently. When I echo $@ and _args the output looks identical except for the last argument.

My question is - what is the difference between $@ and $_args that make the first valid input and the second not?

*The application is ripcheckc

I hope my question makes sense.

EDIT: Solved.

1
  • I solved it by using double quotes. Like this: ripcheckc "${_args[@]}".
    – Nordanfors
    Commented Dec 5, 2013 at 14:36

4 Answers 4

60

I have used this bash one-liner before

set -- "${@:1:$(($#-1))}"

It sets the argument list to the current argument list, less the last argument.


How it works:

  • $# is the number of arguments
  • $((...)) is an arithmetic expression, so $(($#-1)) is one less than the number of arguments.
  • ${variable:position:count} is a substring expression: it extracts count characters from variable starting at position. In the special case where variable is @, which means the argument list, it extracts count arguments from the list beginning at position. Here, position is 1 for the first argument and count is one less than the number of arguments worked out previously.
  • set -- arg1...argn sets the argument list to the given arguments

So the end result is that the argument list is replaced with a new list, where the new list is the original list except for the last argument.

6
  • Simple and elegant: I like it. I used a variation of @chepner's suggestion below to "remove" the first two arguments in order to properly spawn a sub-process: spawnProc "${@:3:$#}". Admittedly cryptic but that's due to BASH's roots. It's elegant though and simple. Commented Feb 24, 2017 at 15:47
  • Could I request a play-by-play of this thing?
    – Steven Lu
    Commented Dec 5, 2017 at 23:38
  • 1
    @StevenLu I have amended the answer to provide an explanation for you. I hope it helps.
    – starfry
    Commented Dec 6, 2017 at 8:47
  • That’s perfect!
    – Steven Lu
    Commented Dec 6, 2017 at 14:52
  • Great explanation, note there's an even more compact way of writing by @uvsmtid below, which is "${@:1:${#}-1}" Commented Mar 2, 2022 at 17:16
14

Assuming that you already have an array, you can say:

unset "array[${#array[@]}-1]"

For example, if your script contains:

array=( "$@" )
unset "array[${#array[@]}-1]"    # Removes last element -- also see: help unset
for i in "${array[@]}"; do
  echo "$i"
done

invoking it with: bash scriptname foo bar baz produces:

foo
bar
9
  • Thank you, that is a better way to get rid of the last argument. My issue is that there seems to be something different about $@ compared to $array that qualifies it as valid input for the application I am passing it to. I just can't figure out what it is.
    – Nordanfors
    Commented Dec 5, 2013 at 11:42
  • @user1047256 Try passing ${array[@]} to your application.
    – devnull
    Commented Dec 5, 2013 at 11:48
  • I tried what you suggested but still no go. Fyi I changed my nick to Nordanfors. :-)
    – Nordanfors
    Commented Dec 5, 2013 at 12:30
  • 2
    @Nordanfors Always quote variables.
    – devnull
    Commented Dec 5, 2013 at 14:41
  • 1
    The next answer should be accepted, not this one. It has triple the upvotes.
    – Noumenon
    Commented Dec 5, 2018 at 23:32
11

You can also get all but the last argument with

"${@:0:$#}"

which, honestly, is a little sketchy, since it seems to be (ab)using the fact that arguments are numbered starting with 1, not 0.

Update: This only works due to a bug (fixed in 4.1.2 at the latest) in handling $@. It works in version 3.2.

4
  • 11
    Code "${@:1:${#}-1}" is exactly what makes full list of arguments like in "$@" without the last one.
    – uvsmtid
    Commented Mar 14, 2015 at 9:49
  • Right. The sketchy part is using 0 as the first argument to avoid having to subtract 1 from the second.
    – chepner
    Commented Mar 14, 2015 at 11:33
  • Actually, index 0 includes non-argument - path to the script being executed (a value automatically provided by shell). You are not avoiding subtraction of 1 - the last argument is still included and you must subtract 1. Using 0 is simply adding one more (likely not required) value to the list.
    – uvsmtid
    Commented Mar 14, 2015 at 11:38
  • 1
    It would appear this worked in bash 3.2 due to a bug that was subsequently fixed.
    – chepner
    Commented Mar 14, 2015 at 11:43
2
set -- "${@:1:$#-1}"

sets the parameter list to first up to penultimate, removing the last one

Not the answer you're looking for? Browse other questions tagged or ask your own question.