protocol-http1 0.31.0 → 0.33.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- checksums.yaml.gz.sig +0 -0
- data/lib/protocol/http1/body/chunked.rb +22 -2
- data/lib/protocol/http1/body/fixed.rb +18 -2
- data/lib/protocol/http1/body/remainder.rb +17 -5
- data/lib/protocol/http1/body.rb +16 -0
- data/lib/protocol/http1/connection.rb +247 -24
- data/lib/protocol/http1/error.rb +3 -1
- data/lib/protocol/http1/reason.rb +3 -1
- data/lib/protocol/http1/version.rb +1 -1
- data/lib/protocol/http1.rb +8 -1
- data.tar.gz.sig +0 -0
- metadata +3 -2
- metadata.gz.sig +0 -0
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: d5e39284335ca68719e14bdd3c85f2d9ca6c7fa28939f2625149dc85aa65d2e8
|
4
|
+
data.tar.gz: f75598dfa2d6a89c1738c49bbfa76d627cb9badb2220254b4c82b489564f424a
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 806650cfeeba0aced96ea8ca2d0694c6f5ddce0738b2b3f0493bc6fe70ccbd891981a3aa976cd4644e0622ca9351dd87f80ad997e9ef14794f7b7f2741bd9b98
|
7
|
+
data.tar.gz: 2acf1dd39dbbb9f4f5a34a53135c4c828b6f697f4e32fd74fd02ee4494b06818f2253bb9286b1e6862ac2138e2e61f08fe353af5342792e7587ca5a348c01baa
|
checksums.yaml.gz.sig
CHANGED
Binary file
|
@@ -1,7 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
# Released under the MIT License.
|
4
|
-
# Copyright, 2019-
|
4
|
+
# Copyright, 2019-2025, by Samuel Williams.
|
5
5
|
# Copyright, 2023, by Thomas Morgan.
|
6
6
|
|
7
7
|
require "protocol/http/body/readable"
|
@@ -9,9 +9,16 @@ require "protocol/http/body/readable"
|
|
9
9
|
module Protocol
|
10
10
|
module HTTP1
|
11
11
|
module Body
|
12
|
+
# Represents a chunked body, which is a series of chunks, each with a length prefix.
|
13
|
+
#
|
14
|
+
# See https://tools.ietf.org/html/rfc7230#section-4.1 for more details on the chunked transfer encoding.
|
12
15
|
class Chunked < HTTP::Body::Readable
|
13
16
|
CRLF = "\r\n"
|
14
17
|
|
18
|
+
# Initialize the chunked body.
|
19
|
+
#
|
20
|
+
# @parameter connection [Protocol::HTTP1::Connection] the connection to read the body from.
|
21
|
+
# @parameter headers [Protocol::HTTP::Headers] the headers to read the trailer into, if any.
|
15
22
|
def initialize(connection, headers)
|
16
23
|
@connection = connection
|
17
24
|
@finished = false
|
@@ -22,19 +29,25 @@ module Protocol
|
|
22
29
|
@count = 0
|
23
30
|
end
|
24
31
|
|
32
|
+
# @attribute [Integer] the number of chunks read so far.
|
25
33
|
attr :count
|
26
34
|
|
35
|
+
# @attribute [Integer] the length of the body if known.
|
27
36
|
def length
|
28
|
-
# We only know the length once we've read
|
37
|
+
# We only know the length once we've read the final chunk:
|
29
38
|
if @finished
|
30
39
|
@length
|
31
40
|
end
|
32
41
|
end
|
33
42
|
|
43
|
+
# @returns [Boolean] true if the body is empty, in other words {read} will return `nil`.
|
34
44
|
def empty?
|
35
45
|
@connection.nil?
|
36
46
|
end
|
37
47
|
|
48
|
+
# Close the connection and mark the body as finished.
|
49
|
+
#
|
50
|
+
# @parameter error [Exception | Nil] the error that caused the body to be closed, if any.
|
38
51
|
def close(error = nil)
|
39
52
|
if connection = @connection
|
40
53
|
@connection = nil
|
@@ -49,7 +62,12 @@ module Protocol
|
|
49
62
|
|
50
63
|
VALID_CHUNK_LENGTH = /\A[0-9a-fA-F]+\z/
|
51
64
|
|
65
|
+
# Read a chunk of data.
|
66
|
+
#
|
52
67
|
# Follows the procedure outlined in https://tools.ietf.org/html/rfc7230#section-4.1.3
|
68
|
+
#
|
69
|
+
# @returns [String | Nil] the next chunk of data, or `nil` if the body is finished.
|
70
|
+
# @raises [EOFError] if the connection is closed before the expected length is read.
|
53
71
|
def read
|
54
72
|
if !@finished
|
55
73
|
if @connection
|
@@ -96,12 +114,14 @@ module Protocol
|
|
96
114
|
end
|
97
115
|
end
|
98
116
|
|
117
|
+
# @returns [String] a human-readable representation of the body.
|
99
118
|
def inspect
|
100
119
|
"\#<#{self.class} #{@length} bytes read in #{@count} chunks>"
|
101
120
|
end
|
102
121
|
|
103
122
|
private
|
104
123
|
|
124
|
+
# Read the trailer from the connection, and add any headers to the trailer.
|
105
125
|
def read_trailer
|
106
126
|
while line = @connection.read_line?
|
107
127
|
# Empty line indicates end of trailer:
|
@@ -1,14 +1,19 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
# Released under the MIT License.
|
4
|
-
# Copyright, 2019-
|
4
|
+
# Copyright, 2019-2025, by Samuel Williams.
|
5
5
|
|
6
6
|
require "protocol/http/body/readable"
|
7
7
|
|
8
8
|
module Protocol
|
9
9
|
module HTTP1
|
10
10
|
module Body
|
11
|
+
# Represents a fixed length body.
|
11
12
|
class Fixed < HTTP::Body::Readable
|
13
|
+
# Initialize the body with the given connection and length.
|
14
|
+
#
|
15
|
+
# @parameter connection [Protocol::HTTP1::Connection] the connection to read the body from.
|
16
|
+
# @parameter length [Integer] the length of the body.
|
12
17
|
def initialize(connection, length)
|
13
18
|
@connection = connection
|
14
19
|
|
@@ -16,13 +21,20 @@ module Protocol
|
|
16
21
|
@remaining = length
|
17
22
|
end
|
18
23
|
|
24
|
+
# @attribute [Integer] the length of the body.
|
19
25
|
attr :length
|
26
|
+
|
27
|
+
# @attribute [Integer] the remaining bytes to read.
|
20
28
|
attr :remaining
|
21
29
|
|
30
|
+
# @returns [Boolean] true if the body is empty.
|
22
31
|
def empty?
|
23
32
|
@connection.nil? or @remaining == 0
|
24
33
|
end
|
25
34
|
|
35
|
+
# Close the connection.
|
36
|
+
#
|
37
|
+
# @parameter error [Exception | Nil] the error that caused the connection to be closed, if any.
|
26
38
|
def close(error = nil)
|
27
39
|
if connection = @connection
|
28
40
|
@connection = nil
|
@@ -35,7 +47,10 @@ module Protocol
|
|
35
47
|
super
|
36
48
|
end
|
37
49
|
|
38
|
-
#
|
50
|
+
# Read a chunk of data.
|
51
|
+
#
|
52
|
+
# @returns [String | Nil] the next chunk of data.
|
53
|
+
# @raises [EOFError] if the connection is closed before the expected length is read.
|
39
54
|
def read
|
40
55
|
if @remaining > 0
|
41
56
|
if @connection
|
@@ -57,6 +72,7 @@ module Protocol
|
|
57
72
|
end
|
58
73
|
end
|
59
74
|
|
75
|
+
# @returns [String] a human-readable representation of the body.
|
60
76
|
def inspect
|
61
77
|
"\#<#{self.class} length=#{@length} remaining=#{@remaining} state=#{@connection ? 'open' : 'closed'}>"
|
62
78
|
end
|
@@ -1,26 +1,31 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
# Released under the MIT License.
|
4
|
-
# Copyright, 2019-
|
4
|
+
# Copyright, 2019-2025, by Samuel Williams.
|
5
5
|
|
6
6
|
require "protocol/http/body/readable"
|
7
7
|
|
8
8
|
module Protocol
|
9
9
|
module HTTP1
|
10
10
|
module Body
|
11
|
-
#
|
11
|
+
# Represents the remainder of the body, which reads all the data from the connection until it is finished.
|
12
12
|
class Remainder < HTTP::Body::Readable
|
13
13
|
BLOCK_SIZE = 1024 * 64
|
14
14
|
|
15
|
-
#
|
16
|
-
|
15
|
+
# Initialize the body with the given connection.
|
16
|
+
#
|
17
|
+
# @parameter connection [Protocol::HTTP1::Connection] the connection to read the body from.
|
18
|
+
def initialize(connection, block_size: BLOCK_SIZE)
|
17
19
|
@connection = connection
|
20
|
+
@block_size = block_size
|
18
21
|
end
|
19
22
|
|
23
|
+
# @returns [Boolean] true if the body is empty.
|
20
24
|
def empty?
|
21
25
|
@connection.nil?
|
22
26
|
end
|
23
27
|
|
28
|
+
# Discard the body, which will close the connection and prevent further reads.
|
24
29
|
def discard
|
25
30
|
if connection = @connection
|
26
31
|
@connection = nil
|
@@ -30,14 +35,20 @@ module Protocol
|
|
30
35
|
end
|
31
36
|
end
|
32
37
|
|
38
|
+
# Close the connection.
|
39
|
+
#
|
40
|
+
# @parameter error [Exception | Nil] the error that caused the connection to be closed, if any.
|
33
41
|
def close(error = nil)
|
34
42
|
self.discard
|
35
43
|
|
36
44
|
super
|
37
45
|
end
|
38
46
|
|
47
|
+
# Read a chunk of data.
|
48
|
+
#
|
49
|
+
# @returns [String | Nil] the next chunk of data.
|
39
50
|
def read
|
40
|
-
@connection&.readpartial(
|
51
|
+
@connection&.readpartial(@block_size)
|
41
52
|
rescue EOFError
|
42
53
|
if connection = @connection
|
43
54
|
@connection = nil
|
@@ -47,6 +58,7 @@ module Protocol
|
|
47
58
|
return nil
|
48
59
|
end
|
49
60
|
|
61
|
+
# @returns [String] a human-readable representation of the body.
|
50
62
|
def inspect
|
51
63
|
"\#<#{self.class} state=#{@connection ? 'open' : 'closed'}>"
|
52
64
|
end
|
@@ -0,0 +1,16 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
# Released under the MIT License.
|
4
|
+
# Copyright, 2025, by Samuel Williams.
|
5
|
+
|
6
|
+
require_relative "body/chunked"
|
7
|
+
require_relative "body/fixed"
|
8
|
+
require_relative "body/remainder"
|
9
|
+
|
10
|
+
module Protocol
|
11
|
+
module HTTP1
|
12
|
+
# A collection of classes for handling HTTP/1.1 request and response bodies.
|
13
|
+
module Body
|
14
|
+
end
|
15
|
+
end
|
16
|
+
end
|
@@ -11,10 +11,8 @@ require "protocol/http/headers"
|
|
11
11
|
|
12
12
|
require_relative "reason"
|
13
13
|
require_relative "error"
|
14
|
+
require_relative "body"
|
14
15
|
|
15
|
-
require_relative "body/chunked"
|
16
|
-
require_relative "body/fixed"
|
17
|
-
require_relative "body/remainder"
|
18
16
|
require "protocol/http/body/head"
|
19
17
|
|
20
18
|
require "protocol/http/methods"
|
@@ -39,19 +37,31 @@ module Protocol
|
|
39
37
|
|
40
38
|
# HTTP/1.x header parser:
|
41
39
|
FIELD_NAME = TOKEN
|
42
|
-
|
43
|
-
|
40
|
+
OWS = /[ \t]*/
|
41
|
+
# A field value is any string of characters that does not contain a null character, CR, or LF. After reflecting on the RFCs and surveying real implementations, I came to the conclusion that the RFCs are too restrictive. Most servers only check for the presence of null bytes, and obviously CR/LF characters have semantic meaning in the parser. So, I decided to follow this defacto standard, even if I'm not entirely happy with it.
|
42
|
+
FIELD_VALUE = /[^\0\r\n]+/.freeze
|
43
|
+
HEADER = /\A(#{FIELD_NAME}):#{OWS}(?:(#{FIELD_VALUE})#{OWS})?\z/.freeze
|
44
44
|
|
45
45
|
VALID_FIELD_NAME = /\A#{FIELD_NAME}\z/.freeze
|
46
46
|
VALID_FIELD_VALUE = /\A#{FIELD_VALUE}\z/.freeze
|
47
47
|
|
48
48
|
DEFAULT_MAXIMUM_LINE_LENGTH = 8192
|
49
49
|
|
50
|
+
# Represents a single HTTP/1.x connection, which may be used to send and receive multiple requests and responses.
|
50
51
|
class Connection
|
51
52
|
CRLF = "\r\n"
|
53
|
+
|
54
|
+
# The HTTP/1.0 version string.
|
52
55
|
HTTP10 = "HTTP/1.0"
|
56
|
+
|
57
|
+
# The HTTP/1.1 version string.
|
53
58
|
HTTP11 = "HTTP/1.1"
|
54
59
|
|
60
|
+
# Initialize the connection with the given stream.
|
61
|
+
#
|
62
|
+
# @parameter stream [IO] the stream to read and write data from.
|
63
|
+
# @parameter persistent [Boolean] whether the connection is persistent.
|
64
|
+
# @parameter state [Symbol] the initial state of the connection, typically idle.
|
55
65
|
def initialize(stream, persistent: true, state: :idle, maximum_line_length: DEFAULT_MAXIMUM_LINE_LENGTH)
|
56
66
|
@stream = stream
|
57
67
|
|
@@ -63,16 +73,12 @@ module Protocol
|
|
63
73
|
@maximum_line_length = maximum_line_length
|
64
74
|
end
|
65
75
|
|
76
|
+
# The underlying IO stream.
|
66
77
|
attr :stream
|
67
78
|
|
68
|
-
#
|
69
|
-
#
|
70
|
-
# the connection can be reused after the response is sent.
|
71
|
-
# This setting is automatically managed according to the nature of the request
|
72
|
-
# and response.
|
73
|
-
# Changing to false is safe.
|
74
|
-
# Changing to true from outside this class should generally be avoided and,
|
75
|
-
# depending on the response semantics, may be reset to false anyway.
|
79
|
+
# @attribute [Boolean] true if the connection is persistent.
|
80
|
+
#
|
81
|
+
# This determines what connection headers are sent in the response and whether the connection can be reused after the response is sent. This setting is automatically managed according to the nature of the request and response. Changing to false is safe. Changing to true from outside this class should generally be avoided and, depending on the response semantics, may be reset to false anyway.
|
76
82
|
attr_accessor :persistent
|
77
83
|
|
78
84
|
# The current state of the connection.
|
@@ -114,29 +120,40 @@ module Protocol
|
|
114
120
|
# State transition methods use a trailing "!".
|
115
121
|
attr_accessor :state
|
116
122
|
|
123
|
+
# @return [Boolean] whether the connection is in the idle state.
|
117
124
|
def idle?
|
118
125
|
@state == :idle
|
119
126
|
end
|
120
127
|
|
128
|
+
# @return [Boolean] whether the connection is in the open state.
|
121
129
|
def open?
|
122
130
|
@state == :open
|
123
131
|
end
|
124
132
|
|
133
|
+
# @return [Boolean] whether the connection is in the half-closed local state.
|
125
134
|
def half_closed_local?
|
126
135
|
@state == :half_closed_local
|
127
136
|
end
|
128
137
|
|
138
|
+
# @return [Boolean] whether the connection is in the half-closed remote state.
|
129
139
|
def half_closed_remote?
|
130
140
|
@state == :half_closed_remote
|
131
141
|
end
|
132
142
|
|
143
|
+
# @return [Boolean] whether the connection is in the closed state.
|
133
144
|
def closed?
|
134
145
|
@state == :closed
|
135
146
|
end
|
136
147
|
|
137
|
-
#
|
148
|
+
# @attribute [Integer] the number of requests and responses processed by this connection.
|
138
149
|
attr :count
|
139
150
|
|
151
|
+
# Indicates whether the connection is persistent given the version, method, and headers.
|
152
|
+
#
|
153
|
+
# @parameter version [String] the HTTP version.
|
154
|
+
# @parameter method [String] the HTTP method.
|
155
|
+
# @parameter headers [Hash] the HTTP headers.
|
156
|
+
# @return [Boolean] whether the connection can be persistent.
|
140
157
|
def persistent?(version, method, headers)
|
141
158
|
if method == HTTP::Methods::CONNECT
|
142
159
|
return false
|
@@ -166,19 +183,21 @@ module Protocol
|
|
166
183
|
end
|
167
184
|
end
|
168
185
|
|
186
|
+
# Write the appropriate header for connection upgrade.
|
169
187
|
def write_upgrade_header(upgrade)
|
170
188
|
@stream.write("connection: upgrade\r\nupgrade: #{upgrade}\r\n")
|
171
189
|
end
|
172
190
|
|
173
|
-
# Indicates whether the connection has been hijacked meaning its
|
174
|
-
#
|
175
|
-
# @
|
191
|
+
# Indicates whether the connection has been hijacked meaning its IO has been handed over and is not usable anymore.
|
192
|
+
#
|
193
|
+
# @returns [Boolean] hijack status
|
176
194
|
def hijacked?
|
177
195
|
@stream.nil?
|
178
196
|
end
|
179
197
|
|
180
|
-
#
|
181
|
-
#
|
198
|
+
# Hijack the connection - that is, take over the underlying IO and close the connection.
|
199
|
+
#
|
200
|
+
# @returns [IO | Nil] the underlying non-blocking IO.
|
182
201
|
def hijack!
|
183
202
|
@persistent = false
|
184
203
|
|
@@ -193,13 +212,14 @@ module Protocol
|
|
193
212
|
end
|
194
213
|
end
|
195
214
|
|
215
|
+
# Close the read end of the connection and transition to the half-closed remote state (or closed if already in the half-closed local state).
|
196
216
|
def close_read
|
197
217
|
@persistent = false
|
198
218
|
@stream&.close_read
|
199
219
|
self.receive_end_stream!
|
200
220
|
end
|
201
221
|
|
202
|
-
# Close the connection and underlying stream.
|
222
|
+
# Close the connection and underlying stream and transition to the closed state.
|
203
223
|
def close(error = nil)
|
204
224
|
@persistent = false
|
205
225
|
|
@@ -214,6 +234,9 @@ module Protocol
|
|
214
234
|
end
|
215
235
|
end
|
216
236
|
|
237
|
+
# Force a transition to the open state.
|
238
|
+
#
|
239
|
+
# @raises [ProtocolError] if the connection is not in the idle state.
|
217
240
|
def open!
|
218
241
|
unless @state == :idle
|
219
242
|
raise ProtocolError, "Cannot open connection in state: #{@state}!"
|
@@ -224,6 +247,16 @@ module Protocol
|
|
224
247
|
return self
|
225
248
|
end
|
226
249
|
|
250
|
+
# Write a request to the connection. It is expected you will write the body after this method.
|
251
|
+
#
|
252
|
+
# Transitions to the open state.
|
253
|
+
#
|
254
|
+
# @parameter authority [String] the authority of the request.
|
255
|
+
# @parameter method [String] the HTTP method.
|
256
|
+
# @parameter target [String] the request target.
|
257
|
+
# @parameter version [String] the HTTP version.
|
258
|
+
# @parameter headers [Hash] the HTTP headers.
|
259
|
+
# @raises [ProtocolError] if the connection is not in the idle state.
|
227
260
|
def write_request(authority, method, target, version, headers)
|
228
261
|
open!
|
229
262
|
|
@@ -233,6 +266,12 @@ module Protocol
|
|
233
266
|
write_headers(headers)
|
234
267
|
end
|
235
268
|
|
269
|
+
# Write a response to the connection. It is expected you will write the body after this method.
|
270
|
+
#
|
271
|
+
# @parameter version [String] the HTTP version.
|
272
|
+
# @parameter status [Integer] the HTTP status code.
|
273
|
+
# @parameter headers [Hash] the HTTP headers.
|
274
|
+
# @parameter reason [String] the reason phrase, defaults to the standard reason phrase for the status code.
|
236
275
|
def write_response(version, status, headers, reason = Reason::DESCRIPTIONS[status])
|
237
276
|
unless @state == :open or @state == :half_closed_remote
|
238
277
|
raise ProtocolError, "Cannot write response in state: #{@state}!"
|
@@ -244,6 +283,13 @@ module Protocol
|
|
244
283
|
write_headers(headers)
|
245
284
|
end
|
246
285
|
|
286
|
+
# Write an interim response to the connection. It is expected you will eventually write the final response after this method.
|
287
|
+
#
|
288
|
+
# @parameter version [String] the HTTP version.
|
289
|
+
# @parameter status [Integer] the HTTP status code.
|
290
|
+
# @parameter headers [Hash] the HTTP headers.
|
291
|
+
# @parameter reason [String] the reason phrase, defaults to the standard reason phrase for the status code.
|
292
|
+
# @raises [ProtocolError] if the connection is not in the open or half-closed remote state.
|
247
293
|
def write_interim_response(version, status, headers, reason = Reason::DESCRIPTIONS[status])
|
248
294
|
unless @state == :open or @state == :half_closed_remote
|
249
295
|
raise ProtocolError, "Cannot write interim response in state: #{@state}!"
|
@@ -257,6 +303,10 @@ module Protocol
|
|
257
303
|
@stream.flush
|
258
304
|
end
|
259
305
|
|
306
|
+
# Write headers to the connection.
|
307
|
+
#
|
308
|
+
# @parameter headers [Hash] the headers to write.
|
309
|
+
# @raises [BadHeader] if the header name or value is invalid.
|
260
310
|
def write_headers(headers)
|
261
311
|
headers.each do |name, value|
|
262
312
|
# Convert it to a string:
|
@@ -277,14 +327,25 @@ module Protocol
|
|
277
327
|
end
|
278
328
|
end
|
279
329
|
|
330
|
+
# Read some data from the connection.
|
331
|
+
#
|
332
|
+
# @parameter length [Integer] the maximum number of bytes to read.
|
280
333
|
def readpartial(length)
|
281
334
|
@stream.readpartial(length)
|
282
335
|
end
|
283
336
|
|
337
|
+
# Read some data from the connection.
|
338
|
+
#
|
339
|
+
# @parameter length [Integer] the number of bytes to read.
|
284
340
|
def read(length)
|
285
341
|
@stream.read(length)
|
286
342
|
end
|
287
343
|
|
344
|
+
# Read a line from the connection.
|
345
|
+
#
|
346
|
+
# @returns [String | Nil] the line read, or nil if the connection is closed.
|
347
|
+
# @raises [EOFError] if the connection is closed.
|
348
|
+
# @raises [LineLengthError] if the line is too long.
|
288
349
|
def read_line?
|
289
350
|
if line = @stream.gets(CRLF, @maximum_line_length)
|
290
351
|
unless line.chomp!(CRLF)
|
@@ -296,10 +357,17 @@ module Protocol
|
|
296
357
|
return line
|
297
358
|
end
|
298
359
|
|
360
|
+
# Read a line from the connection.
|
361
|
+
#
|
362
|
+
# @raises [EOFError] if a line could not be read.
|
363
|
+
# @raises [LineLengthError] if the line is too long.
|
299
364
|
def read_line
|
300
365
|
read_line? or raise EOFError
|
301
366
|
end
|
302
367
|
|
368
|
+
# Read a request line from the connection.
|
369
|
+
#
|
370
|
+
# @returns [Tuple(String, String, String) | Nil] the method, path, and version of the request, or nil if the connection is closed.
|
303
371
|
def read_request_line
|
304
372
|
return unless line = read_line?
|
305
373
|
|
@@ -312,6 +380,13 @@ module Protocol
|
|
312
380
|
return method, path, version
|
313
381
|
end
|
314
382
|
|
383
|
+
# Read a request from the connection, including the request line and request headers, and prepares to read the request body.
|
384
|
+
#
|
385
|
+
# Transitions to the open state.
|
386
|
+
#
|
387
|
+
# @yields {|host, method, path, version, headers, body| ...} if a block is given.
|
388
|
+
# @returns [Tuple(String, String, String, String, HTTP::Headers, Protocol::HTTP1::Body) | Nil] the host, method, path, version, headers, and body of the request, or `nil` if the connection is closed.
|
389
|
+
# @raises [ProtocolError] if the connection is not in the idle state.
|
315
390
|
def read_request
|
316
391
|
open!
|
317
392
|
|
@@ -341,6 +416,10 @@ module Protocol
|
|
341
416
|
end
|
342
417
|
end
|
343
418
|
|
419
|
+
# Read a response line from the connection.
|
420
|
+
#
|
421
|
+
# @returns [Tuple(String, Integer, String)] the version, status, and reason of the response.
|
422
|
+
# @raises [EOFError] if the connection is closed.
|
344
423
|
def read_response_line
|
345
424
|
version, status, reason = read_line.split(/\s+/, 3)
|
346
425
|
|
@@ -349,10 +428,21 @@ module Protocol
|
|
349
428
|
return version, status, reason
|
350
429
|
end
|
351
430
|
|
352
|
-
|
431
|
+
# Indicates whether the status code is an interim status code.
|
432
|
+
#
|
433
|
+
# @parameter status [Integer] the status code.
|
434
|
+
# @returns [Boolean] whether the status code is an interim status code.
|
435
|
+
def interim_status?(status)
|
353
436
|
status != 101 and status >= 100 and status < 200
|
354
437
|
end
|
355
438
|
|
439
|
+
# Read a response from the connection.
|
440
|
+
#
|
441
|
+
# @parameter method [String] the HTTP method.
|
442
|
+
# @yields {|version, status, reason, headers, body| ...} if a block is given.
|
443
|
+
# @returns [Tuple(String, Integer, String, HTTP::Headers, Protocol::HTTP1::Body)] the version, status, reason, headers, and body of the response.
|
444
|
+
# @raises [ProtocolError] if the connection is not in the open or half-closed local state.
|
445
|
+
# @raises [EOFError] if the connection is closed.
|
356
446
|
def read_response(method)
|
357
447
|
unless @state == :open or @state == :half_closed_local
|
358
448
|
raise ProtocolError, "Cannot read response in state: #{@state}!"
|
@@ -383,6 +473,11 @@ module Protocol
|
|
383
473
|
end
|
384
474
|
end
|
385
475
|
|
476
|
+
# Read headers from the connection until an empty line is encountered.
|
477
|
+
#
|
478
|
+
# @returns [HTTP::Headers] the headers read.
|
479
|
+
# @raises [EOFError] if the connection is closed.
|
480
|
+
# @raises [BadHeader] if a header could not be parsed.
|
386
481
|
def read_headers
|
387
482
|
fields = []
|
388
483
|
|
@@ -391,7 +486,7 @@ module Protocol
|
|
391
486
|
break if line.empty?
|
392
487
|
|
393
488
|
if match = line.match(HEADER)
|
394
|
-
fields << [match[1], match[2]]
|
489
|
+
fields << [match[1], match[2] || ""]
|
395
490
|
else
|
396
491
|
raise BadHeader, "Could not parse header: #{line.inspect}"
|
397
492
|
end
|
@@ -400,6 +495,11 @@ module Protocol
|
|
400
495
|
return HTTP::Headers.new(fields)
|
401
496
|
end
|
402
497
|
|
498
|
+
# Transition to the half-closed local state, in other words, the connection is closed for writing.
|
499
|
+
#
|
500
|
+
# If the connection is already in the half-closed remote state, it will transition to the closed state.
|
501
|
+
#
|
502
|
+
# @raises [ProtocolError] if the connection is not in the open state.
|
403
503
|
def send_end_stream!
|
404
504
|
if @state == :open
|
405
505
|
@state = :half_closed_local
|
@@ -410,7 +510,15 @@ module Protocol
|
|
410
510
|
end
|
411
511
|
end
|
412
512
|
|
413
|
-
#
|
513
|
+
# Write an upgrade body to the connection.
|
514
|
+
#
|
515
|
+
# This writes the upgrade header and the body to the connection. If the body is `nil`, you should coordinate writing to the stream.
|
516
|
+
#
|
517
|
+
# The connection will not be persistent after this method is called.
|
518
|
+
#
|
519
|
+
# @parameter protocol [String] the protocol to upgrade to.
|
520
|
+
# @parameter body [Object | Nil] the body to write.
|
521
|
+
# @returns [IO] the underlying IO stream.
|
414
522
|
def write_upgrade_body(protocol, body = nil)
|
415
523
|
# Once we upgrade the connection, it can no longer handle other requests:
|
416
524
|
@persistent = false
|
@@ -434,6 +542,15 @@ module Protocol
|
|
434
542
|
self.send_end_stream!
|
435
543
|
end
|
436
544
|
|
545
|
+
# Write a tunnel body to the connection.
|
546
|
+
#
|
547
|
+
# This writes the connection header and the body to the connection. If the body is `nil`, you should coordinate writing to the stream.
|
548
|
+
#
|
549
|
+
# The connection will not be persistent after this method is called.
|
550
|
+
#
|
551
|
+
# @parameter version [String] the HTTP version.
|
552
|
+
# @parameter body [Object | Nil] the body to write.
|
553
|
+
# @returns [IO] the underlying IO stream.
|
437
554
|
def write_tunnel_body(version, body = nil)
|
438
555
|
@persistent = false
|
439
556
|
|
@@ -456,7 +573,12 @@ module Protocol
|
|
456
573
|
self.send_end_stream!
|
457
574
|
end
|
458
575
|
|
459
|
-
|
576
|
+
# Write an empty body to the connection.
|
577
|
+
#
|
578
|
+
# If given, the body will be closed.
|
579
|
+
#
|
580
|
+
# @parameter body [Object | Nil] the body to write.
|
581
|
+
def write_empty_body(body = nil)
|
460
582
|
@stream.write("content-length: 0\r\n\r\n")
|
461
583
|
@stream.flush
|
462
584
|
|
@@ -465,6 +587,14 @@ module Protocol
|
|
465
587
|
self.send_end_stream!
|
466
588
|
end
|
467
589
|
|
590
|
+
# Write a fixed length body to the connection.
|
591
|
+
#
|
592
|
+
# If the request was a `HEAD` request, the body will be closed, and no data will be written.
|
593
|
+
#
|
594
|
+
# @parameter body [Object] the body to write.
|
595
|
+
# @parameter length [Integer] the length of the body.
|
596
|
+
# @parameter head [Boolean] whether the request was a `HEAD` request.
|
597
|
+
# @raises [ContentLengthError] if the body length does not match the content length specified.
|
468
598
|
def write_fixed_length_body(body, length, head)
|
469
599
|
@stream.write("content-length: #{length}\r\n\r\n")
|
470
600
|
|
@@ -499,6 +629,15 @@ module Protocol
|
|
499
629
|
self.send_end_stream!
|
500
630
|
end
|
501
631
|
|
632
|
+
# Write a chunked body to the connection.
|
633
|
+
#
|
634
|
+
# If the request was a `HEAD` request, the body will be closed, and no data will be written.
|
635
|
+
#
|
636
|
+
# If trailers are given, they will be written after the body.
|
637
|
+
#
|
638
|
+
# @parameter body [Object] the body to write.
|
639
|
+
# @parameter head [Boolean] whether the request was a `HEAD` request.
|
640
|
+
# @parameter trailer [Hash | Nil] the trailers to write.
|
502
641
|
def write_chunked_body(body, head, trailer = nil)
|
503
642
|
@stream.write("transfer-encoding: chunked\r\n\r\n")
|
504
643
|
|
@@ -535,6 +674,10 @@ module Protocol
|
|
535
674
|
self.send_end_stream!
|
536
675
|
end
|
537
676
|
|
677
|
+
# Write the body to the connection and close the connection.
|
678
|
+
#
|
679
|
+
# @parameter body [Object] the body to write.
|
680
|
+
# @parameter head [Boolean] whether the request was a `HEAD` request.
|
538
681
|
def write_body_and_close(body, head)
|
539
682
|
# We can't be persistent because we don't know the data length:
|
540
683
|
@persistent = false
|
@@ -559,6 +702,10 @@ module Protocol
|
|
559
702
|
end
|
560
703
|
|
561
704
|
# The connection (stream) was closed. It may now be in the idle state.
|
705
|
+
#
|
706
|
+
# Sub-classes may override this method to perform additional cleanup.
|
707
|
+
#
|
708
|
+
# @parameter error [Exception | Nil] the error that caused the connection to be closed, if any.
|
562
709
|
def closed(error = nil)
|
563
710
|
end
|
564
711
|
|
@@ -578,6 +725,14 @@ module Protocol
|
|
578
725
|
self.closed(error)
|
579
726
|
end
|
580
727
|
|
728
|
+
# Write a body to the connection.
|
729
|
+
#
|
730
|
+
# The behavior of this method is determined by the HTTP version, the body, and the request method. We try to choose the best approach possible, given the constraints, connection persistence, whether the length is known, etc.
|
731
|
+
#
|
732
|
+
# @parameter version [String] the HTTP version.
|
733
|
+
# @parameter body [Object] the body to write.
|
734
|
+
# @parameter head [Boolean] whether the request was a `HEAD` request.
|
735
|
+
# @parameter trailer [Hash | Nil] the trailers to write.
|
581
736
|
def write_body(version, body, head = false, trailer = nil)
|
582
737
|
# HTTP/1.0 cannot in any case handle trailers.
|
583
738
|
if version == HTTP10 # or te: trailers was not present (strictly speaking not required.)
|
@@ -609,6 +764,11 @@ module Protocol
|
|
609
764
|
end
|
610
765
|
end
|
611
766
|
|
767
|
+
# Indicate that the end of the stream (body) has been received.
|
768
|
+
#
|
769
|
+
# This will transition to the half-closed remote state if the connection is open, or the closed state if the connection is half-closed local.
|
770
|
+
#
|
771
|
+
# @raises [ProtocolError] if the connection is not in the open or half-closed remote state.
|
612
772
|
def receive_end_stream!
|
613
773
|
if @state == :open
|
614
774
|
@state = :half_closed_remote
|
@@ -619,19 +779,34 @@ module Protocol
|
|
619
779
|
end
|
620
780
|
end
|
621
781
|
|
782
|
+
# Read the body, assuming it is using the chunked transfer encoding.
|
783
|
+
#
|
784
|
+
# @parameters headers [Hash] the headers of the request.
|
785
|
+
# @returns [Protocol::HTTP1::Body::Chunked] the body.
|
622
786
|
def read_chunked_body(headers)
|
623
787
|
Body::Chunked.new(self, headers)
|
624
788
|
end
|
625
789
|
|
790
|
+
# Read the body, assuming it has a fixed length.
|
791
|
+
#
|
792
|
+
# @parameters length [Integer] the length of the body.
|
793
|
+
# @returns [Protocol::HTTP1::Body::Fixed] the body.
|
626
794
|
def read_fixed_body(length)
|
627
795
|
Body::Fixed.new(self, length)
|
628
796
|
end
|
629
797
|
|
798
|
+
# Read the body, assuming that we read until the connection is closed.
|
799
|
+
#
|
800
|
+
# @returns [Protocol::HTTP1::Body::Remainder] the body.
|
630
801
|
def read_remainder_body
|
631
802
|
@persistent = false
|
632
803
|
Body::Remainder.new(self)
|
633
804
|
end
|
634
805
|
|
806
|
+
# Read the body, assuming that we are not receiving any actual data, but just the length.
|
807
|
+
#
|
808
|
+
# @parameters length [Integer] the length of the body.
|
809
|
+
# @returns [Protocol::HTTP::Body::Head] the body.
|
635
810
|
def read_head_body(length)
|
636
811
|
# We are not receiving any body:
|
637
812
|
self.receive_end_stream!
|
@@ -639,21 +814,41 @@ module Protocol
|
|
639
814
|
Protocol::HTTP::Body::Head.new(length)
|
640
815
|
end
|
641
816
|
|
817
|
+
# Read the body, assuming it is a tunnel.
|
818
|
+
#
|
819
|
+
# Invokes {read_remainder_body}.
|
820
|
+
#
|
821
|
+
# @returns [Protocol::HTTP::Body::Remainder] the body.
|
642
822
|
def read_tunnel_body
|
643
823
|
read_remainder_body
|
644
824
|
end
|
645
825
|
|
826
|
+
# Read the body, assuming it is an upgrade.
|
827
|
+
#
|
828
|
+
# Invokes {read_remainder_body}.
|
829
|
+
#
|
830
|
+
# @returns [Protocol::HTTP::Body::Remainder] the body.
|
646
831
|
def read_upgrade_body
|
647
832
|
# When you have an incoming upgrade request body, we must be extremely careful not to start reading it until the upgrade has been confirmed, otherwise if the upgrade was rejected and we started forwarding the incoming request body, it would desynchronize the connection (potential security issue).
|
648
833
|
# We mitigate this issue by setting @persistent to false, which will prevent the connection from being reused, even if the upgrade fails (potential performance issue).
|
649
834
|
read_remainder_body
|
650
835
|
end
|
651
836
|
|
837
|
+
# The HTTP `HEAD` method.
|
652
838
|
HEAD = "HEAD"
|
839
|
+
|
840
|
+
# The HTTP `CONNECT` method.
|
653
841
|
CONNECT = "CONNECT"
|
654
842
|
|
843
|
+
# The pattern for valid content length values.
|
655
844
|
VALID_CONTENT_LENGTH = /\A\d+\z/
|
656
845
|
|
846
|
+
# Extract the content length from the headers, if possible.
|
847
|
+
#
|
848
|
+
# @parameter headers [Hash] the headers.
|
849
|
+
# @yields {|length| ...} if a content length is found.
|
850
|
+
# @parameter length [Integer] the content length.
|
851
|
+
# @raises [BadRequest] if the content length is invalid.
|
657
852
|
def extract_content_length(headers)
|
658
853
|
if content_length = headers.delete(CONTENT_LENGTH)
|
659
854
|
if content_length =~ VALID_CONTENT_LENGTH
|
@@ -664,6 +859,17 @@ module Protocol
|
|
664
859
|
end
|
665
860
|
end
|
666
861
|
|
862
|
+
# Read the body of the response.
|
863
|
+
#
|
864
|
+
# - The `HEAD` method is used to retrieve the headers of the response without the body, so {read_head_body} is invoked if there is a content length, otherwise nil is returned.
|
865
|
+
# - A 101 status code indicates that the connection will be upgraded, so {read_upgrade_body} is invoked.
|
866
|
+
# - Interim status codes (1xx), no content (204) and not modified (304) status codes do not have a body, so nil is returned.
|
867
|
+
# - The `CONNECT` method is used to establish a tunnel, so {read_tunnel_body} is invoked.
|
868
|
+
# - Otherwise, the body is read according to {read_body}.
|
869
|
+
#
|
870
|
+
# @parameter method [String] the HTTP method.
|
871
|
+
# @parameter status [Integer] the HTTP status code.
|
872
|
+
# @parameter headers [Hash] the headers of the response.
|
667
873
|
def read_response_body(method, status, headers)
|
668
874
|
# RFC 7230 3.3.3
|
669
875
|
# 1. Any response to a HEAD request and any response with a 1xx
|
@@ -704,6 +910,14 @@ module Protocol
|
|
704
910
|
return read_body(headers, true)
|
705
911
|
end
|
706
912
|
|
913
|
+
# Read the body of the request.
|
914
|
+
#
|
915
|
+
# - The `CONNECT` method is used to establish a tunnel, so the body is read until the connection is closed.
|
916
|
+
# - The `UPGRADE` method is used to upgrade the connection to a different protocol (typically WebSockets), so the body is read until the connection is closed.
|
917
|
+
# - Otherwise, the body is read according to {read_body}.
|
918
|
+
#
|
919
|
+
# @parameter method [String] the HTTP method.
|
920
|
+
# @parameter headers [Hash] the headers of the request.
|
707
921
|
def read_request_body(method, headers)
|
708
922
|
# 2. Any 2xx (Successful) response to a CONNECT request implies that
|
709
923
|
# the connection will become a tunnel immediately after the empty
|
@@ -724,6 +938,15 @@ module Protocol
|
|
724
938
|
return read_body(headers)
|
725
939
|
end
|
726
940
|
|
941
|
+
# Read the body of the message.
|
942
|
+
#
|
943
|
+
# - The `transfer-encoding` header is used to determine if the body is chunked.
|
944
|
+
# - Otherwise, if the `content-length` is present, the body is read until the content length is reached.
|
945
|
+
# - Otherwise, if `remainder` is true, the body is read until the connection is closed.
|
946
|
+
#
|
947
|
+
# @parameter headers [Hash] the headers of the message.
|
948
|
+
# @parameter remainder [Boolean] whether to read the remainder of the body.
|
949
|
+
# @returns [Object] the body.
|
727
950
|
def read_body(headers, remainder = false)
|
728
951
|
# 3. If a Transfer-Encoding header field is present and the chunked
|
729
952
|
# transfer coding (Section 4.1) is the final encoding, the message
|
data/lib/protocol/http1/error.rb
CHANGED
@@ -1,12 +1,13 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
# Released under the MIT License.
|
4
|
-
# Copyright, 2019-
|
4
|
+
# Copyright, 2019-2025, by Samuel Williams.
|
5
5
|
|
6
6
|
require "protocol/http/error"
|
7
7
|
|
8
8
|
module Protocol
|
9
9
|
module HTTP1
|
10
|
+
# The base class for all HTTP/1.x errors.
|
10
11
|
class Error < HTTP::Error
|
11
12
|
end
|
12
13
|
|
@@ -14,6 +15,7 @@ module Protocol
|
|
14
15
|
class ProtocolError < Error
|
15
16
|
end
|
16
17
|
|
18
|
+
# The request line was too long.
|
17
19
|
class LineLengthError < Error
|
18
20
|
end
|
19
21
|
|
@@ -1,13 +1,15 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
# Released under the MIT License.
|
4
|
-
# Copyright, 2019-
|
4
|
+
# Copyright, 2019-2025, by Samuel Williams.
|
5
5
|
|
6
6
|
require "protocol/http/error"
|
7
7
|
|
8
8
|
module Protocol
|
9
9
|
module HTTP1
|
10
|
+
# Reason phrases for HTTP status codes.
|
10
11
|
module Reason
|
12
|
+
# Get the reason phrase for the given status code.
|
11
13
|
DESCRIPTIONS = {
|
12
14
|
100 => "Continue",
|
13
15
|
101 => "Switching Protocols",
|
data/lib/protocol/http1.rb
CHANGED
@@ -1,7 +1,14 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
# Released under the MIT License.
|
4
|
-
# Copyright, 2019-
|
4
|
+
# Copyright, 2019-2025, by Samuel Williams.
|
5
5
|
|
6
6
|
require_relative "http1/version"
|
7
7
|
require_relative "http1/connection"
|
8
|
+
|
9
|
+
# @namespace
|
10
|
+
module Protocol
|
11
|
+
# @namespace
|
12
|
+
module HTTP1
|
13
|
+
end
|
14
|
+
end
|
data.tar.gz.sig
CHANGED
Binary file
|
metadata
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: protocol-http1
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.33.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Samuel Williams
|
@@ -41,7 +41,7 @@ cert_chain:
|
|
41
41
|
Q2K9NVun/S785AP05vKkXZEFYxqG6EW012U4oLcFl5MySFajYXRYbuUpH6AY+HP8
|
42
42
|
voD0MPg1DssDLKwXyt1eKD/+Fq0bFWhwVM/1XiAXL7lyYUyOq24KHgQ2Csg=
|
43
43
|
-----END CERTIFICATE-----
|
44
|
-
date: 2025-03-
|
44
|
+
date: 2025-03-20 00:00:00.000000000 Z
|
45
45
|
dependencies:
|
46
46
|
- !ruby/object:Gem::Dependency
|
47
47
|
name: protocol-http
|
@@ -62,6 +62,7 @@ extensions: []
|
|
62
62
|
extra_rdoc_files: []
|
63
63
|
files:
|
64
64
|
- lib/protocol/http1.rb
|
65
|
+
- lib/protocol/http1/body.rb
|
65
66
|
- lib/protocol/http1/body/chunked.rb
|
66
67
|
- lib/protocol/http1/body/fixed.rb
|
67
68
|
- lib/protocol/http1/body/remainder.rb
|
metadata.gz.sig
CHANGED
Binary file
|