Passing flags to command via variables in bash

后端 未结 2 1152
南笙
南笙 2021-01-14 01:20

I have a complex script that takes variables from files and uses them to run programs (Wine specifically)

Passing options from the variables in the other file isn\'t

相关标签:
2条回答
  • 2021-01-14 01:56

    Short answer: see BashFAQ #050: I'm trying to put a command in a variable, but the complex cases always fail!

    Long answer: putting quotes in a variable doesn't do anything useful, because bash parses quotes before it replaces variable with their values; this means that by the time it's replaced $run with run.exe -withoption "This text" it's already done quote parsing, and isn't going to go back and notice those double-quotes, which means they don't do what you expect.

    BTW, using echo to check what's happening is extremely misleading, because echo shows you what its arguments looked like after they're parsed, but you're reading it as though it was showing them before parsing. Instead, use @Eduardo's suggestion of -x to see what's really going. (BTW, you can also use set -x to turn that on for the interesting part of the script, then set +x to turn it off.)

    So how do you solve it? Generally the best way is to put your command in an array rather than a simple variable, and then use the idiom "${arrayname[@]}" to pass each array element as a separate argument:

    run=(run.exe -withoption "This text")
    wine "${run[@]}"
    
    0 讨论(0)
  • 2021-01-14 02:20

    The problem is that "This and text" are treated as separate arguments, each containing a double-quote, rather than as a single argument This text. You can see this if you write a function to print out one argument per line; this:

    function echo_on_separate_lines ()
    {
      local arg
      for arg in "$@" ; do
        echo "<< $arg >>"
      done
    }
    run="run.exe -withoption \"This text\""
    echo_on_separate_lines $run
    

    prints this:

    << run.exe >>
    << -withoption >>
    << "This >>
    << text" >>
    

    rather than this:

    << run.exe >>
    << -withoption >>
    << This text >>
    

    The simplest solution is to tack on an eval to re-process the quoting:

    run="run.exe -withoption \"This text\""
    wine $run     # or better yet:   wine "$run"
    

    But a more robust solution is to have run be an array, and then you can refer to it as "${run[@]}":

    run=(run.exe -withoption "This text")
    wine "${run[@]}"
    

    so that the quoting is handled properly from the get-go.

    0 讨论(0)
提交回复
热议问题