173 lines
6.0 KiB
Ruby
173 lines
6.0 KiB
Ruby
module Rex
|
|
module Java
|
|
module Serialization
|
|
module Model
|
|
# This class provides a Java Stream representation
|
|
class Stream < Element
|
|
|
|
include Rex::Java::Serialization
|
|
|
|
# @!attribute magic
|
|
# @return [String] The stream signature
|
|
attr_accessor :magic
|
|
# @!attribute version
|
|
# @return [Fixnum] The stream version
|
|
attr_accessor :version
|
|
# @!attribute contents
|
|
# @return [Array] The stream's contents
|
|
attr_accessor :contents
|
|
|
|
def initialize
|
|
self.magic = STREAM_MAGIC
|
|
self.version = STREAM_VERSION
|
|
self.contents = []
|
|
end
|
|
|
|
# Deserializes a Java::Serialization::Model::Stream
|
|
#
|
|
# @param io [IO] the io to read from
|
|
# @return [self] if deserialization succeeds
|
|
# @raise [RuntimeError] if deserialization doesn't succeed
|
|
def decode(io)
|
|
self.magic = decode_magic(io)
|
|
self.constant_name = decode_version(io)
|
|
|
|
self.contents << decode_content(io) until io.eof?
|
|
|
|
self
|
|
end
|
|
|
|
# Serializes the Java::Serialization::Model::Stream
|
|
#
|
|
# @return [String] if serialization succeeds
|
|
# @raise [RuntimeError] if serialization doesn't succeed
|
|
def encode
|
|
encoded = ''
|
|
encoded << magic
|
|
encoded << [version].pack('n')
|
|
contents.each do |content|
|
|
encoded << encode_content(content)
|
|
end
|
|
encoded
|
|
end
|
|
|
|
private
|
|
|
|
# Deserializes the magic stream value
|
|
#
|
|
# @param io [IO] the io to read from
|
|
# @return [String] if deserialization succeeds
|
|
# @raise [RuntimeError] if deserialization doesn't succed
|
|
def decode_magic(io)
|
|
magic = io.read(2)
|
|
unless magic && magic == STREAM_MAGIC
|
|
raise ::RuntimeError, 'Failed to unserialize Stream'
|
|
end
|
|
|
|
magic
|
|
end
|
|
|
|
# Deserializes the version stream
|
|
#
|
|
# @param io [IO] the io to read from
|
|
# @return [Fixnum] if deserialization succeeds
|
|
# @raise [RuntimeError] if deserialization doesn't succed
|
|
def decode_version(io)
|
|
version = io.read(2)
|
|
unless version && version.unpack('n')[0] == STREAM_VERSION
|
|
raise ::RuntimeError, 'Failed to unserialize Stream'
|
|
end
|
|
|
|
version
|
|
end
|
|
|
|
# Deserializes a content
|
|
#
|
|
# @param io [IO] the io to read from
|
|
# @return [Rex::Java::Serialization::Model::Element] if deserialization succeeds
|
|
# @raise [RuntimeError] if deserialization doesn't succeed or unsupported content
|
|
def decode_content(io)
|
|
opcode = io.read(1)
|
|
raise ::RuntimeError, 'Failed to unserialize content' if opcode.nil?
|
|
opcode = opcode.unpack('C')[0]
|
|
content = nil
|
|
|
|
case opcode
|
|
when TC_BLOCKDATA
|
|
content = BlockData.decode(io)
|
|
when TC_BLOCKDATALONG
|
|
content = BlockDataLong.decode(io)
|
|
when TC_OBJECT
|
|
content = NewObject.decode(io)
|
|
when TC_CLASS
|
|
content = ClassDesc.decode(io)
|
|
when TC_ARRAY
|
|
content = NewArray.decode(io)
|
|
when TC_STRING
|
|
content = Utf.decode(io)
|
|
when TC_LONGSTRING
|
|
content = LongUtf.decode(io)
|
|
when TC_ENUM
|
|
content = NewEnum.decode(io)
|
|
when TC_CLASSDESC
|
|
content = NewClassDesc.decode(io)
|
|
when TC_PROXYCLASSDESC
|
|
raise ::RuntimeError, 'Failed to unserialize unsupported TC_PROXYCLASSDESC content'
|
|
when TC_REFERENCE
|
|
raise ::RuntimeError, 'Failed to unserialize unsupported TC_REFERENCE content'
|
|
when TC_NULL
|
|
content = NullReference.decode(io)
|
|
when TC_EXCEPTION
|
|
raise ::RuntimeError, 'Failed to unserialize unsupported TC_EXCEPTION content'
|
|
when TC_RESET
|
|
content = Reset.decode(io)
|
|
else
|
|
raise ::RuntimeError, 'Failed to unserialize content'
|
|
end
|
|
|
|
content
|
|
end
|
|
|
|
# Serializes a content
|
|
#
|
|
# @param content [Rex::Java::Serialization::Model::Element] the content to serialize
|
|
# @return [String] if serialization succeeds
|
|
# @raise [RuntimeError] if serialization doesn't succeed
|
|
def encode_content(content)
|
|
encoded = ''
|
|
|
|
case content
|
|
when Rex::Java::Serialization::Model::BlockData
|
|
encoded << [TC_BLOCKDATA].pack('C')
|
|
when Rex::Java::Serialization::Model::BlockDataLong
|
|
encoded << [TC_BLOCKDATALONG].pack('C')
|
|
when Rex::Java::Serialization::Model::NewObject
|
|
encoded << [TC_OBJECT].pack('C')
|
|
when Rex::Java::Serialization::Model::ClassDesc
|
|
encoded << [TC_CLASS].pack('C')
|
|
when Rex::Java::Serialization::Model::NewArray
|
|
encoded << [TC_ARRAY].pack('C')
|
|
when Rex::Java::Serialization::Model::Utf
|
|
encoded << [TC_STRING].pack('C')
|
|
when Rex::Java::Serialization::Model::LongUtf
|
|
encoded << [TC_LONGSTRING].pack('C')
|
|
when Rex::Java::Serialization::Model::NewEnum
|
|
encoded << [TC_ENUM].pack('C')
|
|
when Rex::Java::Serialization::Model::NewClassDesc
|
|
encoded << [TC_CLASSDESC].pack('C')
|
|
when Rex::Java::Serialization::Model::NullReference
|
|
encoded << [TC_NULL].pack('C')
|
|
when Rex::Java::Serialization::Model::Reset
|
|
encoded << [TC_RESET].pack('C')
|
|
else
|
|
raise ::RuntimeError, 'Failed to serialize content'
|
|
end
|
|
|
|
encoded << content.encode
|
|
encoded
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end |