Class: EventMachine::Protocols::HttpClient2::Request

Inherits:
Object
  • Object
show all
Includes:
Deferrable
Defined in:
lib/em/protocols/httpclient2.rb

Constant Summary collapse

HttpResponseRE =

– TODO, inefficient how we’re handling this. Part of it is done so as to make sure we don’t have problems in detecting chunked-encoding, content-length, etc.

/\AHTTP\/(1.[01]) ([\d]{3})/i
ClenRE =
/\AContent-length:\s*(\d+)/i
ChunkedRE =
/\ATransfer-encoding:\s*chunked/i
ColonRE =
/\:\s*/

Constants included from Deferrable

Deferrable::Pool

Instance Attribute Summary collapse

Instance Method Summary collapse

Methods included from Deferrable

#callback, #cancel_callback, #cancel_errback, #cancel_timeout, #errback, #fail, future, #set_deferred_status, #succeed, #timeout

Constructor Details

#initialize(conn, args) ⇒ Request

Returns a new instance of Request.



68
69
70
71
72
73
74
75
76
# File 'lib/em/protocols/httpclient2.rb', line 68

def initialize conn, args
  @conn = conn
  @args = args
  @header_lines = []
  @headers = {}
  @blanks = 0
  @chunk_trailer = nil
  @chunking = nil
end

Instance Attribute Details

#contentObject (readonly)

Returns the value of attribute content.



65
66
67
# File 'lib/em/protocols/httpclient2.rb', line 65

def content
  @content
end

#header_linesObject (readonly)

Returns the value of attribute header_lines.



63
64
65
# File 'lib/em/protocols/httpclient2.rb', line 63

def header_lines
  @header_lines
end

#headersObject (readonly)

Returns the value of attribute headers.



64
65
66
# File 'lib/em/protocols/httpclient2.rb', line 64

def headers
  @headers
end

#internal_errorObject (readonly)

Returns the value of attribute internal_error.



66
67
68
# File 'lib/em/protocols/httpclient2.rb', line 66

def internal_error
  @internal_error
end

#statusObject (readonly)

Returns the value of attribute status.



62
63
64
# File 'lib/em/protocols/httpclient2.rb', line 62

def status
  @status
end

#versionObject (readonly)

Returns the value of attribute version.



61
62
63
# File 'lib/em/protocols/httpclient2.rb', line 61

def version
  @version
end

Instance Method Details

#receive_chunk_header(ln) ⇒ Object

– Cf RFC 2616 pgh 3.6.1 for the format of HTTP chunks.



139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
# File 'lib/em/protocols/httpclient2.rb', line 139

def receive_chunk_header ln
  if ln.length > 0
    chunksize = ln.to_i(16)
    if chunksize > 0
      @conn.set_text_mode(ln.to_i(16))
    else
      @content = @content ? @content.join : ''
      @chunk_trailer = true
    end
  else
    # We correctly come here after each chunk gets read.
    # p "Got A BLANK chunk line"
  end

end

#receive_chunk_trailer(ln) ⇒ Object



104
105
106
107
108
109
110
111
# File 'lib/em/protocols/httpclient2.rb', line 104

def receive_chunk_trailer ln
  if ln.length == 0
    @conn.pop_request
    succeed(self)
  else
    p "Received chunk trailer line"
  end
end

#receive_chunked_text(text) ⇒ Object

– We get a single chunk. Append it to the incoming content and switch back to line mode.



159
160
161
162
# File 'lib/em/protocols/httpclient2.rb', line 159

def receive_chunked_text text
  # p "RECEIVED #{text.length} CHUNK"
  (@content ||= []) << text
end

#receive_header_line(ln) ⇒ Object

– Allow up to ten blank lines before we get a real response line. Allow no more than 100 lines in the header.



117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
# File 'lib/em/protocols/httpclient2.rb', line 117

def receive_header_line ln
  if ln.length == 0
    if @header_lines.length > 0
      process_header
    else
      @blanks += 1
      if @blanks > 10
        @conn.close_connection
      end
    end
  else
    @header_lines << ln
    if @header_lines.length > 100
      @internal_error = :bad_header
      @conn.close_connection
    end
  end
end

#receive_line(ln) ⇒ Object



92
93
94
95
96
97
98
99
100
# File 'lib/em/protocols/httpclient2.rb', line 92

def receive_line ln
  if @chunk_trailer
    receive_chunk_trailer(ln)
  elsif @chunking
    receive_chunk_header(ln)
  else
    receive_header_line(ln)
  end
end

#receive_sized_text(text) ⇒ Object

– At the present time, we only handle contents that have a length specified by the content-length header.



234
235
236
237
238
# File 'lib/em/protocols/httpclient2.rb', line 234

def receive_sized_text text
  @content = text
  @conn.pop_request
  succeed(self)
end

#receive_text(text) ⇒ Object



226
227
228
# File 'lib/em/protocols/httpclient2.rb', line 226

def receive_text text
  @chunking ? receive_chunked_text(text) : receive_sized_text(text)
end

#send_requestObject



78
79
80
81
82
83
84
85
86
87
88
# File 'lib/em/protocols/httpclient2.rb', line 78

def send_request
  az = @args[:authorization] and az = "Authorization: #{az}\r\n"

  r = [
    "#{@args[:verb]} #{@args[:uri]} HTTP/#{@args[:version] || "1.1"}\r\n",
    "Host: #{@args[:host_header] || "_"}\r\n",
    az || "",
      "\r\n"
  ]
  @conn.send_data r.join
end