我正在使用SSH编写用于静态路由管理的
java GUI程序.我的代码如下:
import com.jcraft.jsch.*; import java.io.*; public class Konsep { String status; static String username; static String hostname; String inputcommand; String output; static Session session; JSch jsch = new JSch(); public String status(String stringstatus) { stringstatus = status; return stringstatus; } public String InputCommand(String inputcommandstatus) { inputcommandstatus = inputcommand; return inputcommandstatus; } public void connect(String usernamelokal,String hostnamelokal,String password,int port) { // JSch jsch=new JSch(); try { Session sessionlokal = jsch.getSession(usernamelokal,hostnamelokal,port); sessionlokal.setPassword(password); UserInfo ui = new UserInfoku.Infoku(); sessionlokal.setUserInfo(ui); sessionlokal.setTimeout(0); sessionlokal.connect(); status = "tersambung \n"; username = usernamelokal; hostname = hostnamelokal; session = sessionlokal; System.out.println(username + " " + hostname); } catch (Exception e) { System.out.println(e); status = "Exception = \n " + e + "\n"; } } public void disconnect() { // JSch jsch=new JSch(); try { Session sessionlokal = jsch.getSession(username,hostname); // System.out.println(username +" "+ hostname); sessionlokal.disconnect(); status = "wes pedhoott \n"; } catch (Exception e) { System.out.println(e); status = "Exception = \n " + e + "\n"; } } public void addRoute() { // JSch jsch=new JSch(); System.out.println(username + " " + hostname); try { Session sessionlokal = session; // =jsch.getSession(username,hostname); Channel channel = sessionlokal.openChannel("exec"); ((ChannelExec) channel).setCommand(inputcommand); channel.setInputStream(null); channel.connect(); ((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,1024); if (i < 0) break; System.out.print(new String(tmp,i)); } if (channel.isClosed()) { System.out.println("exit-status: " + channel.getExitStatus()); break; } try { Thread.sleep(1000); } catch (Exception ee) { } } channel.disconnect(); } catch (Exception e) { System.out.println(e); } } }
问题是当我调用connect方法,然后调用addroute,程序返回
root 192.168.50.2 root 192.168.50.2 com.jcraft.jsch.JSchException: session is down
我一直在试图获得会话状态
Session sessionlokal=session; //returns com.jcraft.jsch.JSchException: ChannelExec
要么
Session sessionlokal=jsch.getSession(username,hostname); //returns session is down
我也试图使用keepalive,但它也不工作.
我的目的是创建一个会话来主持(登录),同时离开会话,执行一个命令或命令,并可能稍后执行其他命令,然后在不需要时关闭该会话(注销).
我在这个论坛上一直在搜索,我发现这个question,但代码是创建一个方法来定义一个先执行的命令,然后创建会话,调用命令的方法并关闭会话.
关于如上所述的任何想法?
解决方法
尝试Session.sendKeepAliveMsg()没有成功之后,我来到了以下解决方案,这似乎是相当稳定的:
private Session getSession() throws Exception { try { if (!session.isConnected()) { logger.info("Session nicht konnektiert. Versuche (erneut) zu konnektieren."); session.connect(); } } catch (Throwable t) { logger.info("Session kaputt. Baue neue."); session = jsch.getSession(user,host,port); session.setConfig(config); session.connect(); } return session; }
更新:几天后,它失败了.
我试图通过杀死服务器上的打开会话来测试它.我以前测试过的所有版本都显示出完全相同的行为,无论在等待几天或杀死服务器进程后,问题出现了,所以我认为这个测试 – 以及上述解决方案的结果是有意义的.不幸的是不是
更新2:最终解决方案,伪装和工作:
private Session getSession() throws Exception { try { ChannelExec testChannel = (ChannelExec) session.openChannel("exec"); testChannel.setCommand("true"); testChannel.connect(); if(logger.isDebugEnabled()) { logger.debug("Session erfolgreich getestet,verwende sie erneut"); } testChannel.exit(); } catch (Throwable t) { logger.info("Session kaputt. Baue neue."); session = jsch.getSession(user,port); session.setConfig(config); session.connect(); } return session; }
此版本在生产环境中运行数周.一天一次,我记录了信息消息.
打开频道和执行某些无人值守命令的费用有点令人讨厌,但我发现没有其他方式可以肯定地确定会话的状态.