chamando Bash Function da definição de alias

5

Estou usando o Xilinx ISE, um programa que requer a criação de um script de fornecedor para configurar o caminho para todas as suas ferramentas. Como este script quebra algumas funcionalidades do meu shell, eu não quero usá-lo no início do shell, mas apenas quando eu precisar dele.

Para isso, eu configurei um alias para source /long/path/to/script . Por conveniência, eu gostaria que as ferramentas fornecidas pelo programa funcionassem sempre sem o script de configuração. A idéia é ter apelidos como o seguinte para cada um dos n programas (por exemplo, vamos supor que eles são denominados p1, ..., pn)

alias p1='source /long/path/;unalias p1;...;unalias pn;p1'

Como existem muitos programas, quero facilitar a atualização da cadeia de unalias. A ideia é ter alias iseRemoveSetup='unalias p1;...;pn' . Para facilitar a construção dos aliases dos programas e iseRemoveSetup , defini algumas funções. No entanto, se eu chamar iseAddToRemoveCall do comando alias (aqui para analyzer ), ele não terá nenhum efeito. Se eu chamá-lo diretamente no shell, ele funciona bem.

Abaixo está a parte relevante do meu .bashrc :

alias iseRemoveSetup=''
function iseAddToRemove() {
  alias iseRemoveSetup='(alias iseRemoveSetup | cut -f2 -d "'")'"unalias $1;"
}
function iseAddToRemoveCall() {
  iseAddToRemove $1
  echo "iseRemoveSetup;$1"
}

alias setupise='source /home/ted/Xilinx/tools/14.7/ISE_DS/settings64.sh'
alias analyzer='setupise;''iseAddToRemoveCall "analyzer"'

Estou ciente de que seria uma ideia ter uma função como:

function iseAlias() {
  alias $1='setupise;''iseAddToRemoveCall $1'
}

No entanto, quero que o problema descrito acima seja resolvido primeiro. O problema fica visível quando executo type iseRemoveSetup , ele gera:

iseRemoveSetup is aliased to ''
    
por ted 09.12.2014 / 13:58

1 resposta

4

O problema é que sua definição de alias (a de iseRemoveSetup ) está em uma subshell:

alias analyzer = 'setupise;' 'analisador iseAddToRemoveCall "'

Para iseAddToRemoveCall "analyzer" , um subshell é iniciado e a definição do alias afeta apenas essa subshell que desaparece assim que essa linha é concluída.

Isso pode ser resolvido mudando

alias analyzer='setupise;''iseAddToRemoveCall "analyzer"'

para

alias analyzer='setupise;$(iseAddToRemoveCall analyzer)'

e substituindo echo "iseRemoveSetup;$1" (em iseAddToRemoveCall ) por echo "unalias $1;$1" . Assim, o unalias se tornaria parte da expansão do alias e seria executado no shell correto.

alternativa

Tudo isso parece muito estranho para mim. Não faria mais sentido iniciar um subshell (digite bash ), executar a configuração, executar os comandos e deixar o subshell ( ^D ) quando terminar?

observações adicionais

man 1 bash :

The first word of the replacement text is tested for aliases, but a word that is identical to an alias being expanded is not expanded a second time. This means that one may alias ls to ls -F, for instance, and bash does not try to recursively expand the replacement text.

Portanto, você não precisa de unalias p1 em sua definição de alias antes de chamar pi dela.

Pode ser melhor usar uma matriz (associativa) com os comandos que você deseja unalias e apenas trabalhar com a matriz.

Isso também é importante porque você faz exatamente o que não deve fazer:

Bash always reads at least one complete line of input before executing any of the commands on that line. Aliases are expanded when a command is read, not when it is executed. Therefore, an alias definition appearing on the same line as another command does not take effect until the next line of input is read. The commands following the alias definition on that line are not affected by the new alias. This behavior is also an issue when functions are executed. Aliases are expanded when a function definition is read, not when the function is executed, because a function definition is itself a compound command. As a consequence, aliases defined in a function are not available until after that function is executed. To be safe, always put alias definitions on a separate line, and do not use alias in compound commands.

    
por 09.12.2014 / 14:24

Tags