我有一个长期运行的Java服务器应用程序,它启动一个子进程来执行特定任务(在这种情况下,使用7z命令行实用程序提取7zip文件的内容,但这个特定的细节在这里不应该相关).

服务器应用程序在Ubuntu 14下使用Java 8运行.
子流程正在通过Java ProcessBuilder API启动.
子进程正在访问的文件可能受密码保护.
如果文件受密码保护且没有提供密码作为命令行参数,则7z程序将尝试向终端显示提示输入密码的消息,然后从终端读取密码.
此时,子流程挂起并且不会完成,除非我点击< Enter>在控制Java进程的终端中两次.

基本问题似乎是7z实用程序使用getpass系统调用来显示提示和读取用户输入.根据文件:

The getpass() function opens /dev/tty (the controlling terminal of the process), outputs the string prompt, turns off echoing, reads one line (the “password”), restores the terminal state and closes /dev/tty again.

由于这是一个服务器应用程序,因此让子进程块等待来自Java终端的输入是不可接受的.我想要的是以编程方式关闭子进程终端上的输入,以便getpass调用立即返回而没有输入,并且子进程以错误代码退出.不幸的是,我关闭输入到子进程终端的所有尝试都产生了与上面相同的行为:

我尝试在启动子流程后立即手动关闭Process.getOutputStream()返回的流.
我尝试使用ProcessBuilder.redirectInput将子进程输入重定向到从空文件和/ dev / null读取.
为了更好的衡量,我甚至尝试将Redirect.INHERIT传递给ProcessBuilder,即使这看起来不像我想要的远程.

看起来这应该是可行的,因为我已经观察到7z启动时所需的行为与完全相同的命令行作为守护进程的socat进程的子进程,其中终端输入连接到/ dev / null,但是我无法使用我所掌握的工具在Java中实现这一目标.


解决方法:

为了排除getpass()函数打开/ dev / tty(进程的控制终端),我们必须安排该实用程序没有控制tty,这是通过调用setsid()实现的.这个C程序(让我们命名一下) notty)进行此调用,然后执行给定的实用程序:

int main(int argc, char *argv[])
{
    setsid();                   // get rid of the controlling tty
    close(0);                   // preclude reading STDIN as well
    execvp(argv[1], argv+1);    // execute the given program file
}

我们可以通过将它添加到ProcessBuilder命令,例如,在Java应用程序中使用它. G.:

        ProcessBuilder pb = new ProcessBuilder("notty", "7z", … /*arguments*/);

标签: linux, java, terminal, tty

相关文章推荐

添加新评论,含*的栏目为必填