Estou tentando usar echo
dentro de uma regra Puppet para adicionar uma linha a .bashrc
, mas não consigo acertar a citação.
'/usr/bin/echo -E PS1=\"[\t--------------------------------------------------------\n-\u@\h:\W]\$\" >> /home/unu/.bashrc'
Essa coisa me dá o seguinte resultado:
PS1="[t--------------------------------------------------------n-u@h:W]$"
Outra tentativa:
'/usr/bin/echo -E PS1="[\t--------------------------------------------------------\n-\u@\h:\W]\$" >> /home/unu/.bashrc'
Essa coisa me dá o seguinte:
PS1=[\t--------------------------------------------------------\n-\u@\h:\W]$
Outro:
'/usr/bin/echo -e PS1="[\\t--------------------------------------------------------\\n-\\u@\\h:\\W]\\$" >> /home/unu/.bashrc'
Este está me dando isso:
PS1=[ --------------------------------------------------------
-\u@\h:\W]$
Não consigo encontrar uma maneira de fazer isso sem ter \
ou "
interpretado de alguma forma. Como devo fazer isso?
Pensei em usar mais aspas, mas isso causa um erro de sintaxe do Puppet:
"/usr/bin/echo -e 'PS1="[\\t--------------------------------------------------------\\n-\\u@\\h:\\W]\\$"' >> /home/unu/.bashrc"
Obtém este resultado:
Error: Could not retrieve catalog from remote server: Error 500 on SERVER: Server Error: Syntax error at '' (file: /etc/puppetlabs/code/environments/production/modules/profile/manifests/ps1.pp, line: 3, column: 38) on node centoslave1
Este é o código inteiro:
class profile::ps1 {
exec { 'myps1':
command => "/usr/bin/echo -e 'PS1="[\\t--------------------------------------------------------\\n-\\u@\\h:\\W]\\$"' >> /home/unu/.bashrc"
}
}
O que está acontecendo
Entender onde isso está dando errado é a chave para acertar. O lado direito do elemento, após o
Em seguida, você o segue com rabiscos que não são sintaticamente corretos para um manifesto: e outra string:=>
, deve ser uma string. Strings têm várias formas no Puppet. Uma string entre aspas duplas começa com aspas duplas e termina no segundo conjunto. Então sua corda éFazendo isso corretamente
Construa a configuração correta de baixo para cima. Comece com o que você deseja que o comando produza nesse arquivo. A saída é um comando shell que define a
Esse comando é, usando aspas simples (preferível a aspas duplas aqui porque não requer escape para essesPS1
variável shell para:\
e$
caracteres) para impedir que o shell desbloqueie as sequências de escape: Você pode produzir tal comando comecho
invocado por um shell (um ponto ao qual retornaremos), mas você precisa garantir, com mais escape, que as aspas simples alcancem oecho
comando:Isso no entanto tem problemas com o fato de
echo
não ter um comportamento consistente entre plataformas e shells, e (ironicamente) sua possível conversão de sequências de escape não é realmente o que você deseja. De fato, apesar da tag em sua pergunta, o shell que executa o comando não precisa ser o shell Bourne Again, dependendo do sistema operacional (que você não especificou). Pode ser o shell Debian Almquist, por exemplo. Para melhores resultados, useprintf
:Observe que a suposição foi até este ponto de que é uma linha de comando fornecida a um shell para execução. O escape e a citação dos argumentos passados para
echo
eprintf
, para que eles acabem com o conteúdo correto, foram de acordo com as regras do shell. Então tome nota do comentário da pergunta que fala sobre provedores.Agora você precisa codificar isso em uma string Puppet. Strings com aspas simples são novamente a melhor escolha aqui, pois você só precisa escapar dos caracteres
'
e\
(o que não é exatamente o mesmo que as regras do shell para strings com aspas simples, observe):Leitura adicional
Existem (pelo menos) dois níveis de coisas consumindo seus personagens:
echo
em siSe você executar:
você obtém a saída
porque o shell comeu as aspas duplas em barras invertidas, não echo. O shell os tomou como sintaxe de shell para significar passar
hello, "world"
para echo como um único argumento. Qual eco então impresso, mais uma nova linha.echo -e
especialmente também interpreta sequências de barras invertidas (dependendo da implementação); a solução simples para esse problema é usarprintf '%s\n' WHATEVER
em vez deecho WHATEVER
.Na maioria das vezes, você pode proteger algo do shell com aspas simples (desde que não contenha aspas simples). Então, você pode querer:
que dá:
Se você tiver que sobreviver a várias camadas de escape de shell, poderá achar
printf '%q\n' WHATEVER
útil, que o imprime em um formato para sobreviver ao shell (mas isso pode ser um recurso somente do Bash, não verifiquei).BTW: Se você precisar proteger uma aspa simples dentro de uma string com aspas simples, você deve fazer algo como
'\''
:que se você olhar de soslaio com cuidado é na verdade
'i'
+\'
+'ll'
.PS:
PS1
passa por outra expansão (veja a documentação do Bash); você sabe sobre aqueles como\u
obviamente. Mas as variáveis também são expandidas - então, se você tivesse algo após o$
no final, precisaria escapar (com três barras invertidas se estiver dentro de uma string com aspas duplas):PS1="foo\\\$PATH % "
Usar
printf
:Você poderia simplesmente usar
cat
:O
ps1-file
pode ser criado manualmente (usando seu editor preferido) para evitar problemas de escape embash
/echo
/printf
. Para que esse arquivo alcance o nó, você precisará adicionar uma seção de arquivo .Isso significa colocar o arquivo original em
modulepath
. Como seu conteúdo é uma pequena quantidade de texto, você pode especificá-lo comcontent =>
, mas isso teria seus próprios problemas de citação/escape.