2

所以我有一些代码可以检查远程 SFTP 服务器上是否有某个文件:

def size
    adapter.sftp.stat(path).size
end

sftp在这种情况下定义的 Net::SFTP::Session 对象在哪里

@sftp = Net::SFTP.start(host, username, password: password)

并且path是我要调用的对象的文件路径stat()

不幸的是,当我尝试执行此代码时,出现此错误:

NoMethodError:
   undefined method `send_data' for nil:NilClass
 # /usr/local/lib/ruby/gems/2.2.0/gems/net-sftp-2.1.2/lib/net/sftp/session.rb:814:in `send_packet'
 # /usr/local/lib/ruby/gems/2.2.0/gems/net-sftp-2.1.2/lib/net/sftp/protocol/base.rb:45:in `send_request'
 # /usr/local/lib/ruby/gems/2.2.0/gems/net-sftp-2.1.2/lib/net/sftp/protocol/01/base.rb:90:in `open'
 # /usr/local/lib/ruby/gems/2.2.0/gems/net-sftp-2.1.2/lib/net/sftp/session.rb:830:in `request'
 # /usr/local/lib/ruby/gems/2.2.0/gems/net-sftp-2.1.2/lib/net/sftp/session.rb:182:in `open'
 # /usr/local/lib/ruby/gems/2.2.0/gems/net-sftp-2.1.2/lib/net/sftp/session.rb:191:in `open!'
 # /usr/local/lib/ruby/gems/2.2.0/gems/net-sftp-2.1.2/lib/net/sftp/operations/file_factory.rb:40:in `open'
 # /Users/Ben/remote_filesystem/lib/remote_filesystem/path/sftp.rb:46:in `size'
 # ./sftp_spec.rb:72:in `block (3 levels) in <top (required)>'

据我所知,通过查看 Net::SFTP::Session 的源代码session.rbchannel.send_data调用了第 814 行的 Net::SFTP::Session,但显然我的 SFTP 会话出于某种原因有一个 Nil 通道。谁能解释如何解决这个问题?

4

2 回答 2

0

如果您正在缓存sftp,则缓存可能已失效。我遇到了这个异常,因为我试图调用ftp.file.open一个ftp不再打开的连接。

于 2016-08-18T00:41:49.347 回答
0

如前所述,这意味着您的 SFTP 会话已终止。

检查 TCP 日志(wireshark 是你的朋友),会话可能在此期间被对等端终止。

发生此类错误的一种情况是,如果您正在执行数据长度超过接收端 TCP 窗口大小的写入(数据)操作。解决方法是使用缓冲区重复写入操作,例如

  io = StringIO.new(data)
  sftp_session.file.open(filename, "w") do |file|
    while buffer = io.read(BUFFER_SIZE)
      file.write(buffer)
    end
  end
于 2018-01-04T19:47:15.167 回答