我有使用j2ssh sshclient在linux服务器上执行远程命令的方法。远程命令可能需要几秒钟到一分钟以上才能执行。我需要Java程序等到命令完成执行才继续,但事实并非如此。 Java程序运行该命令,但在远程命令完成之前继续。这是我的方法:
//The connect is done prior to calling the method.
public static String executeCommand(String host, String command, String path)
throws Exception
{
cd(path);
System.out.println("-- ssh: executing command: " + command + " on "
+ host);
SessionChannelClient session = ssh.openSessionChannel();
session.startShell();
session.getOutputStream().write("sudo -s \n".getBytes());
session.getOutputStream().write(command.getBytes());
session.getOutputStream().write("\n exit\n".getBytes());
IOStreamConnector output = new IOStreamConnector();
java.io.ByteArrayOutputStream bos = new java.io.ByteArrayOutputStream();
output.connect(session.getInputStream(), bos);
String theOutput = bos.toString();
System.out.println("output..." + theOutput);
session.close();
disconnect();
return theOutput;
}
答案 0 :(得分:1)
这里的问题是你启动shell,输出命令然后线程化读取操作,以便从另一个线程执行从Inputstream读取。函数的主线程会立即移动到关闭会话,因此在收到所有输出之前,命令将被终止。
为了防止这种情况,只需从会话的Inputstream读取,直到它在同一个线程上返回EOF,即删除IOStreamConnector的使用并手动读入ByteArrayOutputStream。然后在流返回EOF时调用session.close(),因为这表示已从服务器接收到所有数据。
byte[] buf = new byte[1024];
int r;
ByteArrayOutputStream bos = new java.io.ByteArrayOutputStream();
while((r = session.getInputStream().read(buf)) > -1) {
bos.write(buf, 0, r);
}
答案 1 :(得分:0)
它应该按以下方式工作:
//The connect is done prior to calling the method.
public static String executeCommand(String host, String command, String path)
throws Exception
{
cd(path);
System.out.println("-- ssh: executing command: " + command + " on "
+ host);
SessionChannelClient session = ssh.openSessionChannel();
if ( session.executeCommand(cmd) ) {
java.io.ByteArrayOutputStream bos = new java.io.ByteArrayOutputStream();
output.connect(session.getInputStream(), bos);
String theOutput = bos.toString();
System.out.println("output..." + theOutput);
}
session.close();