Ruby HTTP2 GET请求

时间:2016-09-20 17:27:38

标签: ruby http2

我尝试使用Ruby gem http-2向Google发送GET请求。

我直接从example解除了代码并稍微简化了一下:

require 'http/2'
require 'socket'
require 'openssl'
require 'uri'

uri = URI.parse('http://www.google.com/')
tcp = TCPSocket.new(uri.host, uri.port)
sock = tcp

conn = HTTP2::Client.new
conn.on(:frame) do |bytes|
  # puts "Sending bytes: #{bytes.unpack("H*").first}"
  sock.print bytes
  sock.flush
end
conn.on(:frame_sent) do |frame|
  puts "Sent frame: #{frame.inspect}"
end
conn.on(:frame_received) do |frame|
  puts "Received frame: #{frame.inspect}"
end

stream = conn.new_stream

stream.on(:close) do
  puts 'stream closed'
  sock.close
end

stream.on(:half_close) do
  puts 'closing client-end of the stream'
end

stream.on(:headers) do |h|
  puts "response headers: #{h}"
end

stream.on(:data) do |d|
  puts "response data chunk: <<#{d}>>"
end

head = {
  ':scheme' => uri.scheme,
  ':method' => 'GET',
  ':path' => uri.path
}

puts 'Sending HTTP 2.0 request'
stream.headers(head, end_stream: true)

while !sock.closed? && !sock.eof?
  data = sock.read_nonblock(1024)
  # puts "Received bytes: #{data.unpack("H*").first}"

  begin
    conn << data
  rescue => e
    puts "#{e.class} exception: #{e.message} - closing socket."
    e.backtrace.each { |l| puts "\t" + l }
    sock.close
  end
end

输出结果为:

Sending HTTP 2.0 request
Sent frame: {:type=>:settings, :stream=>0, :payload=>[[:settings_max_concurrent_streams, 100]]}
Sent frame: {:type=>:headers, :flags=>[:end_headers, :end_stream], :payload=>[[":scheme", "http"], [":method", "GET"], [":path", "/"]], :stream=>1}
closing client-end of the stream

(注意:通过运行实际的示例文件,您可以获得与上面相同的输出,即ruby client.rb http://www.google.com/

为什么没有显示响应数据?

1 个答案:

答案 0 :(得分:5)

google.com等公共服务器不支持明文HTTP / 2。

您正在尝试连接到http://google.com,而您应该真正连接到https://google.com(请注意https计划)。

为了做到这一点,您可能需要使用TLS包装TCP套接字(例如参见here),如果http-2没有为您执行此操作。

另请注意,HTTP / 2需要强大的TLS密码和ALPN,因此请确保您拥有OpenSSL的更新版本(至少1.0.2)。

鉴于http-2的作者是强大的HTTP / 2支持者,我猜你唯一的问题是您尝试了明文http而不是https,我期望http-2库可以处理TLS密码强度和ALPN。