У меня есть список строк, и я хочу передать эти строки в качестве аргументов в одном вызове командной строки 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? Предположим, что строка может иметь произвольную длину и содержать произвольные комбинации специальных символов. Мне нужна подпрограмма 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%
&
<>|&^
*@$$A$@#?-_
EDIT:
Это бы сработало? Избегайте каждого необычного символа с помощью обратной косой черты и пропускайте одинарные или двойные кавычки (Пример на Perl, но любой язык может сделать это)
sub escape {
$_[0] =~ s/([^a-zA-Z0-9_])/\\$1/g;
return $_[0];
}