Java SSHを使用した静的ルート管理用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, 0, 1024);
if (i < 0)
break;
System.out.print(new String(tmp, 0, 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
また、キープアライブを使用しようとしましたが、機能しません。
私の意図は、ホストへのセッションを作成し(ログイン)、セッションをアップしたまま、1つまたは複数のコマンドを実行し、後で他のコマンドを実行し、不要になったときにセッションを閉じる(ログアウト)ことです。私はこのフォーラムを検索して、これを見つけました question ですが、コードは、最初に実行するコマンドを定義するメソッドを作成してから、セッションを作成し、コマンドのメソッドを呼び出してセッションを閉じます。
上記のように行う方法についてのアイデアはありますか?
Session.sendKeepAliveMsg()
を試した後、うまくいかなかった次の解決策に行き着きました。
private Session getSession() throws Exception {
try {
if (!session.isConnected()) {
logger.info("Session successfully tested, use it again.");
session.connect();
}
} catch (Throwable t) {
logger.info("Session terminated. Create a new one.");
session = jsch.getSession(user, Host, port);
session.setConfig(config);
session.connect();
}
return session;
}
pdate:数日後、失敗しました。
サーバーで開いているセッションを強制終了することでテストしてみました。この方法でテストした以前のすべてのバージョンは、数日待った後、またはサーバープロセスを強制終了した後に問題が発生したかどうかに関係なく、まったく同じ動作を示したため、このテストと上記のソリューションの結果は意味があると思いました。残念ながらそうではありません。
他の方法で解決し、最新の状態に保つつもりです。
更新2:最終ソリューション、エレガントで機能することが保証されている:
private Session getSession() throws Exception {
try {
ChannelExec testChannel = (ChannelExec) session.openChannel("exec");
testChannel.setCommand("true");
testChannel.connect();
if(logger.isDebugEnabled()) {
logger.debug("Session successfully tested, use it again.");
}
testChannel.exit();
} catch (Throwable t) {
logger.info("Session terminated. Create a new one.");
session = jsch.getSession(user, Host, port);
session.setConfig(config);
session.connect();
}
return session;
}
このバージョンは、実稼働環境で数週間実行されます。 1日1回、情報メッセージをログに記録しています。
チャンネルを開いて何もしないコマンドを実行するコストはやや煩わしいですが、セッションの状態を確実に確認する方法が他にありませんでした。
Session#connect() を呼び出す前に、次のメソッドを呼び出すことで試すことができます。
Session#setServerAliveInterval(int milliseconds)
私はもう1つの関数 Session#sendKeepAliveMsg() を見つけましたが、試すことができます。
プールを使用できますが、私はorg.Apache.commons.pool2を使用したので、プールは接続されたセッションを提供する責任があります。基本的には次のとおりです。
上記は私にとってうまくいきました、検証にロジックを追加する場合、poolConfigはtestOnBorrowまたはtestOnReturnでtrueに設定する必要があることにも注意してください。 KeyedObjectPoolを使用したので、プールからborrowObjectメソッドを使用して、returnObject。