使用shell脚本启动远程(SSH)Java应用程序不会返回本地提示

我见过类似的问题,所有已解决的问题已经解决/不适用.

我在启动Java应用程序的远程计算机中有一个bash脚本.相关的行将是:

#!/usr/bin/env bash
...
java -cp /full/path/to/my.jar com.whatever.hi.wassup.MainClassThing \
    --arg-1 /full/path/to/relevant_dir --arg-2 /full/path/to/another_dir &
...
echo started my app
exit 0

我用小脚本做了很多测试,没有任何问题.我使用相同的方法启动另一个Java应用程序,但没有问题(如果重要的话,它是带有–daemon参数的服务器应用程序).

在本地机器上,我尝试在本地bash脚本中调用远程脚本,如下所示:

#!/usr/bin/env bash
...
ssh remote_host myRemoteScript

我的所有测试和其他Java应用程序都返回本地提示.问题是,这个确实启动远程Java应用程序并一直到最后一行,输出“启动我的应用程序”但脚本停在那里,我没有本地提示,除非我按Ctrl C它.

在工作的Java应用程序上使用ssh -v我得到:

debug1: channel 0: free: client-session, nchannels 1
Transferred: sent 2992, received 7060 bytes, in 24.4 seconds
Bytes per second: sent 122.6, received 289.2
debug1: Exit status 0

然而在非工作应用程序上,我得到了这个:

debug1: client_input_channel_req: channel 0 rtype exit-status reply 0
debug1: client_input_channel_req: channel 0 rtype eow@openssh.com reply 0
"started my app"

但没有本地提示.我尝试使用nohup和disown运行相同的命令,结果相同.正如所看到的,命令有&在末尾.只注释远程脚本中的那一行没有问题(并没有任何用处),所以我认为这与Java应用程序本身有关 – 但是不是对shell的进程控制?在远程计算机上运行myRemoteScript也可以按预期工作.

这是一个无密码登录.这两台机器都是RHEL 7.2.

编辑

还尝试过:

> ssh remote_host“myRemoteScript”
> ssh -T remote_host myRemoteScript
> ssh remote_host“bash -c myRemoteScript”
> ssh remote_host“myRemoteScript; exit”

最佳答案
您可以重定向java进程的标准输入,输出和错误,以便其标准输入,输出和错误不会连接到远程SSH服务器:

java ... < /dev/null > /dev/null 2>&1 &

这应该允许远程服务器立即终止SSH会话.

当您要求远程SSH服务器调用命令,并且您没有为该命令请求TTY时,服务器将为子进程的标准输入,输出和错误创建一组管道.当远程命令退出时,OpenSSH服务器的当前行为似乎不是关闭会话.相反,服务器在检测到远程进程的管道处理标准输出的文件结束条件时关闭会话.

远程命令启动的java进程将管道继承到SSH服务器作为其标准输入等.SSH服务器在这些管道上没有获得EOF指示,因为仍然有一个进程可以访问管道.

您可以通过运行如下命令来测试:

ssh localhost 'sleep 10 < /dev/null > /dev/null &'

如果ssh在没有等待睡眠完成的情况下立即退出,这意味着您已经让远程SSH服务器不等待睡眠.

我自己的测试表明,SSH服务器正在查看从子进程接收标准输出的管道,因此您需要重定向所有这些.但是,如果SSH服务器的行为在将来发生变化,您应该考虑重定向所有三个管道 – 标准输入,输出和错误.

转载注明原文:使用shell脚本启动远程(SSH)Java应用程序不会返回本地提示 - 代码日志