Динамический регистр в bash - PullRequest
0 голосов
/ 14 января 2011

Я пытаюсь выяснить, как создать динамический оператор case в bash-скрипте.

Например, допустим, у меня есть вывод оператора awk со следующим содержимым

red
green
blue

В этом сценарии выходные данные могут измениться в любое время.

Затем я пытаюсь выполнить другую логику, если значение включено в этот вывод awk.

Так что есливышеприведенные данные находятся в $ list, тогда я концептуально хотел бы сделать что-то вроде:

case "${my_var}" in
    $list)
        .....
    something_else)
        .....
esac

Я пытаюсь использовать это для построения динамической пользовательской функции завершения вкладки (см. http://www.debian-administration.org/article/An_introduction_to_bash_completion_part_2 для некоторого фона).

Есть идеи?

Спасибо.

Ответы [ 4 ]

3 голосов
/ 04 ноября 2013

Вы можете создать динамический оператор case в bash, выполнив следующие действия:

1) убедитесь, что список разделен PIPE (|).IE.красный | зеленый | синий

2) обернуть вашу инструкцию case в eval

Например:

valid="red|green|blue"

eval "case \"$choice\" in
    $valid)
        echo do something good here
        ;;
    *)
        echo invalid colour
        ;;
esac"

Это работает для простой обработки переменных, я не могу этого гарантироватьбудет работать во всех случаях.

2 голосов
/ 14 января 2011

A case утверждение, вероятно, не является подходящим инструментом для работы. Если вы сохраняете вывод awk в массиве, вы можете циклически проходить по массиву, чтобы найти, есть ли в нем выбор, и в качестве бонуса определить, какой это тоже индекс.

#!/bin/bash

# Store command output in an array so each word is a separate array item.    
list=($(echo $'red\ngreen\nblue'))
my_var=blue

for ((i = 0; i < ${#list}; i++)); do
    if [[ ${list[$i]} = $my_var ]]; then
        echo "found at index $i"
        break
    fi
done

if ((i == ${#list})); then
    echo "not found"
fi
2 голосов
/ 14 января 2011

Вы не можете сделать это с помощью оператора case, но достаточно легко настроить собственного помощника для проверки членства в списке.

# stub to simulate this arbitrary call
my_awk_command() { printf '%s\n' red green blue; }
# helper to check list membership
list_contains() {
  local tgt="$1"; shift
  while (( $# )); do
    if [[ $1 = "$tgt" ]] ; then
      return 0
    fi
    shift
  done
  return 1
}

# the below is Bash 4 functionality; see BashFAQ #1 on how to replace it
readarray -t awk_output < <(my_awk_command)

if list_contains "$my_var" "${my_awk_command[@]}"; then
  ...something...
elif [[ "$my_var" = something_else ]] ; then
  ...something else...
fi
1 голос
/ 14 января 2011

Вы можете подойти к этому несколькими различными способами:

pattern=($(awk_command))     # red\ngreen\nblue\n
saveIFS=$IFS
IFS='|'
pattern="^(${pattern[*]})$"  # ^(red|green|blue)$  (perhaps hackish)
IFS=$saveIFS

# simple regex match if statement (not hackish)
if [[ $var =~ $pattern ]]
then
    do_something
fi

# or a backwards case statement (very hackish)
case 1 in    # this could be a variable or a command substitution
    $([[ $var =~ $pattern]] && echo 1) )  # the echo 1 could be another command or the 1 could be yet another variable
        do_something;;
    * )
        do_default;;
esac
...