Как избежать расширенных шаблонов расширения пути в выражениях в кавычках?

В дополнение к базовым шаблонам *, ? и [...] оболочка Bash предоставляет расширенные операторы сопоставления шаблонов, такие как !(pattern-list) ("соответствует всем, кроме одного из заданных шаблонов"). Для их использования необходимо установить параметр оболочки extglob. Пример:

~$ mkdir test ; cd test ; touch file1 file2 file3
~/test$ echo *
file1 file2 file3
~/test$ shopt -s extglob  # make sure extglob is set
~/test$ echo !(file2)
file1 file3

Если я передам выражение оболочки программе, которая выполняет его во вложенной оболочке, оператор вызовет ошибку. Вот тест, который запускает подоболочку напрямую (здесь я выполняю из другого каталога, чтобы убедиться, что расширение не произойдет преждевременно):

~/test$ cd ..
~$ bash -c "cd test ; echo *"
file1 file2 file3
~$ bash -c "cd test ; echo !(file2)"  # expected output: file1 file3
bash: -c: line 0: syntax error near unexpected token `('
bash: -c: line 0: `cd test ; echo !(file2)'

Я пробовал все виды побега, но ничего из того, что я придумал, не работало правильно. Я также подозревал, что extglob не установлен в подоболочке, но это не так:

~$ bash -c "shopt -s extglob ; cd test ; echo !(file2)"
bash: -c: line 0: syntax error near unexpected token `('
bash: -c: line 0: `cd test ; echo !(file2)'

Любое решение приветствуется!


person akaihola    schedule 09.02.2009    source источник


Ответы (3)


bash анализирует каждую строку перед ее выполнением, поэтому "shop -s extglob" не вступит в силу, когда bash проверяет синтаксис шаблона подстановки. Опция не может быть включена в той же строке. Вот почему решение «bash -O extglob -c 'xyz'» (от Рэнди Проктора) работает и необходимо.

person Tony Delroy    schedule 12.08.2010

Вот еще один способ, если вы хотите избежать eval и вам нужна возможность включать и выключать extglob в подоболочке. Просто поместите свой шаблон в переменную:

bash -c 'shopt -s extglob; cd test; patt="!(file2)"; echo $patt; shopt -u extglob; echo $patt'

дает этот вывод:

file1 file3
!(file2)

демонстрируя, что extglob был установлен и отключен. Если бы первый echo имел кавычки вокруг $patt, он просто выдал бы шаблон, как второй echo (который, вероятно, должен быть в кавычках).

person Dennis Williamson    schedule 05.06.2009

Что ж, у меня нет реального опыта работы с extglob, но я могу заставить его работать, заключив echo в eval:

$ bash -c 'shopt -s extglob ; cd  test ; eval "echo !(file2)"'
file1 file3
person Nietzche-jou    schedule 09.02.2009