annotate src/viewer_swing/java/com/glavsoft/viewer/swing/ssh/SshConnectionManager.java @ 0:daa24f8a557b

TightVNC original
author YU
date Thu, 11 Sep 2014 07:30:03 +0900
parents
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
0
daa24f8a557b TightVNC original
YU
parents:
diff changeset
1 // Copyright (C) 2010, 2011, 2012, 2013 GlavSoft LLC.
daa24f8a557b TightVNC original
YU
parents:
diff changeset
2 // All rights reserved.
daa24f8a557b TightVNC original
YU
parents:
diff changeset
3 //
daa24f8a557b TightVNC original
YU
parents:
diff changeset
4 //-------------------------------------------------------------------------
daa24f8a557b TightVNC original
YU
parents:
diff changeset
5 // This file is part of the TightVNC software. Please visit our Web site:
daa24f8a557b TightVNC original
YU
parents:
diff changeset
6 //
daa24f8a557b TightVNC original
YU
parents:
diff changeset
7 // http://www.tightvnc.com/
daa24f8a557b TightVNC original
YU
parents:
diff changeset
8 //
daa24f8a557b TightVNC original
YU
parents:
diff changeset
9 // This program is free software; you can redistribute it and/or modify
daa24f8a557b TightVNC original
YU
parents:
diff changeset
10 // it under the terms of the GNU General Public License as published by
daa24f8a557b TightVNC original
YU
parents:
diff changeset
11 // the Free Software Foundation; either version 2 of the License, or
daa24f8a557b TightVNC original
YU
parents:
diff changeset
12 // (at your option) any later version.
daa24f8a557b TightVNC original
YU
parents:
diff changeset
13 //
daa24f8a557b TightVNC original
YU
parents:
diff changeset
14 // This program is distributed in the hope that it will be useful,
daa24f8a557b TightVNC original
YU
parents:
diff changeset
15 // but WITHOUT ANY WARRANTY; without even the implied warranty of
daa24f8a557b TightVNC original
YU
parents:
diff changeset
16 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
daa24f8a557b TightVNC original
YU
parents:
diff changeset
17 // GNU General Public License for more details.
daa24f8a557b TightVNC original
YU
parents:
diff changeset
18 //
daa24f8a557b TightVNC original
YU
parents:
diff changeset
19 // You should have received a copy of the GNU General Public License along
daa24f8a557b TightVNC original
YU
parents:
diff changeset
20 // with this program; if not, write to the Free Software Foundation, Inc.,
daa24f8a557b TightVNC original
YU
parents:
diff changeset
21 // 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
daa24f8a557b TightVNC original
YU
parents:
diff changeset
22 //-------------------------------------------------------------------------
daa24f8a557b TightVNC original
YU
parents:
diff changeset
23 //
daa24f8a557b TightVNC original
YU
parents:
diff changeset
24
daa24f8a557b TightVNC original
YU
parents:
diff changeset
25 package com.glavsoft.viewer.swing.ssh;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
26
daa24f8a557b TightVNC original
YU
parents:
diff changeset
27 import com.glavsoft.utils.Strings;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
28 import com.glavsoft.viewer.CancelConnectionException;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
29 import com.glavsoft.viewer.swing.ConnectionParams;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
30 import com.glavsoft.viewer.swing.Utils;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
31 import com.jcraft.jsch.*;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
32
daa24f8a557b TightVNC original
YU
parents:
diff changeset
33 import javax.swing.*;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
34 import java.io.ByteArrayInputStream;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
35 import java.io.ByteArrayOutputStream;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
36 import java.io.IOException;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
37 import java.io.InputStream;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
38 import java.lang.reflect.InvocationTargetException;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
39 import java.util.logging.Logger;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
40 import java.util.prefs.Preferences;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
41
daa24f8a557b TightVNC original
YU
parents:
diff changeset
42 public class SshConnectionManager implements SshKnownHostsManager {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
43
daa24f8a557b TightVNC original
YU
parents:
diff changeset
44 public static final String SSH_NODE = "com/glavsoft/viewer/ssh";
daa24f8a557b TightVNC original
YU
parents:
diff changeset
45 public static final String KNOWN_HOSTS = "known_hosts";
daa24f8a557b TightVNC original
YU
parents:
diff changeset
46 private Session session;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
47 private String errorMessage = "";
daa24f8a557b TightVNC original
YU
parents:
diff changeset
48 private final JFrame parentWindow;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
49 private JSch jsch;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
50
daa24f8a557b TightVNC original
YU
parents:
diff changeset
51 public SshConnectionManager(JFrame parentWindow) {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
52 this.parentWindow = parentWindow;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
53 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
54
daa24f8a557b TightVNC original
YU
parents:
diff changeset
55 public int connect(ConnectionParams connectionParams) throws CancelConnectionException {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
56 if (Strings.isTrimmedEmpty(connectionParams.sshUserName)) {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
57 connectionParams.sshUserName = getInteractivelySshUserName();
daa24f8a557b TightVNC original
YU
parents:
diff changeset
58 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
59
daa24f8a557b TightVNC original
YU
parents:
diff changeset
60 if (session != null && session.isConnected()) {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
61 session.disconnect();
daa24f8a557b TightVNC original
YU
parents:
diff changeset
62 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
63 jsch = new JSch();
daa24f8a557b TightVNC original
YU
parents:
diff changeset
64 try {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
65 jsch.setKnownHosts(getKnownHostsStream());
daa24f8a557b TightVNC original
YU
parents:
diff changeset
66 } catch (JSchException e) {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
67 Logger.getLogger(this.getClass().getName()).severe("Cannot set JSCH known hosts: " + e.getMessage());
daa24f8a557b TightVNC original
YU
parents:
diff changeset
68 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
69 int port = 0;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
70 try {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
71 session = jsch.getSession(
daa24f8a557b TightVNC original
YU
parents:
diff changeset
72 connectionParams.sshUserName, connectionParams.sshHostName, connectionParams.getSshPortNumber());
daa24f8a557b TightVNC original
YU
parents:
diff changeset
73 UserInfo ui = new SwingSshUserInfo(parentWindow);
daa24f8a557b TightVNC original
YU
parents:
diff changeset
74 session.setUserInfo(ui);
daa24f8a557b TightVNC original
YU
parents:
diff changeset
75 session.connect();
daa24f8a557b TightVNC original
YU
parents:
diff changeset
76 sync();
daa24f8a557b TightVNC original
YU
parents:
diff changeset
77 port = session.setPortForwardingL(0, connectionParams.hostName, connectionParams.getPortNumber());
daa24f8a557b TightVNC original
YU
parents:
diff changeset
78 } catch (JSchException e) {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
79 session.disconnect();
daa24f8a557b TightVNC original
YU
parents:
diff changeset
80 errorMessage = e.getMessage();
daa24f8a557b TightVNC original
YU
parents:
diff changeset
81 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
82 return port;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
83 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
84
daa24f8a557b TightVNC original
YU
parents:
diff changeset
85 private String getInteractivelySshUserName() throws CancelConnectionException {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
86 class Pair {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
87 int intRes;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
88 String stringRes;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
89 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
90 final Pair result = new Pair();
daa24f8a557b TightVNC original
YU
parents:
diff changeset
91 try {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
92 SwingUtilities.invokeAndWait(new Runnable() {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
93 @Override
daa24f8a557b TightVNC original
YU
parents:
diff changeset
94 public void run() {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
95 final JOptionPane pane = new JOptionPane("Please enter the user name for SSH connection:",
daa24f8a557b TightVNC original
YU
parents:
diff changeset
96 JOptionPane.QUESTION_MESSAGE, JOptionPane.OK_CANCEL_OPTION);
daa24f8a557b TightVNC original
YU
parents:
diff changeset
97 pane.setWantsInput(true);
daa24f8a557b TightVNC original
YU
parents:
diff changeset
98 final JDialog dialog = pane.createDialog(parentWindow, "SSH User Name");
daa24f8a557b TightVNC original
YU
parents:
diff changeset
99 Utils.decorateDialog(dialog);
daa24f8a557b TightVNC original
YU
parents:
diff changeset
100 dialog.setVisible(true);
daa24f8a557b TightVNC original
YU
parents:
diff changeset
101 result.stringRes = pane.getInputValue() != null ? (String) pane.getInputValue() : "";
daa24f8a557b TightVNC original
YU
parents:
diff changeset
102 result.intRes = pane.getValue() != null ? (Integer) pane.getValue() : JOptionPane.OK_CANCEL_OPTION;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
103 dialog.dispose();
daa24f8a557b TightVNC original
YU
parents:
diff changeset
104 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
105 });
daa24f8a557b TightVNC original
YU
parents:
diff changeset
106 } catch (InterruptedException e) {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
107 Logger.getLogger(getClass().getName()).severe(e.getMessage());
daa24f8a557b TightVNC original
YU
parents:
diff changeset
108 } catch (InvocationTargetException e) {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
109 Logger.getLogger(getClass().getName()).severe(e.getMessage());
daa24f8a557b TightVNC original
YU
parents:
diff changeset
110 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
111 if (result.intRes != JOptionPane.OK_OPTION) {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
112 throw new CancelConnectionException("No Ssh User Name entered");
daa24f8a557b TightVNC original
YU
parents:
diff changeset
113 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
114 return result.stringRes;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
115 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
116
daa24f8a557b TightVNC original
YU
parents:
diff changeset
117 public boolean isConnected() {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
118 return session.isConnected();
daa24f8a557b TightVNC original
YU
parents:
diff changeset
119 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
120
daa24f8a557b TightVNC original
YU
parents:
diff changeset
121 public String getErrorMessage() {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
122 return errorMessage;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
123 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
124
daa24f8a557b TightVNC original
YU
parents:
diff changeset
125 private InputStream getKnownHostsStream() {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
126 Preferences sshNode = Preferences.userRoot().node(SSH_NODE);
daa24f8a557b TightVNC original
YU
parents:
diff changeset
127 return new ByteArrayInputStream(sshNode.getByteArray(KNOWN_HOSTS, new byte[0]));
daa24f8a557b TightVNC original
YU
parents:
diff changeset
128 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
129
daa24f8a557b TightVNC original
YU
parents:
diff changeset
130 @Override
daa24f8a557b TightVNC original
YU
parents:
diff changeset
131 public void sync() {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
132 ByteArrayOutputStream out = new ByteArrayOutputStream();
daa24f8a557b TightVNC original
YU
parents:
diff changeset
133 HostKeyRepository repository = jsch.getHostKeyRepository();
daa24f8a557b TightVNC original
YU
parents:
diff changeset
134 try {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
135 HostKey[] hostKey = repository.getHostKey();
daa24f8a557b TightVNC original
YU
parents:
diff changeset
136 if (null == hostKey) return;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
137 for (HostKey hk : hostKey) {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
138 String host = hk.getHost();
daa24f8a557b TightVNC original
YU
parents:
diff changeset
139 String type = hk.getType();
daa24f8a557b TightVNC original
YU
parents:
diff changeset
140 if (type.equals("UNKNOWN")) {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
141 write(out, host);
daa24f8a557b TightVNC original
YU
parents:
diff changeset
142 write(out, "\n");
daa24f8a557b TightVNC original
YU
parents:
diff changeset
143 continue;
daa24f8a557b TightVNC original
YU
parents:
diff changeset
144 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
145 write(out, host);
daa24f8a557b TightVNC original
YU
parents:
diff changeset
146 write(out, " ");
daa24f8a557b TightVNC original
YU
parents:
diff changeset
147 write(out, type);
daa24f8a557b TightVNC original
YU
parents:
diff changeset
148 write(out, " ");
daa24f8a557b TightVNC original
YU
parents:
diff changeset
149 write(out, hk.getKey());
daa24f8a557b TightVNC original
YU
parents:
diff changeset
150 write(out, "\n");
daa24f8a557b TightVNC original
YU
parents:
diff changeset
151 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
152 } catch (IOException e) {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
153 Logger.getLogger(this.getClass().getName()).severe("Cannot sync JSCH known hosts: " + e.getMessage());
daa24f8a557b TightVNC original
YU
parents:
diff changeset
154 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
155 Preferences sshNode = Preferences.userRoot().node(SSH_NODE);
daa24f8a557b TightVNC original
YU
parents:
diff changeset
156 sshNode.putByteArray(KNOWN_HOSTS, out.toByteArray());
daa24f8a557b TightVNC original
YU
parents:
diff changeset
157 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
158
daa24f8a557b TightVNC original
YU
parents:
diff changeset
159 private void write(ByteArrayOutputStream out, String str) throws IOException {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
160 try {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
161 out.write(str.getBytes("UTF-8"));
daa24f8a557b TightVNC original
YU
parents:
diff changeset
162 } catch (java.io.UnsupportedEncodingException e) {
daa24f8a557b TightVNC original
YU
parents:
diff changeset
163 out.write(str.getBytes());
daa24f8a557b TightVNC original
YU
parents:
diff changeset
164 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
165 }
daa24f8a557b TightVNC original
YU
parents:
diff changeset
166
daa24f8a557b TightVNC original
YU
parents:
diff changeset
167 }