我有一段连接到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;
}
}
答案 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; 错误。
如何按优先顺序解决此问题:
修复命令不依赖于特定环境。在命令中使用air
的完整路径。 E.g:
/bin/air sandbox run <graph-path>
如果您不知道完整路径,则在常见的* nix系统上,您可以在交互式SSH会话中使用which air
命令。
修复启动脚本,为交互式和非交互式会话设置PATH
相同。
尝试通过登录shell显式运行脚本(使用带有常见* nix shell的--login
开关):
bash --login -c "air sandbox run sandbox run <graph-path>"
如果命令本身依赖于特定的环境设置而无法修复启动脚本,则可以在命令本身中更改环境。该语法取决于远程系统和/或shell。在常见的* nix系统中,这有效:
String command="PATH=\"$PATH;/path/to/air\" && air sandbox run <graph-path>";
另一种(不推荐的)方法是强制执行&#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?
有关类似问题,请参阅
答案 1 :(得分:1)
你可以试着找出&#34; air&#34;居住着
whereis air
然后使用此结果。
类似
/usr/bin/air sandbox run graph
答案 2 :(得分:1)
您可以使用〜/ .ssh / environment文件来设置AB_HOME和PATH变量。