当使用JSch通过Java执行时,某些Unix命令失败并显示“... not found”

时间:2016-10-13 13:05:10

标签: java shell unix ssh jsch

我有一段连接到Unix服务器并执行命令的代码。

我一直在尝试使用简单的命令,但它们运行正常。

我可以登录并获取命令的输出。

我需要通过Java运行Ab-initio图。

我正在使用air sandbox run graph命令。

当我使用SSH客户端登录并运行命令时运行正常。我能够运行图表。但是,当我尝试通过Java运行命令时,它会给我一个“air not found”错误。

对JSch支持的Unix命令有什么限制吗?

知道为什么我无法通过Java代码运行命令吗?

以下是代码:

public static void connect(){
    try{
      JSch jsch=new JSch();  

      String host="*****";
      String user="*****";
      String config =
                "Host foo\n"+
                "  User "+user+"\n"+
                "  Hostname "+host+"\n";

      ConfigRepository configRepository =
            com.jcraft.jsch.OpenSSHConfig.parse(config);

      jsch.setConfigRepository(configRepository);
      Session session=jsch.getSession("foo");

      String passwd ="*****";
      session.setPassword(passwd);
      UserInfo ui = new MyUserInfo(){

          public boolean promptYesNo(String message){

            int foo = 0;
            return foo==0;
          }
      };
      session.setUserInfo(ui);
      session.connect();

      String command="air sandbox run <graph-path>";

      Channel channel=session.openChannel("exec");
      ((ChannelExec)channel).setCommand(command);

      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;
          page_message=new String(tmp, 0, i);
          System.out.print(page_message);
        }
        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 void main(String arg[]){
    connect();
}

public String return_message(){
      String ret_message=page_message;
      return ret_message;
}

public static abstract class MyUserInfo
    implements UserInfo, UIKeyboardInteractive{
    public String getPassword(){ return null; }
    public boolean promptYesNo(String str){ return false; }
    public String getPassphrase(){ return null; }
    public boolean promptPassphrase(String message){ return false; }
    public boolean promptPassword(String message){ return false; }
    public void showMessage(String message){ }
    public String[] promptKeyboardInteractive(String destination,
                        String name,
                        String instruction,
                        String[] prompt,
                        boolean[] echo){
        return null;
    }
}

3 个答案:

答案 0 :(得分:6)

&#34; exec&#34; JSch中的信道(正确地)不为会话分配伪终端(PTY)。因此,可能(可能)获取了一组不同的启动脚本(特别是对于非交互式会话,.bash_profile未获取)。和/或脚本中的不同分支基于TERM环境变量的缺失/存在而被采用。因此环境可能与交互式会话不同,您可以使用SSH客户端。

因此,在您的情况下,PATH可能设置不同;因此无法找到air可执行文件。

要验证这是根本原因,请在SSH客户端中禁用伪终端分配。例如,在PuTTY中,它的 Connection&gt; SSH&gt; TTY&gt;不要分配伪终端。然后,转到 Connection&gt; SSH&gt;远程命令并输入您的air ...命令。检查会话&gt;关闭退出窗口&gt;从不并打开会话。你应该得到相同的&#34; air not found&#34; 错误。

如何按优先顺序解决此问题:

  1. 修复命令不依赖于特定环境。在命令中使用air的完整路径。 E.g:

    /bin/air sandbox run <graph-path>
    

    如果您不知道完整路径,则在常见的* nix系统上,您可以在交互式SSH会话中使用which air命令。

  2. 修复启动脚本,为交互式和非交互式会话设置PATH相同。

  3. 尝试通过登录shell显式运行脚本(使用带有常见* nix shell的--login开关):

    bash --login -c "air sandbox run sandbox run <graph-path>"
    
  4. 如果命令本身依赖于特定的环境设置而无法修复启动脚本,则可以在命令本身中更改环境。该语法取决于远程系统和/或shell。在常见的* nix系统中,这有效:

    String command="PATH=\"$PATH;/path/to/air\" && air sandbox run <graph-path>";
    
  5. 另一种(不推荐的)方法是强制执行&#34; exec&#34;频道使用.setPty方法:

    Channel channel = session.openChannel("exec");
    ((ChannelExec)channel).setPty(true);
    

    使用伪终端自动执行命令可能会带来令人讨厌的副作用。例如,请参阅Is there a simple way to get rid of junk values that come when you SSH using Python's Paramiko library and fetch output from CLI of a remote machine?

  6. 有关类似问题,请参阅

答案 1 :(得分:1)

你可以试着找出&#34; air&#34;居住着

whereis air

然后使用此结果。

类似

/usr/bin/air sandbox run graph

答案 2 :(得分:1)

您可以使用〜/ .ssh / environment文件来设置AB_HOME和PATH变量。