我正在通过JSch在java上建立一个ssh连接,一切似乎工作正常,直到我试图运行这个.sh文件。 shell脚本的名称是 repoUpdate.sh ,它非常简单:
echo ' ****Repository update****'
echo ' Location: /home/cissys/repo/'
echo -e ' Command: svn update /home/cissys/repo/2.3.0'
svn update /home/cissys/repo/2.3.0
这是我直接在linux控制台上获得的输出,并带有正确的命令响应:
[cissys@dsatelnx5 ~]$ repoUpdate.sh
****Repository update****
Location: /home/cissys/repo/
Command: svn update /home/cissys/repo/2.3.0
At revision 9432.
现在,这是我的方法的java代码,其中ssh连接试图调用同一个文件
public void cmremove()
{
try
{
JSch jsch = new JSch();
Session session = jsch.getSession(user, host, port);
UserInfo ui = new SUserInfo(pass, null);
session.setUserInfo(ui);
session.setPassword(pass);
session.connect();
ChannelExec channelExec = (ChannelExec)session.openChannel("exec");
InputStream in = channelExec.getInputStream();
channelExec.setCommand("./repoUpdate.sh");
channelExec.connect();
BufferedReader reader = new BufferedReader(new InputStreamReader(in));
String line;
int index = 0;
while ((line = reader.readLine()) != null)
{
System.out.println(++index + " : " + line);
}
channelExec.disconnect();
session.disconnect();
System.out.println("Done!");
}
catch(Exception e)
{
System.err.println("Error: " + e);
}
}
我得到的回应如下:
run:
1 : ****Repository update****
2 : Location: /home/cissys/repo/
3 : Command: svn update /home/cissys/repo/2.3.0
Done!
BUILD SUCCESSFUL (total time: 2 seconds)
在svn命令(修订版9432)上没有输出或执行迹象。
我认为可能会在某个时候关闭会话,而不是让命令正常执行。如果 updateRepo.sh 文件有类似“cp test.txt test_2.txt”的内容,那么就可以毫无问题地执行此操作。但我只对这个和其他一些特定的.sh文件有这个问题。
任何帮助都将不胜感激。
答案 0 :(得分:5)
我怀疑你的shell命令由于某种原因错误输出 - 也许svn
不在你的路径上,也许还有其他奇怪的环境影响 - 但是你没有得到错误输出,因为你不是'寻找它。通常情况下,错误会从channelExec.getErrStream
获得的流中发送,但在您的代码中,您只能从getOutputStream
流中读取错误。
要诊断这一点,您需要获取这些错误消息。使得一个流用于常规输出和错误消息可能比让你的java程序一次从两个流中提取更容易,所以我将这一行添加为repoUpdate.sh
的顶行:
exec 2>&1
这将导致脚本的其余部分使用您正在读取的一个流作为输出和错误。
此外,在您致电chanelExec.disconnect
之前,您应该在您的java程序中记录退出状态,然后更改您的“完成!”消息基于它是什么:
int exitStatus = channelExec.getExitStatus();
channelExec.disconnect();
session.disconnect();
if (exitStatus < 0) {
System.out.println("Done, but exit status not set!");
} else if (exitStatus > 0) {
System.out.println("Done, but with error!");
} else {
System.out.println("Done!");
}
如果你这样做,你应该收到错误消息,告诉你为什么你的命令不能正常工作。
答案 1 :(得分:5)
所以这就是我的所作所为
我在 repoUpdate.sh 文件的顶部添加了exec 2>&1
,并添加了一段代码来读取@DanielMartin建议的输出错误,结果如下:
public void cmremove()
{
try
{
JSch jsch = new JSch();
Session session = jsch.getSession(user, host, port);
UserInfo ui = new SUserInfo(pass, null);
session.setUserInfo(ui);
session.setPassword(pass);
session.connect();
ChannelExec channelExec = (ChannelExec)session.openChannel("exec");
InputStream in = channelExec.getInputStream();
channelExec.setCommand("./repoUpdate.sh");
channelExec.connect();
BufferedReader reader = new BufferedReader(new InputStreamReader(in));
String line;
int index = 0;
while ((line = reader.readLine()) != null)
{
System.out.println(++index + " : " + line);
}
int exitStatus = channelExec.getExitStatus();
channelExec.disconnect();
session.disconnect();
if(exitStatus < 0){
System.out.println("Done, but exit status not set!");
}
else if(exitStatus > 0){
System.out.println("Done, but with error!");
}
else{
System.out.println("Done!");
}
}
catch(Exception e)
{
System.err.println("Error: " + e);
}
}
所以这实际上帮了很多忙。它确认该命令实际上并没有像我想象的那样正确执行。我在我的java输出上得到了这个:
run:
1 : ****Repository update****
2 : Location: /home/cissys/repo/
3 : Command: svn update /home/cissys/repo/2.3.0
4 : ./repoUpdate.sh[6]: svn: not found [No such file or directory]
Done!
BUILD SUCCESSFUL (total time: 2 seconds)
然后我尝试修改 repoUpdate.sh 文件,添加svn命令的绝对路径(谢谢,@ mysnk)
exec 2>&1
echo ' ****Repository update****'
echo ' Location: /home/cissys/repo/'
echo -e ' Command: svn update /home/cissys/repo/2.3.0'
/opt/subversion/bin/svn update /home/cissys/repo/2.3.0
对于我的java,我得到了我想要的东西:
run:
1 : ****Repository update****
2 : Location: /home/cissys/repo/
3 : Command: svn update /home/cissys/repo/2.3.0
4 : At revision 9443.
Done!
BUILD SUCCESSFUL (total time: 3 seconds)
我发现通过java从会话中获得的$PATH
与我直接在linux控制台上获得的不同。
所以添加svn路径实际上就可以了。非常感谢你的帮助!
答案 2 :(得分:0)
如何使用'svn'命令的绝对路径?