# encoding: binary require "amq/protocol/type_constants" require "amq/protocol/table_value_encoder" require "amq/protocol/table_value_decoder" module AMQ module Protocol class Table # # Behaviors # include TypeConstants # # API # class InvalidTableError < StandardError def initialize(key, value) super("Invalid table value on key #{key}: #{value.inspect} (#{value.class})") end end def self.encode(table) buffer = String.new table ||= {} table.each do |key, value| key = key.to_s # it can be a symbol as well buffer << key.bytesize.chr + key case value when Hash then buffer << TYPE_HASH buffer << self.encode(value) else buffer << TableValueEncoder.encode(value).force_encoding(buffer.encoding) end end [buffer.bytesize].pack(PACK_UINT32).force_encoding(buffer.encoding) + buffer end def self.decode(data) table = Hash.new table_length = data.unpack(PACK_UINT32).first return table if table_length.zero? offset = 4 while offset <= table_length key, offset = decode_table_key(data, offset) type, offset = TableValueDecoder.decode_value_type(data, offset) table[key] = case type when TYPE_STRING v, offset = TableValueDecoder.decode_string(data, offset) v when TYPE_BYTE_ARRAY # Ruby doesn't have a direct counterpart to # ByteBuffer or byte[], so using a string feels # more appropriate than an array of fixnums v, offset = TableValueDecoder.decode_string(data, offset) v when TYPE_INTEGER v, offset = TableValueDecoder.decode_integer(data, offset) v when TYPE_DECIMAL v, offset = TableValueDecoder.decode_big_decimal(data, offset) v when TYPE_TIME v, offset = TableValueDecoder.decode_time(data, offset) v when TYPE_HASH v, offset = TableValueDecoder.decode_hash(data, offset) v when TYPE_BOOLEAN v, offset = TableValueDecoder.decode_boolean(data, offset) v when TYPE_BYTE then v, offset = TableValueDecoder.decode_byte(data, offset) v when TYPE_SIGNED_16BIT then v, offset = TableValueDecoder.decode_short(data, offset) v when TYPE_SIGNED_64BIT then v, offset = TableValueDecoder.decode_long(data, offset) v when TYPE_32BIT_FLOAT then v, offset = TableValueDecoder.decode_32bit_float(data, offset) v when TYPE_64BIT_FLOAT then v, offset = TableValueDecoder.decode_64bit_float(data, offset) v when TYPE_VOID nil when TYPE_ARRAY v, offset = TableValueDecoder.decode_array(data, offset) v else raise ArgumentError, "Not a valid type: #{type.inspect}\nData: #{data.inspect}\nUnprocessed data: #{data[offset..-1].inspect}\nOffset: #{offset}\nTotal size: #{table_length}\nProcessed data: #{table.inspect}" end end table end # self.decode def self.length(data) data.unpack(PACK_UINT32).first end def self.hash_size(value) acc = 0 value.each do |k, v| acc += (1 + k.to_s.bytesize) acc += TableValueEncoder.field_value_size(v) end acc end # self.hash_size(value) def self.decode_table_key(data, offset) key_length = data.slice(offset, 1).unpack(PACK_CHAR).first offset += 1 key = data.slice(offset, key_length) offset += key_length [key, offset] end # self.decode_table_key(data, offset) end # Table end # Protocol end # AMQ