| Class | Net::SFTP::Operations::Download | 
| In: | lib/net/sftp/operations/download.rb lib/net/sftp/operations/download.rb | 
| Parent: | Object | 
A general purpose downloader module for Net::SFTP. It can download files into IO objects, or directly to files on the local file system. It can even download entire directory trees via SFTP, and provides a flexible progress reporting mechanism.
To download a single file from the remote server, simply specify both the remote and local paths:
  downloader = sftp.download("/path/to/remote.txt", "/path/to/local.txt")
By default, this operates asynchronously, so if you want to block until the download finishes, you can use the ‘bang’ variant:
  sftp.download!("/path/to/remote.txt", "/path/to/local.txt")
Or, if you have multiple downloads that you want to run in parallel, you can employ the wait method of the returned object:
  dls = %w(file1 file2 file3).map { |f| sftp.download("remote/#{f}", f) }
  dls.each { |d| d.wait }
To download an entire directory tree, recursively, simply specify :recursive => true:
  sftp.download!("/path/to/remotedir", "/path/to/local", :recursive => true)
This will download "/path/to/remotedir", it‘s contents, it‘s subdirectories, and their contents, recursively, to "/path/to/local" on the local host. (If you specify :recursive => true and the source is not a directory, you‘ll get an error!)
If you want to pull the contents of a file on the remote server, and store the data in memory rather than immediately to disk, you can pass an IO object as the destination:
  require 'stringio'
  io = StringIO.new
  sftp.download!("/path/to/remote", io)
This will only work for single-file downloads. Trying to do so with :recursive => true will cause an error.
The following options are supported:
Sometimes it is desirable to track the progress of a download. There are two ways to do this: either using a callback block, or a special custom object.
Using a block it‘s pretty straightforward:
  sftp.download!("remote", "local") do |event, downloader, *args|
    case event
    when :open then
      # args[0] : file metadata
      puts "starting download: #{args[0].remote} -> #{args[0].local} (#{args[0].size} bytes}"
    when :get then
      # args[0] : file metadata
      # args[1] : byte offset in remote file
      # args[2] : data that was received
      puts "writing #{args[2].length} bytes to #{args[0].local} starting at #{args[1]}"
    when :close then
      # args[0] : file metadata
      puts "finished with #{args[0].remote}"
    when :mkdir then
      # args[0] : local path name
      puts "creating directory #{args[0]}"
    when :finish then
      puts "all done!"
  end
However, for more complex implementations (e.g., GUI interfaces and such) a block can become cumbersome. In those cases, you can create custom handler objects that respond to certain methods, and then pass your handler to the downloader:
  class CustomHandler
    def on_open(downloader, file)
      puts "starting download: #{file.remote} -> #{file.local} (#{file.size} bytes)"
    end
    def on_get(downloader, file, offset, data)
      puts "writing #{data.length} bytes to #{file.local} starting at #{offset}"
    end
    def on_close(downloader, file)
      puts "finished with #{file.remote}"
    end
    def on_mkdir(downloader, path)
      puts "creating directory #{path}"
    end
    def on_finish(downloader)
      puts "all done!"
    end
  end
  sftp.download!("remote", "local", :progress => CustomHandler.new)
If you omit any of those methods, the progress updates for those missing events will be ignored. You can create a catchall method named "call" for those, instead.
| Entry | = | Struct.new(:remote, :local, :directory, :size, :handle, :offset, :sink) | A simple struct for encapsulating information about a single remote file or directory that needs to be downloaded. | |
| DEFAULT_READ_SIZE | = | 32_000 | The default read size. | |
| Entry | = | Struct.new(:remote, :local, :directory, :size, :handle, :offset, :sink) | A simple struct for encapsulating information about a single remote file or directory that needs to be downloaded. | |
| DEFAULT_READ_SIZE | = | 32_000 | The default read size. | 
| local | [R] | The destination of the download (the name of a file or directory on the local server, or an IO object) | 
| local | [R] | The destination of the download (the name of a file or directory on the local server, or an IO object) | 
| options | [R] | The hash of options that was given to this Download instance. | 
| options | [R] | The hash of options that was given to this Download instance. | 
| properties | [R] | The properties hash for this object | 
| properties | [R] | The properties hash for this object | 
| remote | [R] | The source of the download (the name of a file or directory on the remote server) | 
| remote | [R] | The source of the download (the name of a file or directory on the remote server) | 
| sftp | [R] | The SFTP session instance that drives this download. | 
| sftp | [R] | The SFTP session instance that drives this download. | 
Instantiates a new downloader process on top of the given SFTP session. local is either an IO object that should receive the data, or a string identifying the target file or directory on the local host. remote is a string identifying the location on the remote host that the download should source.
This will return immediately, and requires that the SSH event loop be run in order to effect the download. (See wait.)
     # File lib/net/sftp/operations/download.rb, line 146
146:     def initialize(sftp, local, remote, options={}, &progress)
147:       @sftp = sftp
148:       @local = local
149:       @remote = remote
150:       @progress = progress || options[:progress]
151:       @options = options
152:       @active = 0
153:       @properties = options[:properties] || {}
154: 
155:       self.logger = sftp.logger
156: 
157:       if recursive? && local.respond_to?(:write)
158:         raise ArgumentError, "cannot download a directory tree in-memory"
159:       end
160: 
161:       @stack = [Entry.new(remote, local, recursive?)]
162:       process_next_entry
163:     end
          Instantiates a new downloader process on top of the given SFTP session. local is either an IO object that should receive the data, or a string identifying the target file or directory on the local host. remote is a string identifying the location on the remote host that the download should source.
This will return immediately, and requires that the SSH event loop be run in order to effect the download. (See wait.)
     # File lib/net/sftp/operations/download.rb, line 146
146:     def initialize(sftp, local, remote, options={}, &progress)
147:       @sftp = sftp
148:       @local = local
149:       @remote = remote
150:       @progress = progress || options[:progress]
151:       @options = options
152:       @active = 0
153:       @properties = options[:properties] || {}
154: 
155:       self.logger = sftp.logger
156: 
157:       if recursive? && local.respond_to?(:write)
158:         raise ArgumentError, "cannot download a directory tree in-memory"
159:       end
160: 
161:       @stack = [Entry.new(remote, local, recursive?)]
162:       process_next_entry
163:     end
          Forces the transfer to stop.
     # File lib/net/sftp/operations/download.rb, line 178
178:     def abort!
179:       @active = 0
180:       @stack.clear
181:     end
          Forces the transfer to stop.
     # File lib/net/sftp/operations/download.rb, line 178
178:     def abort!
179:       @active = 0
180:       @stack.clear
181:     end
          Returns true if there are any active requests or pending files or directories.
     # File lib/net/sftp/operations/download.rb, line 173
173:     def active?
174:       @active > 0 || stack.any?
175:     end
          Returns true if there are any active requests or pending files or directories.
     # File lib/net/sftp/operations/download.rb, line 173
173:     def active?
174:       @active > 0 || stack.any?
175:     end
          Returns the value of the :recursive key in the options hash that was given when the object was instantiated.
     # File lib/net/sftp/operations/download.rb, line 167
167:     def recursive?
168:       options[:recursive]
169:     end
          Returns the value of the :recursive key in the options hash that was given when the object was instantiated.
     # File lib/net/sftp/operations/download.rb, line 167
167:     def recursive?
168:       options[:recursive]
169:     end