Como usar o comando nohup no Linux
Publicados: 2022-06-25 O comando nohup
do Linux permite que processos importantes continuem em execução mesmo quando a janela do terminal que os iniciou está fechada. Mostramos a você como usar esse comando venerável no Linux atual.
HUP e SIGHUP
O Unix, o ancestral do Linux, foi criado antes do PC ser inventado. Os computadores eram equipamentos grandes e caros. As pessoas interagiam com eles por meio de linhas seriais localmente dentro do mesmo prédio ou remotamente por meio de conexões de modem lentas. Originalmente, eles digitavam suas instruções em teleimpressoras que foram gradualmente substituídas por terminais burros.
Eles foram chamados de burros porque o poder de processamento estava no computador ao qual você estava conectado, não no terminal em que você estava digitando. Os programas estavam sendo executados no computador — onde quer que estivesse localizado — e não no dispositivo em sua mesa.
Se acontecer algo que interrompeu a conexão entre o seu terminal e o computador, o computador detectou a queda de linha e enviou um HUP
ou sinal de desligamento para os programas que você estava executando. Os programas cessaram a execução quando receberam o sinal.
Essa funcionalidade vive no Linux hoje. No seu PC, uma janela de terminal é uma emulação de um terminal físico. Se você tiver processos em execução que foram iniciados a partir dessa janela do terminal e fechar essa janela, o sinal SIGHUP
será enviado aos programas para que eles sejam informados do HUP
e saibam que devem terminar.
Há um efeito cascata que ocorre. Se os processos iniciaram algum processo filho, o SIGHUP também será passado na linha para eles, para que eles saibam que devem terminar.
O comando nohup
inicia processos filho, mas se recusa a passar sinais SIGHUP
para eles. Isso pode parecer um problema, mas na verdade é uma função útil.
O comando nohup
Se você deseja que um processo continue mesmo que a janela do terminal de onde foi iniciado esteja fechada, você precisa de uma maneira de interceptar o SIGHUP
para que o programa nunca o receba. (Na verdade, a janela do terminal não inicia os processos, eles são lançados pela sessão do shell dentro da janela do terminal.) A solução simples e elegante para esse problema é colocar outro processo entre a sessão do shell e o programa, e fazer isso programa de camada intermediária nunca transmite o sinal SIGHUP
.
Isso é o que o nohup
faz. Ele inicia programas para você para que eles sejam um processo filho de nohup
, não um processo filho do shell. Como eles não são um processo filho do shell, eles não receberão diretamente um SIGHUP
do shell. E se o nohup
não passar o SIGHUP
para seus filhos, o programa não receberá o SIGHUP
.
Isso é útil quando, por exemplo, você tem um processo de longa execução que precisa ser executado até a conclusão. Se você fechar acidentalmente a janela do terminal e seu shell, também encerrará o processo. Usar nohup
para iniciar o processo isola o processo do sinal nohup
. Se você estiver trabalhando remotamente em um computador por SSH e não quiser que um processo confidencial seja encerrado se a conexão remota falhar, inicie o processo no computador remoto com nohup
.
Usando nohup
Criamos um programa que não faz nada de útil, mas vai rodar e rodar até ser finalizado. Ele imprime a hora na janela do terminal a cada três segundos. É chamado de long-proc
para “processo longo”.
./long-proc
Se este fosse um programa que fizesse algo útil e quiséssemos que ele continuasse rodando mesmo que a janela do terminal e o shell estivessem fechados, nós o iniciaríamos com nohup
.
nohup ./long-proc
O processo é desacoplado de stdin
e stdout
para que não possa receber nenhuma entrada nem escrever na janela do terminal. Além disso, como ainda está em execução, você não retornará ao prompt de comando. Tudo o que o nohup
faz é tornar o processo impermeável ao fechamento do terminal. Não transforma o processo em uma tarefa em segundo plano.
Agora você precisa reiniciar apenas para encerrar o processo? Não. Para interromper um processo nohup
que você não iniciou como um processo em segundo plano, pressione a combinação de teclas Ctrl+C.
A saída do programa foi capturada para nós em um arquivo chamado “nohup.out”. Podemos revisá-lo com menos.
menos nohup.out
Qualquer coisa que normalmente seria enviada para a janela do terminal é capturada no arquivo. As execuções subsequentes do nohup
serão anexadas ao arquivo “nohup.out” existente.
Uma maneira mais útil de executar o processo é iniciá-lo com nohup
para que ele resista ao fechamento da janela do terminal e torná-lo uma tarefa em segundo plano ao mesmo tempo. Para fazer isso, adicionamos um e comercial “ &
” ao final da linha de comando.
nohup ./long-proc &
Você precisará pressionar “Enter” mais uma vez para retornar ao prompt de comando. Somos informados de que o número da tarefa do processo é 1—o número entre colchetes “ []
”— e que o ID do processo é 13115.
Podemos usar qualquer um deles para encerrar o processo. “Ctrl+C” não funcionará agora porque o programa não tem nenhuma associação com a janela do terminal ou o shell.
Se você esquecer o número do trabalho, poderá usar o comando jobs
para listar as tarefas em segundo plano que foram iniciadas nessa janela do terminal.
empregos
Para matar nossa tarefa, podemos usar o comando kill
e o número do trabalho, precedido por um sinal de porcentagem “ %
”, assim:
matar % 1
Se você fechou a janela do terminal, precisará encontrar o ID do processo e usá-lo com o comando kill
. O comando pgrep
encontrará o ID do processo para os processos que correspondem à pista de pesquisa fornecida. Procuraremos o nome do processo.
pgrep long-proc
Agora podemos usar o ID do processo para encerrar o processo.
matar 13115
Na próxima vez que você pressionar “Enter”, você será informado de que o processo foi encerrado.
Agora vamos ver o que não encerra o processo. Vamos relançá-lo e, em seguida, fechar a janela do terminal.
nohup ./long-proc
Se abrirmos uma nova janela de terminal e procurarmos nosso processo com pgrep
, veremos que ele ainda está em execução. Fechar a janela do terminal que iniciou o processo não teve efeito.
pgrep long-proc
É possível passar vários comandos para nohup
, mas geralmente é melhor iniciá-los separadamente. Isso torna mais fácil manipulá-los como trabalhos em segundo plano. Os comandos não serão executados ao mesmo tempo, eles serão executados um após o outro. A execução não é concorrente, é sequencial. Para executá-los simultaneamente, você precisa iniciá-los separadamente.
Dito isto, para iniciar vários processos ao mesmo tempo, use nohup
para iniciar um shell Bash e use a opção -c
(comandos) com a string de comandos. Use aspas simples “ '
” para agrupar a lista de comandos e os e comerciais duplos “ &&
” para separar os comandos.
nohup bash -c 'ls /bin && ls /sbin'
Se você usar less
para examinar o arquivo “nohup.out”, verá a saída do primeiro processo e, em seguida, a saída do segundo processo.
menos nohup.out
A saída de ambos os comandos foi capturada no arquivo “nohup.out”. Não está entrelaçado, a saída do segundo processo só começa quando o primeiro processo termina.
Se você quiser usar um arquivo próprio em vez de “nohup.out”, você pode redirecionar o comando para o arquivo de sua escolha.
nohup bash -c 'ls /bin && ls /sbin' > meuarquivo.txt
Observe que a mensagem não diz mais “anexando saída para nohupo.out”, ela diz “redirecionando stderr para stdout” e estamos redirecionando stdout para nosso arquivo “myfile.txt”.
Podemos olhar dentro do arquivo “myfile.txt” com menos.
menos meuarquivo.txt
Como antes, ele contém a saída de ambos os comandos.
É engraçado como a história de um utilitário às vezes pode fazer parecer que não tem relevância para os tempos modernos. O comando nohup
é um desses. Algo que foi criado para lidar com desconexões em linhas seriais ainda é útil para os usuários de Linux de hoje em máquinas incrivelmente poderosas.
RELACIONADO: 37 comandos importantes do Linux que você deve conhecer