Class: HTTP2Next::Header::Decompressor
- Inherits:
-
Object
- Object
- HTTP2Next::Header::Decompressor
- Includes:
- Error
- Defined in:
- lib/http/2/next/compressor.rb
Overview
Responsible for decoding received headers and maintaining compression context of the opposing peer. Decompressor must be initialized with appropriate starting context based on local role: client or server.
Constant Summary collapse
- FORBIDDEN_HEADERS =
%w[connection te].freeze
Instance Method Summary collapse
-
#decode(buf, frame = nil) ⇒ Array
Decodes and processes header commands within provided buffer.
-
#header(buf) ⇒ Hash
Decodes header command from provided buffer.
-
#initialize(options = {}) ⇒ Decompressor
constructor
A new instance of Decompressor.
-
#integer(buf, n) ⇒ Integer
Decodes integer value from provided buffer.
-
#string(buf) ⇒ String
Decodes string value from provided buffer.
-
#table_size=(size) ⇒ Object
Set dynamic table size in EncodingContext.
Constructor Details
#initialize(options = {}) ⇒ Decompressor
Returns a new instance of Decompressor.
496 497 498 |
# File 'lib/http/2/next/compressor.rb', line 496 def initialize( = {}) @cc = EncodingContext.new() end |
Instance Method Details
#decode(buf, frame = nil) ⇒ Array
Decodes and processes header commands within provided buffer.
588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 |
# File 'lib/http/2/next/compressor.rb', line 588 def decode(buf, frame = nil) list = [] decoding_pseudo_headers = true @cc.listen_on_table do until buf.empty? field, value = @cc.process(header(buf)) next if field.nil? is_pseudo_header = field.start_with? ":" if !decoding_pseudo_headers && is_pseudo_header raise ProtocolError, "one or more pseudo headers encountered after regular headers" end decoding_pseudo_headers = is_pseudo_header raise ProtocolError, "invalid header received: #{field}" if FORBIDDEN_HEADERS.include?(field) if frame case field when ":method" frame[:method] = value when "content-length" frame[:content_length] = Integer(value) when "trailer" (frame[:trailer] ||= []) << value end end list << [field, value] end end list end |
#header(buf) ⇒ Hash
Decodes header command from provided buffer.
549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 |
# File 'lib/http/2/next/compressor.rb', line 549 def header(buf) peek = buf.readbyte(0) header = {} header[:type], type = HEADREP.find do |_t, desc| mask = (peek >> desc[:prefix]) << desc[:prefix] mask == desc[:pattern] end raise CompressionError unless header[:type] header[:name] = integer(buf, type[:prefix]) case header[:type] when :indexed raise CompressionError if (header[:name]).zero? header[:name] -= 1 when :changetablesize header[:value] = header[:name] else if (header[:name]).zero? header[:name] = string(buf) else header[:name] -= 1 end header[:value] = string(buf) end header end |
#integer(buf, n) ⇒ Integer
Decodes integer value from provided buffer.
511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 |
# File 'lib/http/2/next/compressor.rb', line 511 def integer(buf, n) limit = 2**n - 1 i = !n.zero? ? (buf.getbyte & limit) : 0 m = 0 if i == limit while (byte = buf.getbyte) i += ((byte & 127) << m) m += 7 break if (byte & 128).zero? end end i end |
#string(buf) ⇒ String
Decodes string value from provided buffer.
533 534 535 536 537 538 539 540 541 542 543 |
# File 'lib/http/2/next/compressor.rb', line 533 def string(buf) raise CompressionError, "invalid header block fragment" if buf.empty? huffman = (buf.readbyte(0) & 0x80) == 0x80 len = integer(buf, 7) str = buf.read(len) raise CompressionError, "string too short" unless str.bytesize == len str = Huffman.new.decode(Buffer.new(str)) if huffman str.force_encoding(Encoding::UTF_8) end |
#table_size=(size) ⇒ Object
Set dynamic table size in EncodingContext
502 503 504 |
# File 'lib/http/2/next/compressor.rb', line 502 def table_size=(size) @cc.table_size = size end |