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 / computer / Perguntas / 1583613
Accepted
slhck
slhck
Asked: 2020-09-08 04:27:58 +0800 CST2020-09-08 04:27:58 +0800 CST 2020-09-08 04:27:58 +0800 CST

Construindo um comando Bash com argumentos contendo espaços, pipes e substituição de processo

  • 772

Estou escrevendo um pequeno script utilitário que pode ser parametrizado para complementar timesua execução e também gravar seu stdout em outro arquivo.

Vamos supor que o script básico seja simplesmente:

ls "$@"

Então, o seguinte seria uma implementação ingênua, aqui usando catcomo um manequim para manipulação mais complexa da saída:

if [[ "$TIMEIT" -eq 1 ]]; then
  if [[ "$LOGIT" -eq 1 ]]; then
    time ls "$@" | tee >(cat)
  else
    time ls "$@"
  fi
else
  if [[ "$LOGIT" -eq 1 ]]; then
    ls "$@" | tee >(cat)
  else
    ls "$@"
  fi
fi

Você pode ver qual é o problema com isso, especialmente se novos parâmetros devem ser adicionados.

Uma reescrita que seria mais flexível é:


cmd="ls $@"

if [[ "$TIMEIT" -eq 1 ]]; then
  cmd="time $cmd"
fi

if [[ "$LOGIT" -eq 1 ]]; then
  cmd="$cmd | tee >(cat)"
fi

eval $cmd 

Isso funciona, mas quebra quando você o chama como:

./script.sh "file 1" file2

Porque os nomes dos arquivos não serão tratados corretamente.

Quando aplico esta solução , posso tokenizar adequadamente os argumentos ao preceder o comando com time:

function token_quote {
  local quoted=()
  for token; do
    quoted+=( "$(printf '%q' "$token")" )
  done
  printf '%s\n' "${quoted[*]}"
}

cmd=(ls "$@")

if [[ "$TIMEIT" -eq 1 ]]; then
  cmd=(time "${cmd[@]}")
fi

eval "$(token_quote "${cmd[@]}")"

Mas o mesmo método para manusear o cachimbo não funciona. Quando eu o adapto usando esta solução :

function eval_args {
  local quoted=''
  while (( $# )); do
    if [[ $1 = '|' ]]; then
      quoted+="| "
    else
      printf -v quoted '%s%q ' "$quoted" "$1"
    fi
    shift
  done
  eval "$quoted"
}

cmd=(ls "$@")

# prepend to array
if [[ "$TIMEIT" -eq 1 ]]; then
  cmd=(time "${cmd[@]}")
fi

# append to array?
if [[ "$LOGIT" -eq 1 ]]; then
  cmd=("${cmd[@]}" "|" "tee" ">(cat)")
fi

eval_args "${cmd[@]}"

Ele cria um arquivo >(cat)… então estou entrando na toca do coelho aqui, e parece que este não é o caminho a seguir.

O que posso fazer para construir o comando e lidar com espaços/caracteres especiais em nomes de arquivos, incluindo pipes/substituição de processo?

linux bash
  • 1 1 respostas
  • 701 Views

1 respostas

  • Voted
  1. Best Answer
    user1686
    2020-09-08T06:28:28+08:002020-09-08T06:28:28+08:00

    avaliação

    Seu último método pode funcionar, contanto que você o ignore para suas adições personalizadas e o use apenas para gerar o comando original. Ou seja, altere eval_args para criar apenas a string de comando:

    build_args() {
        local arg quoted=""
        for arg; do
            printf -v quoted '%s%q ' "$quoted" "$1"
        done
        echo "$quoted"
    }
    
    cmd=$(build_args ls "$@")
    if (( TIMEIT )); then
        cmd="time $cmd"
    fi
    if (( LOGIT )); then
        cmd="$cmd | tee >(foo)"
    fi
    
    eval "$cmd"
    

    Na verdade, o loop é desnecessário, pois o printf do bash repetirá automaticamente a mesma string de formato com argumentos extras:

    argv=(ls -la "$@")
    cmd=$(printf '%q ' "${argv[@]}")
    eval "time $cmd | tee >(foo)"
    

    O Bash 5.1 possui um ${foo@Q}modificador que é uma alternativa mais conveniente:

    argv=(ls -la "$@")
    cmd=${argv[*]@Q}
    eval "time $cmd | tee >(foo)"
    

    Alternativa para avaliação

    Digamos que seu script tenha seu código principal em uma função:

    run() { ls "$@"; }
    

    Você pode definir condicionalmente uma função wrapper para ele:

    if (( LOGIT )); then
        run_logged() { run "$@" | tee >(cat); }
    else
        run_logged() { run "$@"; }
    fi
    
    if (( TIMEIT )); then
        run_timed() { time run_logged "$@"; }
    else
        run_timed() { run_logged "$@"; }
    fi
    
    run_timed "$@"
    

    (A própria definição da função pode ser gerada em uma string e avaliada.)

    Reexec

    Em alguns casos, pode fazer sentido reexecutar todo o script no novo ambiente:

    if (( UID > 0 )); then
        exec sudo "$0" "$@" || exit
    fi
    
    if (( ! _LOGIT_ACTIVE )); then
        exec _LOGIT_ACTIVE=1 "$0" "$@" | tee >(foo) || exit
    fi
    
    ls "$@"
    

    Diversos

    Quanto aos redirecionamentos especificamente, você pode codificar o uso teee apenas trocar o nome do arquivo (não muito eficiente, pois ainda faz com que o tee seja executado e duplica as gravações, mas ... faz o trabalho):

    if (( LOGIT )); then
        out=/var/log/foo
    else
        out=/dev/null
    fi
    
    ls "$@" | tee $out
    

    O mesmo para redirecionamentos diretos para um arquivo:

    if (( LOGIT )); then
        exec {fd}>/var/log/foo
    else
        fd=1     # stdout
    fi
    
    ls "$@" >&$fd
    
    if (( fd != 1 )); then
        exec {fd}>&-    # close
    fi
    

    Como alternativa, você pode canalizar uma condição ou uma função que envolva a condição (também não é 100% eficiente devido a obrigatório cat):

    ls "$@" | if (( LOGIT )); then tee >(whatever); else cat; fi
    
    • 1

relate perguntas

  • Notificar-enviar notificações aparecendo na janela

  • execute o contêiner do docker como root

  • Como ativar o sensor de impressão digital no domínio e no diretório ativo do Linux

  • Como alterar permanentemente Ctrl + C para Ctrl + K no CentOS 7?

  • como abrir um arquivo de escritório do WSL

Sidebar

Stats

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

    Como posso reduzir o consumo do processo `vmmem`?

    • 11 respostas
  • Marko Smith

    Baixar vídeo do Microsoft Stream

    • 4 respostas
  • Marko Smith

    O Google Chrome DevTools falhou ao analisar o SourceMap: chrome-extension

    • 6 respostas
  • Marko Smith

    O visualizador de fotos do Windows não pode ser executado porque não há memória suficiente?

    • 5 respostas
  • Marko Smith

    Como faço para ativar o WindowsXP agora que o suporte acabou?

    • 6 respostas
  • Marko Smith

    Área de trabalho remota congelando intermitentemente

    • 7 respostas
  • Marko Smith

    O que significa ter uma máscara de sub-rede /32?

    • 6 respostas
  • Marko Smith

    Ponteiro do mouse movendo-se nas teclas de seta pressionadas no Windows?

    • 1 respostas
  • Marko Smith

    O VirtualBox falha ao iniciar com VERR_NEM_VM_CREATE_FAILED

    • 8 respostas
  • Marko Smith

    Os aplicativos não aparecem nas configurações de privacidade da câmera e do microfone no MacBook

    • 5 respostas
  • Martin Hope
    CiaranWelsh Como posso reduzir o consumo do processo `vmmem`? 2020-06-10 02:06:58 +0800 CST
  • Martin Hope
    Jim Pesquisa do Windows 10 não está carregando, mostrando janela em branco 2020-02-06 03:28:26 +0800 CST
  • Martin Hope
    v15 Por que uma conexão de Internet gigabit/s via cabo (coaxial) não oferece velocidades simétricas como fibra? 2020-01-25 08:53:31 +0800 CST
  • Martin Hope
    fixer1234 O "HTTPS Everywhere" ainda é relevante? 2019-10-27 18:06:25 +0800 CST
  • Martin Hope
    andre_ss6 Área de trabalho remota congelando intermitentemente 2019-09-11 12:56:40 +0800 CST
  • Martin Hope
    Riley Carney Por que colocar um ponto após o URL remove as informações de login? 2019-08-06 10:59:24 +0800 CST
  • Martin Hope
    zdimension Ponteiro do mouse movendo-se nas teclas de seta pressionadas no Windows? 2019-08-04 06:39:57 +0800 CST
  • Martin Hope
    jonsca Todos os meus complementos do Firefox foram desativados repentinamente, como posso reativá-los? 2019-05-04 17:58:52 +0800 CST
  • Martin Hope
    MCK É possível criar um código QR usando texto? 2019-04-02 06:32:14 +0800 CST
  • Martin Hope
    SoniEx2 Altere o nome da ramificação padrão do git init 2019-04-01 06:16:56 +0800 CST

Hot tag

windows-10 linux windows microsoft-excel networking ubuntu worksheet-function bash command-line hard-drive

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