在 Java 中使用 JSch exec 从 ArrayList 执行命令列表

IT小君   2021-11-15T14:02:54

使用 JSch “exec” 通道,我连接到远程服务器并执行命令。我不能做的是执行存储在ArrayList. 我使用 for 循环将每个元素传递给.setCommand(elem);但出于某种奇怪的原因,只有最后一个命令被执行。

ArrayList<String> lists = new ArrayList<String>();
lists.add("hostname");
lists.add("df -l");
Channel channel=session.openChannel("exec");

for (String elem : lists){
    ((ChannelExec)channel).setCommand(elem);
}

完整代码

public class Exec{
  public static void main(String[] arg){
    try {
      JSch jsch=new JSch();

      String host=null;
      if(arg.length>0){
        host=arg[0];
      }
      else{
        host=JOptionPane.showInputDialog("Enter username@hostname",
                                         System.getProperty("user.name")+
                                         "@localhost"); 
      }
      String user=host.substring(0, host.indexOf('@'));
      host=host.substring(host.indexOf('@')+1);

      Session session=jsch.getSession(user, host, 22);


      UserInfo ui=new MyUserInfo();
      session.setUserInfo(ui);
      session.connect();

      ArrayList<String> lists = new ArrayList<String>();
        lists.add("hostname");
        lists.add("df -l");
      Channel channel=session.openChannel("exec");

      for (String elem : lists){
          ((ChannelExec)channel).setCommand(elem);
      }

      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()){
          if(in.available()>0) continue; 
          System.out.println("exit-status: "+channel.getExitStatus());
          break;
        }
        try{Thread.sleep(1000);}catch(Exception ee){}
      }
      channel.disconnect();
      session.disconnect();
    }
    catch(Exception e){
      System.out.println(e);
    }
  }

  public static class MyUserInfo implements UserInfo, UIKeyboardInteractive{
    public String getPassword(){ return passwd; }
    public boolean promptYesNo(String str){
      Object[] options={ "yes", "no" };
      int foo=JOptionPane.showOptionDialog(null, 
             str,
             "Warning", 
             JOptionPane.DEFAULT_OPTION, 
             JOptionPane.WARNING_MESSAGE,
             null, options, options[0]);
       return foo==0;
    }

    String passwd;
    JTextField passwordField=(JTextField)new JPasswordField(20);

    public String getPassphrase(){ return null; }
    public boolean promptPassphrase(String message){ return true; }
    public boolean promptPassword(String message){
      Object[] ob={passwordField}; 
      int result=
        JOptionPane.showConfirmDialog(null, ob, message,
                                      JOptionPane.OK_CANCEL_OPTION);
      if(result==JOptionPane.OK_OPTION){
        passwd=passwordField.getText();
        return true;
      }
      else{ 
        return false; 
      }
    }
    public void showMessage(String message){
      JOptionPane.showMessageDialog(null, message);
    }
    final GridBagConstraints gbc = 
      new GridBagConstraints(0,0,1,1,1,1,
                             GridBagConstraints.NORTHWEST,
                             GridBagConstraints.NONE,
                             new Insets(0,0,0,0),0,0);
    private Container panel;
    public String[] promptKeyboardInteractive(String destination,
                                              String name,
                                              String instruction,
                                              String[] prompt,
                                              boolean[] echo){
      panel = new JPanel();
      panel.setLayout(new GridBagLayout());

      gbc.weightx = 1.0;
      gbc.gridwidth = GridBagConstraints.REMAINDER;
      gbc.gridx = 0;
      panel.add(new JLabel(instruction), gbc);
      gbc.gridy++;

      gbc.gridwidth = GridBagConstraints.RELATIVE;

      JTextField[] texts=new JTextField[prompt.length];
      for(int i=0; i<prompt.length; i++){
        gbc.fill = GridBagConstraints.NONE;
        gbc.gridx = 0;
        gbc.weightx = 1;
        panel.add(new JLabel(prompt[i]),gbc);

        gbc.gridx = 1;
        gbc.fill = GridBagConstraints.HORIZONTAL;
        gbc.weighty = 1;
        if(echo[i]){
          texts[i]=new JTextField(20);
        }
        else{
          texts[i]=new JPasswordField(20);
        }
        panel.add(texts[i], gbc);
        gbc.gridy++;
      }

      if(JOptionPane.showConfirmDialog(null, panel, 
                                       destination+": "+name,
                                       JOptionPane.OK_CANCEL_OPTION,
                                       JOptionPane.QUESTION_MESSAGE)
         ==JOptionPane.OK_OPTION){
        String[] response=new String[prompt.length];
        for(int i=0; i<prompt.length; i++){
          response[i]=texts[i].getText();
        }
        return response;
      }
      else{
        return null;  // cancel
      }
    }
  }
}
点击广告,支持我们为你提供更好的服务
评论(1)
IT小君

您不能setCommand多次调用“exec”通道只能运行一个“命令”。

但是在大多数系统/外壳上,“命令”实际上可以包含多个命令。

语法将取决于您的系统/外壳。分号通常起作用:

((ChannelExec)channel).setCommand("hostname ; df -l");

一些服务器/外壳也允许换行。

((ChannelExec)channel).setCommand("hostname\ndf -l");

在 *nix 服务器上,您还可以使用&&使以下命令仅在前面的命令成功时才执行:

((ChannelExec)channel).setCommand("hostname && df -l");

另请参阅通过 JSch shell 的多个命令


另一种选择是为每个命令打开一个新通道。请注意,您可以通过单个 SSH 会话拥有多个通道。您不必为每个通道/命令重新连接。

2021-11-15T14:02:54   回复