У меня есть список строк, и я хочу передать эти строки в качестве аргументов в одном вызове командной строки Bash. Для простых буквенно-цифровых строк достаточно просто передать их дословно:
> script.pl foo bar baz yes no
foo
bar
baz
yes
no
Я понимаю, что если аргумент содержит пробелы или обратную косую черту или двойные кавычки, мне нужно обратная косая черта - избегайте двойных кавычек и обратных косых черт, а затем дважды указывайте аргумент.
> script.pl foo bar baz "\"yes\"\\\"no\""
foo
bar
baz
"yes"\"no"
Но когда аргумент содержит восклицательный знак, это происходит:
> script.pl !foo
-bash: !foo: event not found
Двойное цитирование не работает:
> script.pl "!foo"
-bash: !foo: event not found
Не работает обратная косая черта (обратите внимание, как буквальный обратный слэш присутствует на выходе):
> script.pl "\!foo"
\!foo
Я мало знаю о Bash, но я знаю, что есть другие специальные персонажи, которые делают подобные вещи. Какова общая процедура безопасного экранирования произвольной строки для использования в качестве аргумента командной строки в Bash?. Предположим, что строка может быть произвольной длины и содержать произвольные комбинации специальных символов. Я хотел бы использовать подпрограмму escape()
, которую я могу использовать, как показано ниже (пример Perl):
$cmd = join " ", map { escape($_); } @args;
Вот несколько примеров строк, которые должны быть безопасно экранированы этой функцией (я знаю, что некоторые из них выглядят как Windows, например, преднамеренно):
yes
no
Hello, world [string with a comma and space in it]
C:\Program Files\ [path with backslashes and a space in it]
" [i.e. a double-quote]
\ [backslash]
\\ [two backslashes]
\\\ [three backslashes]
\\\\ [four backslashes]
\\\\\ [five backslashes]
"\ [double-quote, backslash]
"\T [double-quote, backslash, T]
"\\T [double-quote, backslash, backslash, T]
!1
!A
"!\/'" [double-quote, exclamation, backslash, forward slash, apostrophe, double quote]
"Jeff's!" [double-quote, J, e, f, f, apostrophe, s, exclamation, double quote]
$PATH
%PATH%
&
<>|&^
*@[email protected]#?-_
EDIT:
Будет ли это трюк? Побег каждого необычного персонажа с обратной косой чертой и опускание одиночных или двойных кавычек. (Пример в Perl, но любой язык может это сделать)
sub escape {
$_[0] =~ s/([^a-zA-Z0-9_])/\\$1/g;
return $_[0];
}