Class: SSHake::Session
- Inherits:
-
BaseSession
- Object
- BaseSession
- SSHake::Session
- Defined in:
- lib/sshake/session.rb
Instance Attribute Summary collapse
-
#host ⇒ String
readonly
Return the host to connect to.
-
#session ⇒ Net::SSH::Session
readonly
The underlying net/ssh session.
Attributes inherited from BaseSession
#id, #klogger, #raise_on_error
Class Method Summary collapse
Instance Method Summary collapse
-
#connect ⇒ void
Connect to the SSH server.
-
#connected? ⇒ Boolean
Is there an established SSH connection.
-
#disconnect ⇒ void
Disconnect the underlying SSH connection.
-
#execute(commands, options = nil, &block) ⇒ Object
rubocop:disable Metrics/AbcSize.
-
#initialize(host, user = nil, **options) ⇒ Sshake::Session
constructor
Create a new SSH session.
-
#kill! ⇒ Object
Kill the underlying connection.
-
#port ⇒ Integer
Return the port that will be connected to.
-
#user ⇒ String
Return the username for the connection.
-
#write_data(path, data, options = nil, &block) ⇒ Object
rubocop:enable Metrics/AbcSize.
Constructor Details
#initialize(host, user = nil, **options) ⇒ Sshake::Session
Create a new SSH session
26 27 28 29 30 31 32 |
# File 'lib/sshake/session.rb', line 26 def initialize(host, user = nil, **) super @host = host @user = user @session_options = @session_options.delete(:klogger) end |
Instance Attribute Details
#host ⇒ String (readonly)
Return the host to connect to
21 22 23 |
# File 'lib/sshake/session.rb', line 21 def host @host end |
#session ⇒ Net::SSH::Session (readonly)
The underlying net/ssh session
16 17 18 |
# File 'lib/sshake/session.rb', line 16 def session @session end |
Class Method Details
.create(*args) ⇒ Object
194 195 196 197 198 199 200 201 202 |
# File 'lib/sshake/session.rb', line 194 def create(*args) session = new(*args) if recorder = Thread.current[:sshake_recorder] return RecordedSession.new(recorder, session) end session end |
Instance Method Details
#connect ⇒ void
This method returns an undefined value.
Connect to the SSH server
51 52 53 54 55 |
# File 'lib/sshake/session.rb', line 51 def connect klogger.debug 'Connecting', id: @id, host: @host, user: @user, port: @session_options[:port] || 22 @session = Net::SSH.start(@host, user, **@session_options) true end |
#connected? ⇒ Boolean
Is there an established SSH connection
60 61 62 |
# File 'lib/sshake/session.rb', line 60 def connected? !@session.nil? end |
#disconnect ⇒ void
This method returns an undefined value.
Disconnect the underlying SSH connection
67 68 69 70 71 72 73 74 75 76 77 78 79 80 |
# File 'lib/sshake/session.rb', line 67 def disconnect return false if @session.nil? begin klogger.debug 'Closing connection', id: @id, host: @host @session.close klogger.debug 'Connection closed', id: @id, host: @host rescue StandardError => e logger.exception(e, 'Connection not closed') nil end @session = nil true end |
#execute(commands, options = nil, &block) ⇒ Object
rubocop:disable Metrics/AbcSize
91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 |
# File 'lib/sshake/session.rb', line 91 def execute(commands, = nil, &block) = (, block) command_to_execute = prepare_commands(commands, ) # Execute the command response = Response.new response.command = command_to_execute connect unless connected? klogger.group(id: @id, host: @host) do klogger.info 'Executing command', command: command_to_execute, timeout: .timeout begin channel = nil Timeout.timeout(.timeout) do channel = @session.open_channel do |ch| response.start_time = Time.now channel.exec(command_to_execute) do |_, success| raise "Command \"#{command_to_execute}\" was unable to execute" unless success ch.send_data(.stdin) if .stdin if .file_to_stream.nil? && .sudo_password.nil? klogger.debug 'Sending EOF to channel' ch.eof! end ch.on_data do |_, data| response.stdout += data .stdout&.call(data) klogger.debug "[stdout] #{data.gsub(/\r/, '').strip}" end ch.on_extended_data do |_, _, data| response.stderr += data.delete("\r") .stderr&.call(data) klogger.debug "[stderr] #{data.gsub(/\r/, '').strip}" if .sudo_password && data =~ /^\[sshake-sudo-password\]:\s\z/ klogger.debug 'Sending sudo password', length: .sudo_password.length ch.send_data "#{.sudo_password}\n" if .file_to_stream.nil? klogger.debug 'Sending EOF after password' ch.eof! end end end ch.on_request('exit-status') do |_, data| response.exit_code = data.read_long&.to_i klogger.info 'Exited', exit_code: response.exit_code end ch.on_request('exit-signal') do |_, data| response.exit_signal = data.read_long end if .file_to_stream ch.on_process do |_, data| next if ch.eof? if ch.output.length < 128 * 1024 if data = .file_to_stream.read(1024 * 1024) ch.send_data(data) response.bytes_streamed += data.bytesize else ch.eof! end end end end end end channel.wait end rescue Timeout::Error klogger.debug 'Command timed out' kill! response.timeout! ensure response.finish_time = Time.now end end handle_response(response, ) end |
#kill! ⇒ Object
Kill the underlying connection
83 84 85 86 87 88 |
# File 'lib/sshake/session.rb', line 83 def kill! klogger.debug 'Attemping to shutdown', id: @id, host: @host @session.shutdown! klogger.debug 'Shutdown success', id: @id, host: @host @session = nil end |
#port ⇒ Integer
Return the port that will be connected to
44 45 46 |
# File 'lib/sshake/session.rb', line 44 def port @session_options[:port] || 22 end |
#user ⇒ String
Return the username for the connection
37 38 39 |
# File 'lib/sshake/session.rb', line 37 def user @user || ENV.fetch('USER', nil) end |
#write_data(path, data, options = nil, &block) ⇒ Object
rubocop:enable Metrics/AbcSize
181 182 183 184 185 186 187 188 189 190 |
# File 'lib/sshake/session.rb', line 181 def write_data(path, data, = nil, &block) connect unless connected? tmp_path = "/tmp/sshake-tmp-file-#{SecureRandom.hex(32)}" @session.sftp.file.open(tmp_path, 'w') do |f| d = data.dup.force_encoding('BINARY') f.write(d.slice!(0, 1024)) until d.empty? end response = execute("mv #{tmp_path} #{path}", , &block) response.success? end |