Class: HTTP::Connection

Inherits:
Object
  • Object
show all
Extended by:
Forwardable
Defined in:
lib/http/connection.rb

Overview

A connection to the HTTP server

Constant Summary collapse

BUFFER_SIZE =

Attempt to read this much data

16_384
HTTP_1_0 =

HTTP/1.0

"1.0".freeze
HTTP_1_1 =

HTTP/1.1

"1.1".freeze

Instance Method Summary collapse

Constructor Details

#initialize(req, options) ⇒ Connection

Returns a new instance of Connection.

Parameters:

Raises:



25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
# File 'lib/http/connection.rb', line 25

def initialize(req, options)
  @persistent           = options.persistent?
  @keep_alive_timeout   = options.keep_alive_timeout.to_f
  @pending_request      = false
  @pending_response     = false
  @failed_proxy_connect = false

  @parser = Response::Parser.new

  @socket = options.timeout_class.new(options.timeout_options)
  @socket.connect(options.socket_class, req.socket_host, req.socket_port, options.nodelay)

  send_proxy_connect_request(req)
  start_tls(req, options)
  reset_timer
rescue SocketError, SystemCallError => e
  raise ConnectionError, "failed to connect: #{e}"
end

Instance Method Details

#closevoid

This method returns an undefined value.

Close the connection



123
124
125
126
127
128
# File 'lib/http/connection.rb', line 123

def close
  @socket.close unless @socket.closed?

  @pending_response = false
  @pending_request  = false
end

#expired?Boolean

Whether our connection has expired

Returns:

  • (Boolean)


138
139
140
# File 'lib/http/connection.rb', line 138

def expired?
  !@conn_expires_at || @conn_expires_at < Time.now
end

#failed_proxy_connect?Boolean

Returns whenever proxy connect failed.

Returns:

  • (Boolean)

    whenever proxy connect failed



54
55
56
# File 'lib/http/connection.rb', line 54

def failed_proxy_connect?
  @failed_proxy_connect
end

#finish_responsevoid

This method returns an undefined value.

Callback for when we've reached the end of a response



111
112
113
114
115
116
117
118
119
# File 'lib/http/connection.rb', line 111

def finish_response
  close unless keep_alive?

  @parser.reset
  @socket.reset_counter if @socket.respond_to?(:reset_counter)
  reset_timer

  @pending_response = false
end

#keep_alive?Boolean

Whether we're keeping the conn alive

Returns:

  • (Boolean)


132
133
134
# File 'lib/http/connection.rb', line 132

def keep_alive?
  !!@keep_alive && !@socket.closed?
end

#read_headers!void

This method returns an undefined value.

Reads data from socket up until headers are loaded



94
95
96
97
98
99
100
101
102
103
104
105
106
107
# File 'lib/http/connection.rb', line 94

def read_headers!
  loop do
    if read_more(BUFFER_SIZE) == :eof
      fail EOFError unless @parser.headers?
      break
    elsif @parser.headers?
      break
    end
  end

  set_keep_alive
rescue IOError, Errno::ECONNRESET, Errno::EPIPE => e
  raise ConnectionError, "failed to read headers: #{e}"
end

#readpartial(size = BUFFER_SIZE) ⇒ String?

Read a chunk of the body

Returns:

  • (String)

    data chunk

  • (nil)

    when no more data left



81
82
83
84
85
86
87
88
89
90
# File 'lib/http/connection.rb', line 81

def readpartial(size = BUFFER_SIZE)
  return unless @pending_response

  finished = (read_more(size) == :eof) || @parser.finished?
  chunk    = @parser.chunk

  finish_response if finished

  chunk.to_s
end

#send_request(req) ⇒ nil

Send a request to the server

Parameters:

  • req (Request)

    Request to send to the server

Returns:

  • (nil)


62
63
64
65
66
67
68
69
70
71
72
73
74
75
# File 'lib/http/connection.rb', line 62

def send_request(req)
  if @pending_response
    fail StateError, "Tried to send a request while one is pending already. Make sure you read off the body."
  elsif @pending_request
    fail StateError, "Tried to send a request while a response is pending. Make sure you've fully read the body from the request."
  end

  @pending_request = true

  req.stream @socket

  @pending_response = true
  @pending_request  = false
end