AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • Início
  • system&network
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • Início
  • system&network
    • Recentes
    • Highest score
    • tags
  • Ubuntu
    • Recentes
    • Highest score
    • tags
  • Unix
    • Recentes
    • tags
  • DBA
    • Recentes
    • tags
  • Computer
    • Recentes
    • tags
  • Coding
    • Recentes
    • tags
Início / unix / Perguntas / 675346
Accepted
Brad Parks
Brad Parks
Asked: 2021-10-30 11:55:05 +0800 CST2021-10-30 11:55:05 +0800 CST 2021-10-30 11:55:05 +0800 CST

bash: atribua a variável usando um subshell, mas saia do script principal se a atribuição falhar

  • 772

Eu gostaria de ter um script bash que me permitisse chamar uma função para ver se alguns dados estão contidos em um arquivo e, se não falhar, o script principal, como o seguinte, que é simplificado para manter isso no ponto.

Isso não funciona (não sai do script principal quando o sub shell falha).

Como posso escrever a require_linefunção para que eu pudesse dizer mais de 20 deles em um arquivo como esse

VALUE1=$(require_line "myKey1")
VALUE2=$(require_line "myKey2")
...

e não exigir um se em torno de cada um?

#!/bin/bash
set -eo pipefail

VALUE=$(require_line "myKey")

require_line(){
  local KEY=$1
  local DATA=$(cat /tmp/myfile)
  local INFO=$(echo "$DATA" | grep "$KEY")

  if [ ! -z "$INFO" ]
  then
    echo "Key not found in $DATA, key: $KEY"
    exit 1;
  fi
  echo "$INFO"
}
bash subshell
  • 2 2 respostas
  • 284 Views

2 respostas

  • Voted
  1. Jim L.
    2021-10-30T13:09:25+08:002021-10-30T13:09:25+08:00

    Vamos reestruturar require_lineum pouco seu script para ajudá-lo.

    Primeiro, podemos nos livrar do inútil cat | grep. Em segundo lugar, podemos usar grepo comportamento intrínseco de 's para indicar o sucesso ou fracasso da busca por KEY, bem como imprimir a chave, se encontrada, para stdout.

    require_line(){
      local KEY="$1"
      local FILE="/tmp/myfile"
    
      if grep "$KEY" "$FILE"
      then
        return 0
      else
        printf 'Key not found in:\n\n"%s"\n\nKey: "%s"\n' "$(cat "$FILE")" "$KEY" >&2
        return 1
      fi
    }
    

    Isso aproveita o comportamento interno do grep. Se a chave for encontrada, grepimprime a linha correspondente e retorna sucesso. Caso contrário, a elseramificação é tomada e uma mensagem é impressa indicando que a chave não foi encontrada. Além disso, no caso grepde falha, a mensagem de erro é impressa para stderrque a mensagem de erro não seja confundida com uma correspondência válida encontrada em $FILE.

    Você pode modificar ainda mais require_linepara aceitar um nome de arquivo como $2 parâmetro alterando a linha para ler local FILE="$2"e, em seguida, passando o nome de arquivo desejado cada vez que invocar require_line.

    Agora, com isso no lugar....

    Você realmente precisa armazenar cada VALUEn para KEYn ou só precisa garantir que todos estejam presentes?

    Agora que você tem uma require_linefunção que retorna claramente um valor de sucesso ou falha, você pode simplesmente ANDjuntar todos os seus testes. Assim que qualquer um deles falhar, o teste geral falhará.

    Supondo que você precise dos valores de correspondência reais, a maneira prolixa de fazer isso seria:

    if value1=$(require_line "key1") &&
       value2=$(require_line "key2") &&
       value3=$(require_line "key3")
    then
       printf "%s\n" "$value1" "$value2" "$value3"
    else
       printf "One or more keys failed.\n" >&2
    fi
    

    Isso será tedioso se você tiver várias chaves para verificar. Usar uma matriz pode ser melhor:

    #!/usr/bin/env bash
    
    require_line(){
      local KEY="$1"
      local FILE="/tmp/myfile" # or perhaps "$2"
    
      if grep "$KEY" "$FILE"
      then
        return 0
      else
        printf 'Key not found in:\n\n"%s"\n\nKey: "%s"\n' "$(cat "$FILE")" "$KEY" >&2
        return 1
      fi
    }
    
    declare keys=("this" "is" "a" "test" "\." "keyN")
    N=${#keys[@]}
    
    declare values=()
    
    j=0
    while [ $j -lt $N ] && values[$j]="$(require_line "${keys[j]}")"
    do
      j=$(($j+1))
    done
    
    if [ $j -lt $N ]
    then
      printf 'error: found only %d keys out of %d:\n' $j $N
      printf '  "%s"\n' "${values[@]}"
    fi
    

    Executando esse código com alguns dados de exemplo:

    $ cat /tmp/myfile
    this is a test.
    $ ./test.sh 
    Key not found in:
    
    "this is a test."
    
    Key: "keyN"
    error: found only 5 keys out of 6:
      "this is a test."
      "this is a test."
      "this is a test."
      "this is a test."
      "this is a test."
      ""
    

    Por fim, se você realmente só precisa verificar se todas as chaves estão presentes sem precisar saber quais são os valores de correspondência, o código orientado a matriz acima pode ser simplificado para simplesmente fazer um loop até que todas as chaves sejam encontradas ou abortar na primeira chave que está faltando.

    • 1
  2. Best Answer
    Totor
    2021-10-30T15:12:27+08:002021-10-30T15:12:27+08:00

    Sair do subshell não sairá do script principal como você o experimentou.

    Penso em 3 (e meia) soluções:

    1. use set -epara que qualquer comando com falha ("não testado") (ou subshell) saia imediatamente do script principal (isso pode ser um exagero ou causar outros problemas),
    2. enviar um sinal da função e capturá-lo com umtrap
    3. use || exit $?depois de cada VALUEn=$(...)assim VALUE=$(require_line "myKey") || exit $?,
    4. combine (3.) com um loop (não tão elegante) usando eval.

    Esse terceiro não exatamente "requer um if em torno de cada um" e ainda seria uma sintaxe bastante compacta IMHO.


    Aliás , essa linha

    echo "Key not found in $DATA, key: $KEY"
    

    ...é realmente inútil se você sair de todo o script logo após isso porque a frase será armazenada na $VALUEnvariável que não será exibida.

    Sugiro imprimir stderrassim:

    echo "my error" 1>&2
    

    Exemplos

    Exemplo para a solução 1

    #!/bin/sh
    set -e
    
    myfunc(){
            echo $1
            if [ "$1" != "OK" ] ; then exit 1 ; fi
    }
    
    VALUE1=$(myfunc "OK") 
    echo $VALUE1
    VALUE2=$(myfunc "NO WAY") 
    echo $VALUE2
    
    echo "main script did not exit"
    
    $ ./test.sh
    OK
    zsh: exit 1     ./test.sh
    

    Mas se eu remover set -edesde o início, recebo:

    $ ./test.sh
    OK
    NO WAY
    main script did not exit
    

    Exemplo para a solução 2

    #!/bin/sh
    
    trap "exit $?" USR1
    
    myfunc(){
            echo $1
            if [ "$1" != "OK" ] ; then kill -USR1 $$ ; fi 
    }
    
    VALUE1=$(myfunc "OK") 
    echo $VALUE1
    VALUE2=$(myfunc "NO WAY")
    echo $VALUE2
    
    echo "main script did not exit"
    
    $ ./test.sh
    OK
    

    Exemplo para a solução 3

    #!/bin/sh
    
    myfunc(){
            echo $1
            if [ "$1" != "OK" ] ; then exit 1 ; fi
    }
    
    VALUE1=$(myfunc "OK") || exit $?
    echo $VALUE1
    VALUE2=$(myfunc "NO WAY") || exit $?
    echo $VALUE2
    
    echo "main script did not exit"
    
    $ ./test.sh
    OK
    zsh: exit 1     ./test.sh
    

    Exemplo para solução 4

    #!/bin/sh
    
    myfunc(){
            echo $1
            if [ "$1" != "OK" ] ; then exit 1 ; fi
    }
    
    I=1
    for key in "OK" "OK" "NO WAY": ; do
            eval "VALUE$I=\$(myfunc \"$key\")" || exit $?
            eval "echo \$VALUE$I"
            I=$(($I+1))
    done
    echo "main script did not exit"
    
    $ ./test.sh
    OK
    OK
    zsh: exit 1     ./test.sh
    
    • 1

relate perguntas

  • exportar variáveis ​​​​env programaticamente, via stdout do comando [duplicado]

  • Problema estranho ao passar variáveis ​​do arquivo de texto

  • Enquanto a linha lê mantendo os espaços de escape?

  • ordem de substituição de processos `te` e `bash`

  • Execute um script muito lento até que seja bem-sucedido

Sidebar

Stats

  • Perguntas 205573
  • respostas 270741
  • best respostas 135370
  • utilizador 68524
  • Highest score
  • respostas
  • Marko Smith

    Possível firmware ausente /lib/firmware/i915/* para o módulo i915

    • 3 respostas
  • Marko Smith

    Falha ao buscar o repositório de backports jessie

    • 4 respostas
  • Marko Smith

    Como exportar uma chave privada GPG e uma chave pública para um arquivo

    • 4 respostas
  • Marko Smith

    Como podemos executar um comando armazenado em uma variável?

    • 5 respostas
  • Marko Smith

    Como configurar o systemd-resolved e o systemd-networkd para usar o servidor DNS local para resolver domínios locais e o servidor DNS remoto para domínios remotos?

    • 3 respostas
  • Marko Smith

    apt-get update error no Kali Linux após a atualização do dist [duplicado]

    • 2 respostas
  • Marko Smith

    Como ver as últimas linhas x do log de serviço systemctl

    • 5 respostas
  • Marko Smith

    Nano - pule para o final do arquivo

    • 8 respostas
  • Marko Smith

    erro grub: você precisa carregar o kernel primeiro

    • 4 respostas
  • Marko Smith

    Como baixar o pacote não instalá-lo com o comando apt-get?

    • 7 respostas
  • Martin Hope
    user12345 Falha ao buscar o repositório de backports jessie 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl Por que a maioria dos exemplos do systemd contém WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky Como exportar uma chave privada GPG e uma chave pública para um arquivo 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Evan Carroll status systemctl mostra: "Estado: degradado" 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim Como podemos executar um comando armazenado em uma variável? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S Por que /dev/null é um arquivo? Por que sua função não é implementada como um programa simples? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 Como ver as últimas linhas x do log de serviço systemctl 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - pule para o final do arquivo 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla Por que verdadeiro e falso são tão grandes? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis Substitua a string em um arquivo de texto enorme (70 GB), uma linha 2017-12-30 06:58:33 +0800 CST

Hot tag

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

  • Início
  • Perguntas
    • Recentes
    • Highest score
  • tag
  • help

Footer

AskOverflow.Dev

About Us

  • About Us
  • Contact Us

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve