protocol-http1 0.20.0 → 0.22.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 +44 -34
- data/lib/protocol/http1/body/fixed.rb +20 -18
- data/lib/protocol/http1/body/remainder.rb +10 -24
- data/lib/protocol/http1/connection.rb +20 -4
- data/lib/protocol/http1/version.rb +1 -1
- data/license.md +1 -1
- data.tar.gz.sig +0 -0
- metadata +2 -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: 550b6e9b8f04f40fbba30a9260ead61bc1921130b069f8d4f6d7005077da0e61
         | 
| 4 | 
            +
              data.tar.gz: 54a8009f974a0292b1500c21f7e8047d5af4d815e7b844fa46ec0d6f6d356fee
         | 
| 5 5 | 
             
            SHA512:
         | 
| 6 | 
            -
              metadata.gz:  | 
| 7 | 
            -
              data.tar.gz:  | 
| 6 | 
            +
              metadata.gz: e44973d7e04933d1e6d3e8d50c98dcc57603fc4cd7491a1ab9ed490ef659551bbf1e0eb425222cc2e61f38b9fc5af864198970505a42750dad8c97bdb68331e7
         | 
| 7 | 
            +
              data.tar.gz: 6fba9bb8ac29bc13966bbebed00ca72b60d9214ba1943185e44760abb72ad5b1fa8254f4222951e5bf983263fef015a78ecdca6e6c8396c3c0e9548bade51576
         | 
    
        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-2024, by Samuel Williams.
         | 
| 5 5 | 
             
            # Copyright, 2023, by Thomas Morgan.
         | 
| 6 6 |  | 
| 7 7 | 
             
            require 'protocol/http/body/readable'
         | 
| @@ -23,14 +23,17 @@ module Protocol | |
| 23 23 | 
             
            				end
         | 
| 24 24 |  | 
| 25 25 | 
             
            				def empty?
         | 
| 26 | 
            -
            					@ | 
| 26 | 
            +
            					@stream.nil?
         | 
| 27 27 | 
             
            				end
         | 
| 28 28 |  | 
| 29 29 | 
             
            				def close(error = nil)
         | 
| 30 | 
            -
            					 | 
| 31 | 
            -
             | 
| 32 | 
            -
            						@ | 
| 33 | 
            -
             | 
| 30 | 
            +
            					if @stream
         | 
| 31 | 
            +
            						# We only close the connection if we haven't completed reading the entire body:
         | 
| 32 | 
            +
            						unless @finished
         | 
| 33 | 
            +
            							@stream.close_read
         | 
| 34 | 
            +
            						end
         | 
| 35 | 
            +
            						
         | 
| 36 | 
            +
            						@stream = nil
         | 
| 34 37 | 
             
            					end
         | 
| 35 38 |  | 
| 36 39 | 
             
            					super
         | 
| @@ -40,35 +43,42 @@ module Protocol | |
| 40 43 |  | 
| 41 44 | 
             
            				# Follows the procedure outlined in https://tools.ietf.org/html/rfc7230#section-4.1.3
         | 
| 42 45 | 
             
            				def read
         | 
| 43 | 
            -
            					 | 
| 44 | 
            -
             | 
| 45 | 
            -
             | 
| 46 | 
            -
             | 
| 47 | 
            -
             | 
| 48 | 
            -
             | 
| 49 | 
            -
             | 
| 50 | 
            -
             | 
| 51 | 
            -
             | 
| 52 | 
            -
             | 
| 53 | 
            -
             | 
| 54 | 
            -
             | 
| 55 | 
            -
             | 
| 56 | 
            -
             | 
| 57 | 
            -
             | 
| 46 | 
            +
            					if !@finished
         | 
| 47 | 
            +
            						if @stream
         | 
| 48 | 
            +
            							length, _extensions = read_line.split(";", 2)
         | 
| 49 | 
            +
            							
         | 
| 50 | 
            +
            							unless length =~ VALID_CHUNK_LENGTH
         | 
| 51 | 
            +
            								raise BadRequest, "Invalid chunk length: #{length.inspect}"
         | 
| 52 | 
            +
            							end
         | 
| 53 | 
            +
            							
         | 
| 54 | 
            +
            							# It is possible this line contains chunk extension, so we use `to_i` to only consider the initial integral part:
         | 
| 55 | 
            +
            							length = Integer(length, 16)
         | 
| 56 | 
            +
            							
         | 
| 57 | 
            +
            							if length == 0
         | 
| 58 | 
            +
            								read_trailer
         | 
| 59 | 
            +
            								
         | 
| 60 | 
            +
            								# The final chunk has been read and the stream is now closed:
         | 
| 61 | 
            +
            								@stream = nil
         | 
| 62 | 
            +
            								@finished = true
         | 
| 63 | 
            +
            								
         | 
| 64 | 
            +
            								return nil
         | 
| 65 | 
            +
            							end
         | 
| 66 | 
            +
            							
         | 
| 67 | 
            +
            							# Read trailing CRLF:
         | 
| 68 | 
            +
            							chunk = @stream.read(length + 2)
         | 
| 69 | 
            +
            							
         | 
| 70 | 
            +
            							# ...and chomp it off:
         | 
| 71 | 
            +
            							chunk.chomp!(CRLF)
         | 
| 72 | 
            +
            							
         | 
| 73 | 
            +
            							@length += length
         | 
| 74 | 
            +
            							@count += 1
         | 
| 75 | 
            +
            							
         | 
| 76 | 
            +
            							return chunk
         | 
| 77 | 
            +
            						end
         | 
| 58 78 |  | 
| 59 | 
            -
            						 | 
| 79 | 
            +
            						# If the stream has been closed before we have read the final chunk, raise an error:
         | 
| 80 | 
            +
            						raise EOFError, "Stream closed before expected length was read!"
         | 
| 60 81 | 
             
            					end
         | 
| 61 | 
            -
            					
         | 
| 62 | 
            -
            					# Read trailing CRLF:
         | 
| 63 | 
            -
            					chunk = @stream.read(length + 2)
         | 
| 64 | 
            -
            					
         | 
| 65 | 
            -
            					# ...and chomp it off:
         | 
| 66 | 
            -
            					chunk.chomp!(CRLF)
         | 
| 67 | 
            -
            					
         | 
| 68 | 
            -
            					@length += length
         | 
| 69 | 
            -
            					@count += 1
         | 
| 70 | 
            -
            					
         | 
| 71 | 
            -
            					return chunk
         | 
| 72 82 | 
             
            				end
         | 
| 73 83 |  | 
| 74 84 | 
             
            				def inspect
         | 
| @@ -93,7 +103,7 @@ module Protocol | |
| 93 103 | 
             
            						if match = line.match(HEADER)
         | 
| 94 104 | 
             
            							@headers.add(match[1], match[2])
         | 
| 95 105 | 
             
            						else
         | 
| 96 | 
            -
            							raise BadHeader, "Could not parse header: #{line. | 
| 106 | 
            +
            							raise BadHeader, "Could not parse header: #{line.inspect}"
         | 
| 97 107 | 
             
            						end
         | 
| 98 108 | 
             
            					end
         | 
| 99 109 | 
             
            				end
         | 
| @@ -11,6 +11,7 @@ module Protocol | |
| 11 11 | 
             
            			class Fixed < HTTP::Body::Readable
         | 
| 12 12 | 
             
            				def initialize(stream, length)
         | 
| 13 13 | 
             
            					@stream = stream
         | 
| 14 | 
            +
            					
         | 
| 14 15 | 
             
            					@length = length
         | 
| 15 16 | 
             
            					@remaining = length
         | 
| 16 17 | 
             
            				end
         | 
| @@ -19,13 +20,17 @@ module Protocol | |
| 19 20 | 
             
            				attr :remaining
         | 
| 20 21 |  | 
| 21 22 | 
             
            				def empty?
         | 
| 22 | 
            -
            					@remaining == 0
         | 
| 23 | 
            +
            					@stream.nil? or @remaining == 0
         | 
| 23 24 | 
             
            				end
         | 
| 24 25 |  | 
| 25 26 | 
             
            				def close(error = nil)
         | 
| 26 | 
            -
            					 | 
| 27 | 
            -
             | 
| 28 | 
            -
            						@ | 
| 27 | 
            +
            					if @stream
         | 
| 28 | 
            +
            						# If we are closing the body without fully reading it, the underlying connection is now in an undefined state.
         | 
| 29 | 
            +
            						if @remaining != 0
         | 
| 30 | 
            +
            							@stream.close_read
         | 
| 31 | 
            +
            						end
         | 
| 32 | 
            +
            						
         | 
| 33 | 
            +
            						@stream = nil
         | 
| 29 34 | 
             
            					end
         | 
| 30 35 |  | 
| 31 36 | 
             
            					super
         | 
| @@ -34,25 +39,22 @@ module Protocol | |
| 34 39 | 
             
            				# @raises EOFError if the stream is closed before the expected length is read.
         | 
| 35 40 | 
             
            				def read
         | 
| 36 41 | 
             
            					if @remaining > 0
         | 
| 37 | 
            -
            						 | 
| 38 | 
            -
             | 
| 39 | 
            -
            							 | 
| 40 | 
            -
             | 
| 41 | 
            -
             | 
| 42 | 
            +
            						if @stream
         | 
| 43 | 
            +
            							# `readpartial` will raise `EOFError` if the stream is finished, or `IOError` if the stream is closed.
         | 
| 44 | 
            +
            							if chunk = @stream.readpartial(@remaining)
         | 
| 45 | 
            +
            								@remaining -= chunk.bytesize
         | 
| 46 | 
            +
            								
         | 
| 47 | 
            +
            								return chunk
         | 
| 48 | 
            +
            							end
         | 
| 42 49 | 
             
            						end
         | 
| 50 | 
            +
            						
         | 
| 51 | 
            +
            						# If the stream has been closed before we have read the expected length, raise an error:
         | 
| 52 | 
            +
            						raise EOFError, "Stream closed before expected length was read!"
         | 
| 43 53 | 
             
            					end
         | 
| 44 54 | 
             
            				end
         | 
| 45 55 |  | 
| 46 | 
            -
            				def join
         | 
| 47 | 
            -
            					buffer = @stream.read(@remaining)
         | 
| 48 | 
            -
            					
         | 
| 49 | 
            -
            					@remaining = 0
         | 
| 50 | 
            -
            					
         | 
| 51 | 
            -
            					return buffer
         | 
| 52 | 
            -
            				end
         | 
| 53 | 
            -
            				
         | 
| 54 56 | 
             
            				def inspect
         | 
| 55 | 
            -
            					"\#<#{self.class} length=#{@length} remaining=#{@remaining}>"
         | 
| 57 | 
            +
            					"\#<#{self.class} length=#{@length} remaining=#{@remaining} state=#{@stream ? 'open' : 'closed'}>"
         | 
| 56 58 | 
             
            				end
         | 
| 57 59 | 
             
            			end
         | 
| 58 60 | 
             
            		end
         | 
| @@ -8,53 +8,39 @@ require 'protocol/http/body/readable' | |
| 8 8 | 
             
            module Protocol
         | 
| 9 9 | 
             
            	module HTTP1
         | 
| 10 10 | 
             
            		module Body
         | 
| 11 | 
            +
            			# A body that reads all remaining data from the stream.
         | 
| 11 12 | 
             
            			class Remainder < HTTP::Body::Readable
         | 
| 12 13 | 
             
            				BLOCK_SIZE = 1024 * 64
         | 
| 13 14 |  | 
| 14 15 | 
             
            				# block_size may be removed in the future. It is better managed by stream.
         | 
| 15 16 | 
             
            				def initialize(stream)
         | 
| 16 17 | 
             
            					@stream = stream
         | 
| 17 | 
            -
            					@empty = false
         | 
| 18 18 | 
             
            				end
         | 
| 19 19 |  | 
| 20 20 | 
             
            				def empty?
         | 
| 21 | 
            -
            					@ | 
| 21 | 
            +
            					@stream.nil?
         | 
| 22 22 | 
             
            				end
         | 
| 23 23 |  | 
| 24 24 | 
             
            				def close(error = nil)
         | 
| 25 | 
            -
            					 | 
| 26 | 
            -
             | 
| 27 | 
            -
             | 
| 25 | 
            +
            					if @stream
         | 
| 26 | 
            +
            						# We can't really do anything in this case except close the connection.
         | 
| 27 | 
            +
            						@stream.close_read
         | 
| 28 | 
            +
            						@stream = nil
         | 
| 29 | 
            +
            					end
         | 
| 28 30 |  | 
| 29 31 | 
             
            					super
         | 
| 30 32 | 
             
            				end
         | 
| 31 33 |  | 
| 32 | 
            -
            				# TODO this is a bit less efficient in order to maintain compatibility with `IO`.
         | 
| 33 34 | 
             
            				def read
         | 
| 34 | 
            -
            					@stream | 
| 35 | 
            +
            					@stream&.readpartial(BLOCK_SIZE)
         | 
| 35 36 | 
             
            				rescue EOFError, IOError
         | 
| 36 | 
            -
            					@ | 
| 37 | 
            -
            					
         | 
| 37 | 
            +
            					@stream = nil
         | 
| 38 38 | 
             
            					# I noticed that in some cases you will get EOFError, and in other cases IOError!?
         | 
| 39 39 | 
             
            					return nil
         | 
| 40 40 | 
             
            				end
         | 
| 41 41 |  | 
| 42 | 
            -
            				def call(stream)
         | 
| 43 | 
            -
            					self.each do |chunk|
         | 
| 44 | 
            -
            						stream.write(chunk)
         | 
| 45 | 
            -
            					end
         | 
| 46 | 
            -
            					
         | 
| 47 | 
            -
            					stream.flush
         | 
| 48 | 
            -
            				end
         | 
| 49 | 
            -
            				
         | 
| 50 | 
            -
            				def join
         | 
| 51 | 
            -
            					@stream.read
         | 
| 52 | 
            -
            				ensure
         | 
| 53 | 
            -
            					@empty = true
         | 
| 54 | 
            -
            				end
         | 
| 55 | 
            -
            				
         | 
| 56 42 | 
             
            				def inspect
         | 
| 57 | 
            -
            					"\#<#{self.class}  | 
| 43 | 
            +
            					"\#<#{self.class} state=#{@stream ? 'open' : 'closed'}>"
         | 
| 58 44 | 
             
            				end
         | 
| 59 45 | 
             
            			end
         | 
| 60 46 | 
             
            		end
         | 
| @@ -4,7 +4,7 @@ | |
| 4 4 | 
             
            # Copyright, 2019-2024, by Samuel Williams.
         | 
| 5 5 | 
             
            # Copyright, 2019, by Brian Morearty.
         | 
| 6 6 | 
             
            # Copyright, 2020, by Bruno Sutic.
         | 
| 7 | 
            -
            # Copyright, 2023, by Thomas Morgan.
         | 
| 7 | 
            +
            # Copyright, 2023-2024, by Thomas Morgan.
         | 
| 8 8 | 
             
            # Copyright, 2024, by Anton Zhuravsky.
         | 
| 9 9 |  | 
| 10 10 | 
             
            require 'protocol/http/headers'
         | 
| @@ -105,7 +105,7 @@ module Protocol | |
| 105 105 | 
             
            			def write_upgrade_header(upgrade)
         | 
| 106 106 | 
             
            				@stream.write("connection: upgrade\r\nupgrade: #{upgrade}\r\n")
         | 
| 107 107 | 
             
            			end
         | 
| 108 | 
            -
             | 
| 108 | 
            +
            			
         | 
| 109 109 | 
             
            			# Indicates whether the connection has been hijacked meaning its
         | 
| 110 110 | 
             
            			# IO has been handed over and is not usable anymore.
         | 
| 111 111 | 
             
            			# @return [Boolean] hijack status
         | 
| @@ -240,7 +240,7 @@ module Protocol | |
| 240 240 | 
             
            					if match = line.match(HEADER)
         | 
| 241 241 | 
             
            						fields << [match[1], match[2]]
         | 
| 242 242 | 
             
            					else
         | 
| 243 | 
            -
            						raise BadHeader, "Could not parse header: #{line. | 
| 243 | 
            +
            						raise BadHeader, "Could not parse header: #{line.inspect}"
         | 
| 244 244 | 
             
            					end
         | 
| 245 245 | 
             
            				end
         | 
| 246 246 |  | 
| @@ -423,6 +423,7 @@ module Protocol | |
| 423 423 | 
             
            			end
         | 
| 424 424 |  | 
| 425 425 | 
             
            			def read_remainder_body
         | 
| 426 | 
            +
            				@persistent = false
         | 
| 426 427 | 
             
            				Body::Remainder.new(@stream)
         | 
| 427 428 | 
             
            			end
         | 
| 428 429 |  | 
| @@ -434,6 +435,12 @@ module Protocol | |
| 434 435 | 
             
            				read_remainder_body
         | 
| 435 436 | 
             
            			end
         | 
| 436 437 |  | 
| 438 | 
            +
            			def read_upgrade_body
         | 
| 439 | 
            +
            				# 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).
         | 
| 440 | 
            +
            				# 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).
         | 
| 441 | 
            +
            				read_remainder_body
         | 
| 442 | 
            +
            			end
         | 
| 443 | 
            +
            			
         | 
| 437 444 | 
             
            			HEAD = "HEAD"
         | 
| 438 445 | 
             
            			CONNECT = "CONNECT"
         | 
| 439 446 |  | 
| @@ -444,7 +451,7 @@ module Protocol | |
| 444 451 | 
             
            					if content_length =~ VALID_CONTENT_LENGTH
         | 
| 445 452 | 
             
            						yield Integer(content_length, 10)
         | 
| 446 453 | 
             
            					else
         | 
| 447 | 
            -
            						raise BadRequest, "Invalid content length: #{content_length. | 
| 454 | 
            +
            						raise BadRequest, "Invalid content length: #{content_length.inspect}"
         | 
| 448 455 | 
             
            					end
         | 
| 449 456 | 
             
            				end
         | 
| 450 457 | 
             
            			end
         | 
| @@ -469,6 +476,10 @@ module Protocol | |
| 469 476 | 
             
            					return nil
         | 
| 470 477 | 
             
            				end
         | 
| 471 478 |  | 
| 479 | 
            +
            				if status == 101
         | 
| 480 | 
            +
            					return read_upgrade_body
         | 
| 481 | 
            +
            				end
         | 
| 482 | 
            +
            				
         | 
| 472 483 | 
             
            				if (status >= 100 and status < 200) or status == 204 or status == 304
         | 
| 473 484 | 
             
            					return nil
         | 
| 474 485 | 
             
            				end
         | 
| @@ -495,6 +506,11 @@ module Protocol | |
| 495 506 | 
             
            					return read_tunnel_body
         | 
| 496 507 | 
             
            				end
         | 
| 497 508 |  | 
| 509 | 
            +
            				# A successful upgrade response implies that the connection will become a tunnel immediately after the empty line that concludes the header fields.
         | 
| 510 | 
            +
            				if headers[UPGRADE]
         | 
| 511 | 
            +
            					return read_upgrade_body
         | 
| 512 | 
            +
            				end
         | 
| 513 | 
            +
            				
         | 
| 498 514 | 
             
            				# 6.  If this is a request message and none of the above are true, then
         | 
| 499 515 | 
             
            				# the message body length is zero (no message body is present).
         | 
| 500 516 | 
             
            				return read_body(headers)
         | 
    
        data/license.md
    CHANGED
    
    | @@ -4,7 +4,7 @@ Copyright, 2019-2024, by Samuel Williams. | |
| 4 4 | 
             
            Copyright, 2019, by Brian Morearty.  
         | 
| 5 5 | 
             
            Copyright, 2020, by Olle Jonsson.  
         | 
| 6 6 | 
             
            Copyright, 2020, by Bruno Sutic.  
         | 
| 7 | 
            -
            Copyright, 2023, by Thomas Morgan.  
         | 
| 7 | 
            +
            Copyright, 2023-2024, by Thomas Morgan.  
         | 
| 8 8 | 
             
            Copyright, 2024, by Anton Zhuravsky.  
         | 
| 9 9 |  | 
| 10 10 | 
             
            Permission is hereby granted, free of charge, to any person obtaining a copy
         | 
    
        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.22.0
         | 
| 5 5 | 
             
            platform: ruby
         | 
| 6 6 | 
             
            authors:
         | 
| 7 7 | 
             
            - Samuel Williams
         | 
| @@ -42,7 +42,7 @@ cert_chain: | |
| 42 42 | 
             
              Q2K9NVun/S785AP05vKkXZEFYxqG6EW012U4oLcFl5MySFajYXRYbuUpH6AY+HP8
         | 
| 43 43 | 
             
              voD0MPg1DssDLKwXyt1eKD/+Fq0bFWhwVM/1XiAXL7lyYUyOq24KHgQ2Csg=
         | 
| 44 44 | 
             
              -----END CERTIFICATE-----
         | 
| 45 | 
            -
            date: 2024- | 
| 45 | 
            +
            date: 2024-09-05 00:00:00.000000000 Z
         | 
| 46 46 | 
             
            dependencies:
         | 
| 47 47 | 
             
            - !ruby/object:Gem::Dependency
         | 
| 48 48 | 
             
              name: protocol-http
         | 
    
        metadata.gz.sig
    CHANGED
    
    | Binary file |