Eu tenho um arquivo que é originário .bashrc
e .zshrc
que contém algumas funções. Anteriormente, eram dois arquivos separados (para cada shell), mas agora estou usando um único arquivo, pois os arquivos eram quase idênticos e é mais fácil para mim gerenciar um único arquivo de funções.
Existe uma função que é diferente para os dois shells, então pensei em usar uma declaração condicional para que a correta seja usada:
[[ $BASH_VERSION ]] && shell=bash
[[ $ZSH_VERSION ]] && shell=zsh
# Search man page for string
if [[ $shell == bash ]]; then
mans() {
local q="\'"
local q_pattern="'${1//$q/$q\\$q$q}'"
local MANPAGER="less -+MFX -p $q_pattern"
man "$2"
}
elif [[ $shell == zsh ]]; then
mans() MANPAGER="less -+MFX -p ${(q+)1}" man $2
fi
Isso parece funcionar bem no zsh, mas ao mudar para o bash, recebo um erro da linha zsh, mesmo que não deva ser usado no bash. Existe alguma maneira de impedir que ele analise essa linha?
Você pode definir alguns aliases
ifbash
,ifzsh
,endif
como::||:<<"MARKER"
...MARKER
é uma maneira idiomática de comentar uma seção de código. É como:<<"MARKER"
executar o:
noop embutido com um here-doc no stdin, mas com o extra:||
, o segundo:
não é executado e o here-document nem mesmo foi criado, apenas analisado.Tenha em atenção que o
endif
não deve ser recuado nem seguido de nada; deve compor toda a linha.ifbash
/ifzsh
s não podem ser aninhados (não que isso faça sentido aqui).BTW,
man
parece estar fazendo a análise de citação naMANPAGER
variável de ambiente por si só (em oposição a invocar um shell, quanto mais seu$SHELL
) e não parece suportar a$'...'
forma de aspas, portanto, usando oq+
sinalizador de expansão de parâmetro que pode usar esse formulário de aspas para algumas strings não está correto. Useqq
, em vez disso, que usa aspas simples (e\'
fora das aspas para escapar das aspas simples¹) como na sua versão do bash.¹ a menos que a
rcquotes
opção esteja habilitada, caso em que''
à larc
é usado. Adicionaremulate -L zsh
no início de sua função, para ter um conjunto de opções locais sãs nessa função, não ajudaria, pois a opção precisa ser desativada no momento em que a função é definida, não executada.O Bash ainda precisa analisar o arquivo inteiro, mesmo que não execute todas as linhas. Normalmente, você não pode ocultar erros de sintaxe por trás de uma condicional, apenas erros de tempo de execução.
Em seu lugar, eu voltaria a usar arquivos separados
.zshrc
e.bashrc
. Em seguida, colocaria as funções comuns em um terceiro arquivo (garantindo que esse arquivo usasse apenas sintaxe compatível com ambos) e o originaria de ambos.zshrc
e.bashrc
.Na minha opinião, este é o melhor dos mundos:
.zshrc
onde nunca seriam lidas pelo bash. Da mesma forma, coisas somente bash vivem em arquivos.bashrc
.Qualquer shell tem que analisar o código dentro do condicional para encontrar a parte onde o condicional termina, mesmo que por nenhum outro motivo.
Além de colocar o código específico do shell em outro arquivo ou em uma string para
eval
, você também pode usar um here-doc para ocultá-lo esource
o here-doc diretamente. Praticamente o mesmo que usareval
, mas um here-doc tem a vantagem de que o código interno pode conter aspas simples e duplas sem problemas irritantes de escape.Por exemplo, este script diz
hello, 'bar', I'm zsh
se executado com zsh e gera um erro no Bash:Não tenho certeza se
source /dev/stdin
funciona com todos os shells e sistemas. (E sim,source
é um nome não padrão para o.
builtin, mas deve funcionar em Bash e zsh, e de qualquer forma o comando só precisa existir no shell que roda dentro do condicional.)Outros já apontaram porque o bash está analisando sintaticamente o conteúdo do
$shell == zsh
condicional.No zsh você pode usar
considerando que isso não é uma sintaxe bash legal (uma função precisaria de {}).
No entanto, no zsh, você também pode usar a sintaxe POSIX para funções, o que manterá os dois shells felizes:
Além disso, eu geralmente os alteraria
[[
para uma sintaxe única[
, para que o arquivo não fosse interrompido se executado em outros shells (comodash
) que não suportam[[
:O método sugerido por @ilkkachu funcionou: