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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 29db57e08ceb65831410a015b07b643fe3223c119327f001485c4d942195516e
4
- data.tar.gz: 01b83de5925e1d89d4ffb729f22bab647e4fa5ac04d239a2ff81899bacb395f9
3
+ metadata.gz: d5e39284335ca68719e14bdd3c85f2d9ca6c7fa28939f2625149dc85aa65d2e8
4
+ data.tar.gz: f75598dfa2d6a89c1738c49bbfa76d627cb9badb2220254b4c82b489564f424a
5
5
  SHA512:
6
- metadata.gz: 564dcdb26845995de2c5f05b0b8e6c464615a6cc7345685c30e5d28bca09ccd03f3b58075da67e7afb3aab5ecb8213db76fb3fad1e59643899f68e0415ca915b
7
- data.tar.gz: a10dbcdcf55d5687eb3e94476869b2ce4af1502a056e5f70b4dfb1c751ba2bf2bad46e11cd8d8ae666314390eec95daef1c39a5ad93debab744718fadd568535
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-2024, by Samuel Williams.
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 everything. This is because the length is not known until the final chunk is 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-2024, by Samuel Williams.
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
- # @raises EOFError if the connection is closed before the expected length is read.
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-2024, by Samuel Williams.
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
- # A body that reads all remaining data from the connection.
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
- # block_size may be removed in the future. It is better managed by connection.
16
- def initialize(connection)
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(BLOCK_SIZE)
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
- FIELD_VALUE = /[^\000-\037]*/.freeze
43
- HEADER = /\A(#{FIELD_NAME}):\s*(#{FIELD_VALUE})\s*\z/.freeze
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
- # Whether the connection is persistent.
69
- # This determines what connection headers are sent in the response and whether
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
- # The number of requests processed.
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
- # IO has been handed over and is not usable anymore.
175
- # @return [Boolean] hijack status
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
- # Effectively close the connection and return the underlying IO.
181
- # @return [IO] the underlying non-blocking IO.
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
- private def interim_status?(status)
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
- # @param protocol [String] the protocol to upgrade to.
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
- def write_empty_body(body)
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
@@ -1,12 +1,13 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  # Released under the MIT License.
4
- # Copyright, 2019-2024, by Samuel Williams.
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-2024, by Samuel Williams.
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",
@@ -5,6 +5,6 @@
5
5
 
6
6
  module Protocol
7
7
  module HTTP1
8
- VERSION = "0.31.0"
8
+ VERSION = "0.33.0"
9
9
  end
10
10
  end
@@ -1,7 +1,14 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  # Released under the MIT License.
4
- # Copyright, 2019-2024, by Samuel Williams.
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.31.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-13 00:00:00.000000000 Z
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