Get*_*ree 283 linux ssh process background-process
我正在通过SSH(Putty)在linux机器上工作.我需要让一个进程在夜间运行,所以我想我可以通过在后台启动进程(在命令末尾有一个&符号)并将stdout重定向到文件来做到这一点.令我惊讶的是,这不起作用.一旦我关闭Putty窗口,该过程就会停止.
我怎样才能防止这种情况发生?
Jes*_*erE 297
查看" nohup "计划.
gpo*_*ojd 164
我建议使用GNU Screen.它允许您在所有进程继续运行时断开与服务器的连接.在我知道它存在之前,我不知道我是如何在没有它的情况下生活的.
Rob*_*ble 79
当会话关闭时,进程收到SIGHUP信号,该信号显然没有捕获.您可以nohup
在启动进程后使用该命令,也可以在启动进程后使用bash内置命令disown -h
来防止这种情况发生:
> help disown
disown: disown [-h] [-ar] [jobspec ...]
By default, removes each JOBSPEC argument from the table of active jobs.
If the -h option is given, the job is not removed from the table, but is
marked so that SIGHUP is not sent to the job if the shell receives a
SIGHUP. The -a option, when JOBSPEC is not supplied, means to remove all
jobs from the job table; the -r option means to remove only running jobs.
Run Code Online (Sandbox Code Playgroud)
ant*_*ger 40
守护进程?nohup的?屏幕?(tmux ftw,屏幕是垃圾;-)
只做其他应用程序从一开始就做的事情 - 双叉.
# ((exec sleep 30)&)
# grep PPid /proc/`pgrep sleep`/status
PPid: 1
# jobs
# disown
bash: disown: current: no such job
Run Code Online (Sandbox Code Playgroud)
砰! 完成:-)我在所有类型的应用程序和许多旧机器上使用过无数次.您可以结合重定向和诸如此类的东西来打开您和流程之间的私人渠道.
创建为coproc.sh:
#!/bin/bash
IFS=
run_in_coproc () {
echo "coproc[$1] -> main"
read -r; echo $REPLY
}
# dynamic-coprocess-generator. nice.
_coproc () {
local i o e n=${1//[^A-Za-z0-9_]}; shift
exec {i}<> <(:) {o}<> >(:) {e}<> >(:)
. /dev/stdin <<COPROC "${@}"
(("\$@")&) <&$i >&$o 2>&$e
$n=( $o $i $e )
COPROC
}
# pi-rads-of-awesome?
for x in {0..5}; do
_coproc COPROC$x run_in_coproc $x
declare -p COPROC$x
done
for x in COPROC{0..5}; do
. /dev/stdin <<RUN
read -r -u \${$x[0]}; echo \$REPLY
echo "$x <- main" >&\${$x[1]}
read -r -u \${$x[0]}; echo \$REPLY
RUN
done
Run Code Online (Sandbox Code Playgroud)
然后
# ./coproc.sh
declare -a COPROC0='([0]="21" [1]="16" [2]="23")'
declare -a COPROC1='([0]="24" [1]="19" [2]="26")'
declare -a COPROC2='([0]="27" [1]="22" [2]="29")'
declare -a COPROC3='([0]="30" [1]="25" [2]="32")'
declare -a COPROC4='([0]="33" [1]="28" [2]="35")'
declare -a COPROC5='([0]="36" [1]="31" [2]="38")'
coproc[0] -> main
COPROC0 <- main
coproc[1] -> main
COPROC1 <- main
coproc[2] -> main
COPROC2 <- main
coproc[3] -> main
COPROC3 <- main
coproc[4] -> main
COPROC4 <- main
coproc[5] -> main
COPROC5 <- main
Run Code Online (Sandbox Code Playgroud)
你去,产生什么.<(:)通过进程替换打开一个匿名管道,它会消失,但管道会因为你有一个句柄而四处乱窜.我通常做一个sleep 1
而不是:
因为它有点活泼,我会得到一个"文件忙"错误 - 如果运行一个真正的命令就不会发生(例如command true
)
"heredoc采购":
. /dev/stdin <<EOF
[...]
EOF
Run Code Online (Sandbox Code Playgroud)
这适用于我曾尝试过的每一个shell,包括busybox/etc(initramfs).我以前从未见过它,我在刺激时独立发现它,谁知道源可以接受args?但是,如果存在这样的事情,它通常可以作为一种更易于管理的评估形式.
Bri*_*uch 34
nohup blah &
Run Code Online (Sandbox Code Playgroud)
用你的进程名称代替等等!
Wil*_*ung 17
就个人而言,我喜欢'批处理'命令.
$ batch
> mycommand -x arg1 -y arg2 -z arg3
> ^D
Run Code Online (Sandbox Code Playgroud)
这会将其填入后台,然后将结果邮寄给您.它是cron的一部分.
Jon*_*ler 11
正如其他人所指出的,要在后台运行进程以便可以断开与SSH会话的连接,您需要让后台进程正确地将其自身与其控制终端取消关联 - 这是SSH会话使用的伪tty.
您可以在书籍中找到有关守护进程的信息,例如Stevens的"高级网络程序,第1卷,第3版"或Rochkind的"高级Unix编程".
我最近(在过去几年)不得不处理一个没有正确守护自己的顽固计划.我最终通过创建一个通用的守护程序处理它 - 类似于nohup,但有更多的控件可用.
Usage: daemonize [-abchptxV][-d dir][-e err][-i in][-o out][-s sigs][-k fds][-m umask] -- command [args...]
-V print version and exit
-a output files in append mode (O_APPEND)
-b both output and error go to output file
-c create output files (O_CREAT)
-d dir change to given directory
-e file error file (standard error - /dev/null)
-h print help and exit
-i file input file (standard input - /dev/null)
-k fd-list keep file descriptors listed open
-m umask set umask (octal)
-o file output file (standard output - /dev/null)
-s sig-list ignore signal numbers
-t truncate output files (O_TRUNC)
-p print daemon PID on original stdout
-x output files must be new (O_EXCL)
Run Code Online (Sandbox Code Playgroud)
在不使用GNU getopt()函数的系统上,双破折号是可选的; 在Linux等上有必要(或者你必须在环境中指定POSIXLY_CORRECT).因为双破折号在任何地方都可以使用,所以最好使用它.
如果你想要源代码,你仍然可以联系我(gmail dot com的名字点最后一个名字)daemonize
.
但是,代码是现在(终于)在我的GitHub上可用的SOQ(堆栈溢出问题)存储库中的文件daemonize-1.10.tgz
在
包
的子目录.
在基于Debian的系统上(在远程计算机上)安装:
sudo apt-get install tmux
用法:
TMUX
运行你想要的命令
要重命名会话:
按Ctrl + B然后$
设置名称
退出会话:
按Ctrl + B然后D
(这离开了tmux会话).然后,您可以注销SSH.
当您需要再次返回/检查时,启动SSH,然后输入
tmux附加session_name
它会带你回到你的tmux会话.
如果父进程被终止,Nohup允许客户端进程被杀死,以便在您注销时进行参数.更好的还是使用:
nohup /bin/sh -c "echo \$\$ > $pidfile; exec $FOO_BIN $FOO_CONFIG " > /dev/null
Run Code Online (Sandbox Code Playgroud)
Nohup使您开始免受终止的过程,当您退出时,您的SSH会话及其子进程将被终止.我给出的命令为您提供了一种方法,您可以将应用程序的pid存储在pid文件中,以便以后可以正确地终止它并允许进程在您注销后运行.
如果您使用screen以root身份运行进程,请注意特权提升攻击的可能性.如果您自己的帐户以某种方式受到损害,将有一种直接的方式来接管整个服务器.
如果需要定期运行此进程并且您在服务器上具有足够的访问权限,则更好的选择是使用cron运行该作业.您也可以使用init.d(超级守护程序)在后台启动您的进程,它可以在完成后立即终止.
nohup
如果要将详细信息记录到文件中,则非常好.但是当它转到后台时,如果你的脚本要求,你就无法给它一个密码.我想你一定要试试screen
.它是一个实用程序,您可以使用yum在您的Linux发行版上安装,例如在CentOS上,yum install screen
然后通过putty或其他软件在您的shell类型中访问您的服务器screen
.它将在putty中打开screen [0].做你的工作.您可以在同一个putty会话中创建更多屏幕[1],屏幕[2]等.
您需要了解的基本命令:
开始屏幕
为了ç reate下一屏
要移动到您创建的n ext屏幕
要d etach
在工作期间关闭你的腻子.下次通过putty类型登录时
要重新连接到屏幕,您可以在屏幕上看到您的进程仍在运行.并退出屏幕类型#exit.
欲了解更多详情,请参阅man screen
.
小智 5
对于大多数进程,您可以使用以下旧的Linux命令行技巧来伪守护进程:
# ((mycommand &)&)
Run Code Online (Sandbox Code Playgroud)
例如:
# ((sleep 30 &)&)
# exit
Run Code Online (Sandbox Code Playgroud)
然后启动一个新的终端窗口,并:
# ps aux | grep sleep
Run Code Online (Sandbox Code Playgroud)
将显示sleep 30
仍在运行。
您所做的是从孩子的孩子开始的过程,当您退出时,nohup
通常会触发该过程退出的命令不会级联到孙子,而将其作为孤立的过程继续运行。
我更喜欢这个“设置它和忘记它”的方法,无需处理nohup
,screen
,TMUX,I / O重定向,或任何东西。