| author | wenzelm | 
| Sat, 15 Oct 2016 15:23:06 +0200 | |
| changeset 64224 | 3ed43cfc8b14 | 
| parent 64222 | 184e3a932778 | 
| child 64226 | 65f7d2eea2d7 | 
| permissions | -rw-r--r-- | 
| 64123 | 1 | /* Title: Pure/General/ssh.scala | 
| 2 | Author: Makarius | |
| 3 | ||
| 64124 | 4 | SSH client based on JSch (see also http://www.jcraft.com/jsch/examples). | 
| 64123 | 5 | */ | 
| 6 | ||
| 7 | package isabelle | |
| 8 | ||
| 9 | ||
| 64134 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 10 | import java.io.{InputStream, OutputStream, ByteArrayOutputStream}
 | 
| 64131 | 11 | |
| 64134 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 12 | import scala.collection.{mutable, JavaConversions}
 | 
| 64131 | 13 | |
| 64222 | 14 | import com.jcraft.jsch.{JSch, Logger => JSch_Logger, Session => JSch_Session, SftpException,
 | 
| 64132 | 15 | OpenSSHConfig, UserInfo, Channel => JSch_Channel, ChannelExec, ChannelSftp, SftpATTRS} | 
| 64123 | 16 | |
| 17 | ||
| 18 | object SSH | |
| 19 | {
 | |
| 64185 | 20 | /* target machine: user@host syntax */ | 
| 64141 | 21 | |
| 64185 | 22 | object Target | 
| 64141 | 23 |   {
 | 
| 24 | val Pattern = "^([^@]+)@(.+)$".r | |
| 25 | ||
| 26 | def parse(s: String): (String, String) = | |
| 27 |       s match {
 | |
| 28 | case Pattern(user, host) => (user, host) | |
| 29 |         case _ => ("", s)
 | |
| 30 | } | |
| 31 | ||
| 32 | def unapplySeq(s: String): Option[List[String]] = | |
| 64185 | 33 |       parse(s) match {
 | 
| 34 | case (_, "") => None | |
| 35 | case (user, host) => Some(List(user, host)) | |
| 36 | } | |
| 64141 | 37 | } | 
| 38 | ||
| 64142 | 39 | val default_port = 22 | 
| 40 | ||
| 64141 | 41 | |
| 64123 | 42 | /* init */ | 
| 43 | ||
| 64130 | 44 | def init(options: Options): SSH = | 
| 64123 | 45 |   {
 | 
| 64130 | 46 |     val config_dir = Path.explode(options.string("ssh_config_dir"))
 | 
| 64123 | 47 |     if (!config_dir.is_dir) error("Bad ssh config directory: " + config_dir)
 | 
| 48 | ||
| 49 | val jsch = new JSch | |
| 50 | ||
| 64130 | 51 |     val config_file = Path.explode(options.string("ssh_config_file"))
 | 
| 64123 | 52 | if (config_file.is_file) | 
| 53 | jsch.setConfigRepository(OpenSSHConfig.parseFile(File.platform_path(config_file))) | |
| 54 | ||
| 55 |     val known_hosts = config_dir + Path.explode("known_hosts")
 | |
| 56 | if (!known_hosts.is_file) known_hosts.file.createNewFile | |
| 57 | jsch.setKnownHosts(File.platform_path(known_hosts)) | |
| 58 | ||
| 64130 | 59 | val identity_files = | 
| 60 |       Library.space_explode(':', options.string("ssh_identity_files")).map(Path.explode(_))
 | |
| 64123 | 61 | for (identity_file <- identity_files if identity_file.is_file) | 
| 62 | jsch.addIdentity(File.platform_path(identity_file)) | |
| 63 | ||
| 64130 | 64 | new SSH(options, jsch) | 
| 64123 | 65 | } | 
| 66 | ||
| 64130 | 67 | def connect_timeout(options: Options): Int = | 
| 68 |     options.seconds("ssh_connect_timeout").ms.toInt
 | |
| 69 | ||
| 64123 | 70 | |
| 71 | /* logging */ | |
| 72 | ||
| 73 | def logging(verbose: Boolean = true, debug: Boolean = false) | |
| 74 |   {
 | |
| 75 | JSch.setLogger(if (verbose) new Logger(debug) else null) | |
| 76 | } | |
| 77 | ||
| 78 | private class Logger(debug: Boolean) extends JSch_Logger | |
| 79 |   {
 | |
| 80 | def isEnabled(level: Int): Boolean = level != JSch_Logger.DEBUG || debug | |
| 81 | ||
| 82 | def log(level: Int, msg: String) | |
| 83 |     {
 | |
| 84 |       level match {
 | |
| 85 | case JSch_Logger.ERROR | JSch_Logger.FATAL => Output.error_message(msg) | |
| 86 | case JSch_Logger.WARN => Output.warning(msg) | |
| 87 | case _ => Output.writeln(msg) | |
| 88 | } | |
| 89 | } | |
| 90 | } | |
| 91 | ||
| 92 | ||
| 64128 | 93 | /* user info */ | 
| 94 | ||
| 95 | object No_User_Info extends UserInfo | |
| 96 |   {
 | |
| 97 | def getPassphrase: String = null | |
| 98 | def getPassword: String = null | |
| 99 | def promptPassword(msg: String): Boolean = false | |
| 100 | def promptPassphrase(msg: String): Boolean = false | |
| 101 | def promptYesNo(msg: String): Boolean = false | |
| 102 | def showMessage(msg: String): Unit = Output.writeln(msg) | |
| 103 | } | |
| 104 | ||
| 105 | ||
| 64132 | 106 | /* channel */ | 
| 64129 | 107 | |
| 64134 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 108 | class Channel[C <: JSch_Channel] private[SSH]( | 
| 64166 | 109 | val session: Session, val kind: String, val channel: C) | 
| 64129 | 110 |   {
 | 
| 64130 | 111 | override def toString: String = kind + " " + session.toString | 
| 64129 | 112 | |
| 64142 | 113 |     def close() { channel.disconnect }
 | 
| 64129 | 114 | } | 
| 115 | ||
| 64132 | 116 | |
| 64191 | 117 | /* Sftp channel */ | 
| 118 | ||
| 119 | type Attrs = SftpATTRS | |
| 120 | ||
| 121 | sealed case class Dir_Entry(name: String, attrs: Attrs) | |
| 122 |   {
 | |
| 123 | def is_file: Boolean = attrs.isReg | |
| 124 | def is_dir: Boolean = attrs.isDir | |
| 125 | } | |
| 126 | ||
| 127 | class Sftp private[SSH](session: Session, kind: String, channel: ChannelSftp) | |
| 128 | extends Channel[ChannelSftp](session, kind, channel) | |
| 129 |   {
 | |
| 130 | channel.connect(connect_timeout(session.options)) | |
| 131 | ||
| 132 | def home: String = channel.getHome() | |
| 133 | ||
| 134 |     def chmod(permissions: Int, remote_path: String) { channel.chmod(permissions, remote_path) }
 | |
| 135 | def mv(remote_path1: String, remote_path2: String): Unit = | |
| 136 | channel.rename(remote_path1, remote_path2) | |
| 137 |     def rm(remote_path: String) { channel.rm(remote_path) }
 | |
| 138 |     def mkdir(remote_path: String) { channel.mkdir(remote_path) }
 | |
| 139 |     def rmdir(remote_path: String) { channel.rmdir(remote_path) }
 | |
| 140 | ||
| 64222 | 141 | def stat(remote_path: String): Option[Dir_Entry] = | 
| 142 |       try { Some(Dir_Entry(remote_path, channel.stat(remote_path))) }
 | |
| 143 |       catch { case _: SftpException => None }
 | |
| 144 | ||
| 145 | def is_file(remote_path: String): Boolean = stat(remote_path).map(_.is_file) getOrElse false | |
| 146 | def is_dir(remote_path: String): Boolean = stat(remote_path).map(_.is_dir) getOrElse false | |
| 64191 | 147 | |
| 148 | def read_dir(remote_path: String): List[Dir_Entry] = | |
| 149 |     {
 | |
| 150 | val dir = channel.ls(remote_path) | |
| 151 |       (for {
 | |
| 152 | i <- (0 until dir.size).iterator | |
| 153 | a = dir.get(i).asInstanceOf[AnyRef] | |
| 154 | name = Untyped.get[String](a, "filename") | |
| 155 | attrs = Untyped.get[Attrs](a, "attrs") | |
| 156 | if name != "." && name != ".." | |
| 157 | } yield Dir_Entry(name, attrs)).toList | |
| 158 | } | |
| 159 | ||
| 160 | def find_files(remote_path: String, pred: Dir_Entry => Boolean = _ => true): List[Dir_Entry] = | |
| 161 |     {
 | |
| 162 | def find(dir: String): List[Dir_Entry] = | |
| 163 | read_dir(dir).flatMap(entry => | |
| 164 |           {
 | |
| 165 | val file = dir + "/" + entry.name | |
| 166 | if (entry.is_dir) find(file) | |
| 167 | else if (pred(entry)) List(entry.copy(name = file)) | |
| 168 | else Nil | |
| 169 | }) | |
| 170 | find(remote_path) | |
| 171 | } | |
| 172 | ||
| 173 | def open_input(remote_path: String): InputStream = channel.get(remote_path) | |
| 174 | def open_output(remote_path: String): OutputStream = channel.put(remote_path) | |
| 175 | ||
| 176 | def read_file(remote_path: String, local_path: Path): Unit = | |
| 177 | channel.get(remote_path, File.platform_path(local_path)) | |
| 178 | def read_bytes(remote_path: String): Bytes = | |
| 179 | using(open_input(remote_path))(Bytes.read_stream(_)) | |
| 180 | def read(remote_path: String): String = | |
| 181 | using(open_input(remote_path))(File.read_stream(_)) | |
| 182 | ||
| 183 | def write_file(remote_path: String, local_path: Path): Unit = | |
| 184 | channel.put(File.platform_path(local_path), remote_path) | |
| 185 | def write_bytes(remote_path: String, bytes: Bytes): Unit = | |
| 186 | using(open_output(remote_path))(bytes.write_stream(_)) | |
| 187 | def write(remote_path: String, text: String): Unit = | |
| 188 | using(open_output(remote_path))(stream => Bytes(text).write_stream(stream)) | |
| 189 | } | |
| 190 | ||
| 191 | ||
| 64132 | 192 | /* exec channel */ | 
| 193 | ||
| 64134 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 194 | private val exec_wait_delay = Time.seconds(0.3) | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 195 | |
| 64166 | 196 | class Exec private[SSH](session: Session, kind: String, channel: ChannelExec) | 
| 197 | extends Channel[ChannelExec](session, kind, channel) | |
| 64131 | 198 |   {
 | 
| 199 |     def kill(signal: String) { channel.sendSignal(signal) }
 | |
| 64134 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 200 | |
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 201 | val exit_status: Future[Int] = | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 202 |       Future.thread("ssh_wait") {
 | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 203 | while (!channel.isClosed) Thread.sleep(exec_wait_delay.ms) | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 204 | channel.getExitStatus | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 205 | } | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 206 | |
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 207 | val stdin: OutputStream = channel.getOutputStream | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 208 | val stdout: InputStream = channel.getInputStream | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 209 | val stderr: InputStream = channel.getErrStream | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 210 | |
| 64166 | 211 | // connect after preparing streams | 
| 212 | channel.connect(connect_timeout(session.options)) | |
| 64134 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 213 | |
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 214 | def result( | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 215 | progress_stdout: String => Unit = (_: String) => (), | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 216 | progress_stderr: String => Unit = (_: String) => (), | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 217 | strict: Boolean = true): Process_Result = | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 218 |     {
 | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 219 | stdin.close | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 220 | |
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 221 | def read_lines(stream: InputStream, progress: String => Unit): List[String] = | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 222 |       {
 | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 223 | val result = new mutable.ListBuffer[String] | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 224 | val line_buffer = new ByteArrayOutputStream(100) | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 225 | def line_flush() | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 226 |         {
 | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 227 | val line = line_buffer.toString(UTF8.charset_name) | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 228 | progress(line) | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 229 | result += line | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 230 | line_buffer.reset | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 231 | } | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 232 | |
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 233 | var c = 0 | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 234 | var finished = false | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 235 |         while (!finished) {
 | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 236 |           while ({ c = stream.read; c != -1 && c != 10 }) line_buffer.write(c)
 | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 237 | if (c == 10) line_flush() | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 238 |           else if (channel.isClosed) {
 | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 239 | if (line_buffer.size > 0) line_flush() | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 240 | finished = true | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 241 | } | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 242 | else Thread.sleep(exec_wait_delay.ms) | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 243 | } | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 244 | |
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 245 | result.toList | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 246 | } | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 247 | |
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 248 |       val out_lines = Future.thread("ssh_stdout") { read_lines(stdout, progress_stdout) }
 | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 249 |       val err_lines = Future.thread("ssh_stderr") { read_lines(stderr, progress_stderr) }
 | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 250 | |
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 251 | def terminate() | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 252 |       {
 | 
| 64136 | 253 | close | 
| 64134 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 254 | out_lines.join | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 255 | err_lines.join | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 256 | exit_status.join | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 257 | } | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 258 | |
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 259 | val rc = | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 260 |         try { exit_status.join }
 | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 261 |         catch { case Exn.Interrupt() => terminate(); Exn.Interrupt.return_code }
 | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 262 | |
| 64136 | 263 | close | 
| 64134 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 264 | if (strict && rc == Exn.Interrupt.return_code) throw Exn.Interrupt() | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 265 | |
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 266 | Process_Result(rc, out_lines.join, err_lines.join) | 
| 
57581e4026fe
proper support for exec channel (see also bash.scala);
 wenzelm parents: 
64133diff
changeset | 267 | } | 
| 64131 | 268 | } | 
| 269 | ||
| 270 | ||
| 64123 | 271 | /* session */ | 
| 272 | ||
| 64166 | 273 | class Session private[SSH](val options: Options, val session: JSch_Session) | 
| 64123 | 274 |   {
 | 
| 64166 | 275 | def update_options(new_options: Options): Session = new Session(new_options, session) | 
| 276 | ||
| 64123 | 277 | override def toString: String = | 
| 278 | (if (session.getUserName == null) "" else session.getUserName + "@") + | |
| 279 | (if (session.getHost == null) "" else session.getHost) + | |
| 64142 | 280 | (if (session.getPort == default_port) "" else ":" + session.getPort) + | 
| 64126 | 281 | (if (session.isConnected) "" else " (disconnected)") | 
| 64123 | 282 | |
| 64142 | 283 |     def close() { session.disconnect }
 | 
| 64123 | 284 | |
| 64191 | 285 | def sftp(): Sftp = | 
| 286 |     {
 | |
| 287 | val kind = "sftp" | |
| 288 | val channel = session.openChannel(kind).asInstanceOf[ChannelSftp] | |
| 289 | new Sftp(this, kind, channel) | |
| 290 | } | |
| 64135 
865dda40e1cc
provide execute operation, similar to Isabelle_System.bash;
 wenzelm parents: 
64134diff
changeset | 291 | |
| 64166 | 292 | def exec(command: String): Exec = | 
| 64129 | 293 |     {
 | 
| 64130 | 294 | val kind = "exec" | 
| 295 | val channel = session.openChannel(kind).asInstanceOf[ChannelExec] | |
| 64190 
c62b99e3ec07
provide USER_HOME, such that symbolic Path.explode("~") can be used remotely;
 wenzelm parents: 
64185diff
changeset | 296 |       channel.setCommand("export USER_HOME=\"$HOME\"\n" + command)
 | 
| 64166 | 297 | new Exec(this, kind, channel) | 
| 64129 | 298 | } | 
| 64123 | 299 | |
| 64191 | 300 | def execute(command: String, | 
| 301 | progress_stdout: String => Unit = (_: String) => (), | |
| 302 | progress_stderr: String => Unit = (_: String) => (), | |
| 303 | strict: Boolean = true): Process_Result = | |
| 304 | exec(command).result(progress_stdout, progress_stderr, strict) | |
| 64137 | 305 | |
| 306 | ||
| 307 | /* tmp dirs */ | |
| 308 | ||
| 309 | def rm_tree(remote_dir: String): Unit = | |
| 310 |       execute("rm -r -f " + File.bash_string(remote_dir)).check
 | |
| 311 | ||
| 312 | def tmp_dir(): String = | |
| 313 |       execute("mktemp -d -t tmp.XXXXXXXXXX").check.out
 | |
| 314 | ||
| 315 | def with_tmp_dir[A](body: String => A): A = | |
| 316 |     {
 | |
| 317 | val remote_dir = tmp_dir() | |
| 318 |       try { body(remote_dir) } finally { rm_tree(remote_dir) }
 | |
| 319 | } | |
| 64123 | 320 | } | 
| 321 | } | |
| 322 | ||
| 64130 | 323 | class SSH private(val options: Options, val jsch: JSch) | 
| 64123 | 324 | {
 | 
| 64166 | 325 | def update_options(new_options: Options): SSH = new SSH(new_options, jsch) | 
| 326 | ||
| 64142 | 327 | def open_session(host: String, port: Int = SSH.default_port, user: String = ""): SSH.Session = | 
| 64123 | 328 |   {
 | 
| 64141 | 329 | val session = jsch.getSession(if (user == "") null else user, host, port) | 
| 64125 
a034dac5ca3c
clarified (hardwired!) default (see also jEdit/FTP);
 wenzelm parents: 
64124diff
changeset | 330 | |
| 
a034dac5ca3c
clarified (hardwired!) default (see also jEdit/FTP);
 wenzelm parents: 
64124diff
changeset | 331 | session.setUserInfo(SSH.No_User_Info) | 
| 
a034dac5ca3c
clarified (hardwired!) default (see also jEdit/FTP);
 wenzelm parents: 
64124diff
changeset | 332 |     session.setConfig("MaxAuthTries", "3")
 | 
| 
a034dac5ca3c
clarified (hardwired!) default (see also jEdit/FTP);
 wenzelm parents: 
64124diff
changeset | 333 | |
| 64130 | 334 |     if (options.bool("ssh_compression")) {
 | 
| 64123 | 335 |       session.setConfig("compression.s2c", "zlib@openssh.com,zlib,none")
 | 
| 336 |       session.setConfig("compression.c2s", "zlib@openssh.com,zlib,none")
 | |
| 337 |       session.setConfig("compression_level", "9")
 | |
| 338 | } | |
| 64125 
a034dac5ca3c
clarified (hardwired!) default (see also jEdit/FTP);
 wenzelm parents: 
64124diff
changeset | 339 | |
| 64130 | 340 | session.connect(SSH.connect_timeout(options)) | 
| 341 | new SSH.Session(options, session) | |
| 64123 | 342 | } | 
| 343 | } |