Eu sou capaz de executar este comando com sucesso:
tail -f my_file.txt | grep foo
Ele mostra apenas as linhas com a string foo
e continua mostrando-as.
Mas quando eu executo este comando:
tail -f my_file.txt | grep foo | grep bar
Ele não mostra nenhuma linha, embora existam linhas que incluam foo
e bar
.
Sei que existe uma solução para usar vários padrões em uma única grep
chamada, mas quero saber por que essa linha falhou.
Isso ocorre porque o comportamento padrão da biblioteca de tempo de execução C é armazenar gravações em stdout até que um bloco completo de dados seja gravado (alguns kilobytes, geralmente), a menos que stdout esteja conectado a um terminal.
Você obterá a saída assim que o grep do meio imprimir um bloco completo, mas precisará aguardar novamente o preenchimento do próximo bloco e assim por diante. É uma otimização para taxa de transferência e funciona muito melhor quando o comando esquerdo apenas executa alguma tarefa e termina, em vez de esperar por algo.
GNU grep tem a
--line-buffered
opção de desligar esse buffer, então isso deve funcionar melhor:O último
grep
é impresso no terminal, portanto, é armazenado em buffer de linha por padrão e não precisa de uma opção.Consulte Desativar o armazenamento em buffer no canal para obter soluções genéricas para o problema de armazenamento em buffer.
Neste caso particular de dois greps, você poderia usar, por exemplo, um único AWK, como Stéphane Chazelas mencionou em um comentário:
(Aliás, você também pode fazer coisas como
awk '/foo/ && !/bar/'
, pegar linhas comfoo
mas nãobar
.)Fazer o mesmo em grep seria mais difícil, pois
grep -e foo -e bar
corresponde a qualquer linha que contenha oufoo
.bar
Você precisaria de algo comoem vez de.
De um ponto de vista booleano, parece que você está esperando foo OR bar com seu grep, mas do jeito que você fez, você deve esperar foo AND bar -- As únicas linhas que serão grepadas para bar são aquelas que passaram pelo foo grep em primeiro lugar.
Se você quiser que foo/bar apareça no stdout, você precisa usar:
Você pode adicionar quantas palavras-chave "bonito|muito|como|estes", só não se esqueça das aspas. (Ou use vários\|sem aspas\|termos )
Em uma segunda observação: como é um tail -f (follow), também pode acontecer de your_file.txt não ser anexado com uma linha contendo as duas palavras-chave naquele momento muito específico, geralmente é melhor cortar uma parte desse arquivo que é conhecido por ter o que você espera:
A partir daí, você pode tentar qualquer grep que desejar com um texto conhecido. (Acho que as últimas 500 linhas são suficientes, ajuste conforme necessário.)