5

Jschを使用してsshを介して単一のコマンドを実行できましたが、2番目のコマンドを実行しようとすると失敗します

デバッグのために、この問題を次の行に落とし込みました。

import java.io.IOException;
import java.io.InputStream;

import com.jcraft.jsch.Channel;
import com.jcraft.jsch.ChannelExec;
import com.jcraft.jsch.JSch;
import com.jcraft.jsch.JSchException;
import com.jcraft.jsch.Session;

public class Exec {

    public static void test(Session session) throws Exception {
        Channel channel = session.openChannel("exec");
        ((ChannelExec) channel).setCommand("pwd");

        channel.setInputStream(null);

        ((ChannelExec) channel).setErrStream(System.err);

        InputStream in = channel.getInputStream();

        channel.connect();

        byte[] tmp = new byte[1024];
        while (true) {
            while (in.available() > 0) {
                int i = in.read(tmp, 0, 1024);
                if (i < 0)
                    break;
                System.out.print(new String(tmp, 0, i));
            }
            if (channel.isClosed()) {
                System.out.println("exit-status: " + channel.getExitStatus());
                break;
            }
            try {
                Thread.sleep(1000);
            } catch (Exception ee) {
            }
        }
        channel.disconnect();
    }

    public static void main(String[] arg) {
        try {
            JSch jsch = new JSch();         
            Session session = jsch.getSession("nck", "127.0.0.1", 22);          
            session.setPassword("asd");         
            session.setConfig("StrictHostKeyChecking", "no");           
            session.connect();

            test(session); // This one succeeds with exit-status: 0         
            test(session); // This one fails with exit-status: 255

            session.disconnect();
        } catch (Exception e) {
            //
        }
    }
}

これは主に公式の Exec の例ですが、これにより次の出力が得られます。

/home/nck
exit-status: 0
exit-status: 255

最初のコマンドは正常に実行されますが、2 番目のコマンドは実行されません。

何か案は ?

4

6 に答える 6

7

@krishnaの回答に基づいた完全に機能するJavaクラスを次に示します。

/**
 * Run several ssh commands in a single JSch session
 */
import com.jcraft.jsch.Channel;
import com.jcraft.jsch.JSch;
import com.jcraft.jsch.Session;
import java.io.*;

class SshMultiCommands
{
   public static void main(String[] args) throws Exception
   {
      JSch jsch = new JSch();
      String user = "user";            //CHANGE ME
      String host = "192.168.222.157"; //CHANGE ME
      String passwd = "password";      //CHANGE ME
      int port = 22;    
      Session session = jsch.getSession(user, host, port);
      session.setPassword(passwd);

      session.setConfig("StrictHostKeyChecking", "no");

      session.connect();

      Channel channel = session.openChannel("shell");
      OutputStream ops = channel.getOutputStream();
      PrintStream ps = new PrintStream(ops, true);

      channel.connect();
      InputStream input = channel.getInputStream();

      //commands
      ps.println("ls -lag");
      ps.println("cd /etc");
      ps.println("ls");
      ps.println("exit");
      ps.close();

      printResult(input, channel);

      channel.disconnect();
      session.disconnect();
   }

   /**
    * @param input
    * @param channel
    */
   private static void printResult(InputStream input,
                                   Channel channel) throws Exception
   {
      int SIZE = 1024;
      byte[] tmp = new byte[SIZE];
      while (true)
      {
         while (input.available() > 0)
         {
            int i = input.read(tmp, 0, SIZE);
            if(i < 0)
               break;
             System.out.print(new String(tmp, 0, i));
         }
         if(channel.isClosed())
         {
            System.out.println("exit-status: " + channel.getExitStatus());
            break;
         }
         try
         {
            Thread.sleep(300);
         }
         catch (Exception ee)
         {
         }
      }
   }
}
于 2013-05-24T03:18:19.450 に答える
4

ちょっと私はjschとUbuntuでまったく同じ問題を抱えています。(どのように)解決しましたか? 実行ごとに新しいセッションを作成すると時間がかかりすぎますか? 現時点では、jsch 例外をキャッチし、「セッションがダウンしていません」を検索してから、セッションを再接続してコマンドを再度実行します。それは機能しますが、良い解決策ではありません。

編集:私は間違った方法で、今のところ私の問題を解決しました

channel = session.openChannel("shell");

「実行」の代わりに。

于 2012-03-30T11:07:02.913 に答える
3

シェルを使用して複数のコマンドを実行できます。以下は、シェル経由でコマンドを実行するコードです。

Channel channel=session.openChannel("shell");
OutputStream ops = channel.getOutputStream();
PrintStream ps = new PrintStream(ops, true);

channel.connect();
ps.println("mkdir folder"); 
ps.println("dir");
//give commands to be executed inside println.and can have any no of commands sent.
ps.close();

InputStream in=channel.getInputStream();
byte[] bt=new byte[1024];

while(true) {
    while(in.available()>0) {
        int i=in.read(bt, 0, 1024);
        if(i<0)
            break;
        String str=new String(bt, 0, i);

        //displays the output of the command executed.
        System.out.print(str);
    }
    if(channel.isClosed()) 
        break;

    Thread.sleep(1000);
    channel.disconnect();
    session.disconnect();   
}
于 2013-01-03T09:41:30.993 に答える
2

exec を使用する場合。コマンド内にセミコロン (;) を入れることで、複数のコマンドを実行できます。

お気に入り:

pwd;ls- -l; 
cd foldername

((ChannelExec) channel).setCommand("pwd;ls -l; cd foldername");
于 2014-10-21T13:31:03.943 に答える
1

私はまったく同じコードを使用しましたが、ssh を介して実行したスクリプトの出力を読み取る際に問題が発生しました。プログラムのすべてのコンテンツが欠落している場合もあれば、コンテンツの一部のみが欠落している場合もありました。ネストされたループで if(channel.isclosed) を変更することで問題を解決しました。これが新しいコードです

while (true) {
        while (in.available() > 0) {
            int i = in.read(tmp, 0, 1024);
            if (i < 0)
                break;
            System.out.print(new String(tmp, 0, i));
        }
        if (channel.isClosed()) {
             if(in.available() > 0) {
                int i = in.read(tmp, 0, 1024);
                System.out.print(new String(tmp, 0, i));
             } 
            System.out.println("exit-status: " + channel.getExitStatus());
            break;
        }
        try {
            Thread.sleep(1000);
        } catch (Exception ee) {
}
于 2012-04-23T13:13:56.390 に答える
0

This wasn't a bug in jsch, it was a bug of my ssh server (cygwin - I bet I configured something wrong...)

Edit: hmm I had the same problem with jsch with Ubuntu ... weird

Guess as workarround I'll have to open a new session for each query.

于 2011-08-27T11:13:22.470 に答える