Errno::EPIPE: сломанная труба с Net::FTP
Подобно этому вопросу, я могу подключиться и даже проверить / изменить рабочий каталог, но позвонив list
бросает Errno::EPIPE: Broken pipe
ошибка.
До сих пор я видел это только из терминала (OS X). Я пробовал пару разных хостов, и я убедился, что могу нормально получить доступ с помощью FileZilla. Я также подтвердил, что могу выводить список файлов через обычную командную строку ftp
сессия.
Может ли это быть проблемой брандмауэра? Или проблема с моей оболочкой?
Вот пример сеанса IRB...
irb> ftp = Net::FTP.new 'host.domain.com'
=> #<Net::FTP:0x007fc10d053ab8 @mon_owner=nil, @mon_count=0, @mon_mutex=#<Mutex:0x007fc10d053a68>, @binary=true, @passive=false, @debug_mode=false, @resume=false, @sock=#<TCPSocket:fd 7>, @logged_in=false, @last_response="220 ESS FTP\n", @last_response_code="220">
irb> ftp.debug_mode = true
=> true
irb> ftp.login 'user', '*********'
put: USER user
get: 331 Password required for user
put: PASS *********
get: 230 Logged on
put: TYPE I
get: 200 Type set to I
=> true
irb> ftp.pwd
put: PWD
get: 257 "/" is current directory.
=> "/"
irb> ftp.chdir 'Inventory'
put: CWD Inventory
get: 250 CWD successful. "/Inventory" is current directory.
irb> ftp.list
put: TYPE A
get: 200 Type set to A
put: PORT 10,142,96,98,199,111
put: TYPE I
Errno::EPIPE: Broken pipe
from /Users/acobster/.rvm/rubies/ruby-1.9.3-p392/lib/ruby/1.9.1/net/ftp.rb:257:in `write'
from /Users/acobster/.rvm/rubies/ruby-1.9.3-p392/lib/ruby/1.9.1/net/ftp.rb:257:in `putline'
from /Users/acobster/.rvm/rubies/ruby-1.9.3-p392/lib/ruby/1.9.1/net/ftp.rb:334:in `block in voidcmd'
from /Users/acobster/.rvm/rubies/ruby-1.9.3-p392/lib/ruby/1.9.1/monitor.rb:211:in `mon_synchronize'
from /Users/acobster/.rvm/rubies/ruby-1.9.3-p392/lib/ruby/1.9.1/net/ftp.rb:333:in `voidcmd'
from /Users/acobster/.rvm/rubies/ruby-1.9.3-p392/lib/ruby/1.9.1/net/ftp.rb:162:in `send_type_command'
from /Users/acobster/.rvm/rubies/ruby-1.9.3-p392/lib/ruby/1.9.1/net/ftp.rb:151:in `binary='
from /Users/acobster/.rvm/rubies/ruby-1.9.3-p392/lib/ruby/1.9.1/net/ftp.rb:180:in `ensure in with_binary'
from /Users/acobster/.rvm/rubies/ruby-1.9.3-p392/lib/ruby/1.9.1/net/ftp.rb:180:in `with_binary'
from /Users/acobster/.rvm/rubies/ruby-1.9.3-p392/lib/ruby/1.9.1/net/ftp.rb:477:in `block in retrlines'
from /Users/acobster/.rvm/rubies/ruby-1.9.3-p392/lib/ruby/1.9.1/monitor.rb:211:in `mon_synchronize'
from /Users/acobster/.rvm/rubies/ruby-1.9.3-p392/lib/ruby/1.9.1/net/ftp.rb:476:in `retrlines'
from /Users/acobster/.rvm/rubies/ruby-1.9.3-p392/lib/ruby/1.9.1/net/ftp.rb:722:in `list'
from (irb):9
from /Users/acobster/.rvm/rubies/ruby-1.9.3-p392/bin/irb:16:in `<main>'
Пожалуйста, помогите, спасибо!
Обновление: я могу сделать ftp.list
из сеанса IRB на удаленном сервере...
$ ssh myserver
server $ irb
irb> ftp = Net::FTP.new 'host.domain.com'
=> #<Net::FTP:0x2b7d3c5edc80 @last_response="220 ESS FTP\n", @debug_mode=false, @mon_entering_queue=[], @sock=#<TCPSocket:0x2b7d3c5edb18>, @passive=false, @mon_count=0, @binary=true, @mon_owner=nil, @last_response_code="220", @resume=false, @mon_waiting_queue=[]>
irb> ftp.login 'user', 'pass'
=> "230 Logged on\n"
irb> ftp.pwd
=> "/"
irb> ftp.chdir 'Inventory'
=> nil
irb> ftp.list
=> ["-r--r--r-- 1 ftp ftp 302301 Feb 26 20:27 inventory.zip", "-r--r--r-- 1 ftp ftp 1450282 Feb 26 20:24 InventoryWeb.txt"]
... но я не уверен, какой вывод из этого сделать? Моя локальная версия Ruby? Может ли это быть проблемой брандмауэра / оболочки?
1 ответ
положить: ПОРТ 10,142,96,98,199,111
Вы используете FTP в активном режиме с частным IP-адресом 10.142.96.199. В активном режиме сервер должен создать соединение для передачи данных с вашим хостом, а в пассивном режиме (команда PASV вместо команды PORT) клиент будет подключаться к серверу.
Поскольку сервер не находится в той же частной сети, что и ваш хост (в противном случае вы не сможете подключиться к нему из удаленной системы), он не сможет подключиться к частному IP-адресу 10.142.96.98, указанному в команде PORT, потому что частные IP-адреса не могут быть доступны из Интернета. Сервер, вероятно, поймет это, отправит сообщение об ошибке и закроет соединение. Ваш FTP-клиент, вероятно, попытается отправить другую команду (например, QUIT или ABOR), но, поскольку сервер уже закрыл соединение, эта отправка приведет к EPIPE.
FileZilla работает, вероятно, потому что он будет использовать пассивный режим вместо активного режима. Чтобы использовать пассивный режим с Net::FTP, вы должны установить ftp.passive = true
,