我试图通过JSch(空间检查命令)使用JSch执行以下Linux命令:
df -h -P |awk '{print $6" "$4}' |awk '{ if ($1 == "/On_Demand") {print}}' | awk '{print $2}' | awk '{ if ($1 ~ /M/) {print ($1*1)} else if($1 ~ /G/) {print ($1*1024)} else if($1 ~ /T/) {print ($1*1024*1024)} else if($1==0) {print ($1*1)} else if($1 ~ /K/) {print ($1/1024)} else {print ("-1")}}'
备份命令:
export ORACLE_HOME=/path/to/ora/home;export ORACLE_SID=sid;/U01/NEW_DEMO/path/to/ora/home/bin/expdp username/password directory=ON_DEMAND schemas=xyz dumpfile=expdp_xyz.dmp logfile=xyz.log;tail -n 1 /On_Demand/xyz.log
读取输出的代码:
public String getOutput() {
LOGGER.debug("[getOutput]");
StringBuffer output = new StringBuffer();
InputStream in = null;
if (channel != null && channel.isConnected()) {
try {
in = channel.getInputStream();
byte[] tmp = new byte[1024];
while (true) {
while (in.available() > 0) {
int i = in.read(tmp, 0, 1024);
if (i < 0)
break;
output.append(new String(tmp, 0, i));
}
if (channel.isClosed()) {
LOGGER.debug("[getOutput] Channel is closed, so breaking while loop with exit code: "+channel.getExitStatus());
break;
}
}
} catch (IOException e) {
LOGGER.error(e.toString());
e.printStackTrace();
} finally {
channel.disconnect();
}
} else {
System.out.println("Channel is disconnected");
}
return output.toString();
}
问题是,当我尝试读取命令输出时,很多次我获得了可用空间,但有时候我没有输出。
当再次调用SSH类(比如2或3次,在新会话中建立的地方)来触发相同的命令时,我得到输出(这很奇怪)。
同样的问题出现在另一个功能中,我发出命令来备份Oracle模式,当我尝试读取Oracle生成的日志文件时(备份和尾部命令都被立即触发,半冒号分开) ,我没有输出,但日志文件在那里。多次尝试触发相同的备份命令可以输出。
这是一个月以来我的工具中出现的一个大问题。
对此有任何帮助将不胜感激。
提前致谢。
答案 0 :(得分:1)
我相信你的代码中存在争用条件。
如果命令设法生成输出并退出,则在in.available()
和channel.isClosed()
测试之间,您将丢失输出。
了解official Exec.java
example如何在in.available()
区块中重新测试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) continue;
System.out.println("exit-status: "+channel.getExitStatus());
break;
}
try{Thread.sleep(1000);}catch(Exception ee){}
}