503 lines
16 KiB
Ruby
503 lines
16 KiB
Ruby
|
|
# encoding: UTF-8
|
||
|
|
|
||
|
|
require "spec_helper.rb"
|
||
|
|
require 'rexml/document'
|
||
|
|
require 'bigdecimal'
|
||
|
|
require 'rational'
|
||
|
|
|
||
|
|
describe "when serializing" do
|
||
|
|
before :each do
|
||
|
|
RocketAMF::ClassMapper.reset
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should raise exception with invalid version number" do
|
||
|
|
lambda {
|
||
|
|
RocketAMF.serialize("", 5)
|
||
|
|
}.should raise_error("unsupported version 5")
|
||
|
|
end
|
||
|
|
|
||
|
|
describe "AMF0" do
|
||
|
|
it "should serialize nils" do
|
||
|
|
output = RocketAMF.serialize(nil, 0)
|
||
|
|
output.should == object_fixture('amf0-null.bin')
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize booleans" do
|
||
|
|
output = RocketAMF.serialize(true, 0)
|
||
|
|
output.should === object_fixture('amf0-boolean.bin')
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize numbers" do
|
||
|
|
output = RocketAMF.serialize(3.5, 0)
|
||
|
|
output.should == object_fixture('amf0-number.bin')
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize Numeric conformers" do
|
||
|
|
output = RocketAMF.serialize(BigDecimal.new("3.5"), 0)
|
||
|
|
output.should == object_fixture('amf0-number.bin')
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize strings" do
|
||
|
|
output = RocketAMF.serialize("this is a テスト", 0)
|
||
|
|
output.should == object_fixture('amf0-string.bin')
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize frozen strings" do
|
||
|
|
output = RocketAMF.serialize("this is a テスト".freeze, 0)
|
||
|
|
output.should == object_fixture('amf0-string.bin')
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize arrays" do
|
||
|
|
output = RocketAMF.serialize(['a', 'b', 'c', 'd'], 0)
|
||
|
|
output.should == object_fixture('amf0-strict-array.bin')
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize references" do
|
||
|
|
obj = OtherClass.new
|
||
|
|
obj.foo = "baz"
|
||
|
|
obj.bar = 3.14
|
||
|
|
|
||
|
|
output = RocketAMF.serialize({'0' => obj, '1' => obj}, 0)
|
||
|
|
output.should == object_fixture('amf0-ref-test.bin')
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize Time objects" do
|
||
|
|
output = RocketAMF.serialize(Time.utc(2003, 2, 13, 5), 0)
|
||
|
|
output.bytesize.should == 11
|
||
|
|
output[0,9].should == object_fixture('amf0-time.bin')[0,9] # Ignore TZ
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize Date objects" do
|
||
|
|
output = RocketAMF.serialize(Date.civil(2020, 5, 30), 0)
|
||
|
|
output.bytesize.should == 11
|
||
|
|
output[0,9].should == object_fixture('amf0-date.bin')[0,9] # Ignore TZ
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize DateTime objects" do
|
||
|
|
output = RocketAMF.serialize(DateTime.civil(2003, 2, 13, 5), 0)
|
||
|
|
output.bytesize.should == 11
|
||
|
|
output[0,9].should == object_fixture('amf0-time.bin')[0,9] # Ignore TZ
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize hashes as objects" do
|
||
|
|
output = RocketAMF.serialize({:baz => nil, "foo" => "bar"}, 0)
|
||
|
|
output.should == object_fixture('amf0-untyped-object.bin')
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize unmapped objects" do
|
||
|
|
obj = RubyClass.new
|
||
|
|
obj.foo = "bar"
|
||
|
|
|
||
|
|
output = RocketAMF.serialize(obj, 0)
|
||
|
|
output.should == object_fixture('amf0-untyped-object.bin')
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize mapped objects" do
|
||
|
|
obj = RubyClass.new
|
||
|
|
obj.foo = "bar"
|
||
|
|
RocketAMF::ClassMapper.define {|m| m.map :as => 'org.amf.ASClass', :ruby => 'RubyClass'}
|
||
|
|
|
||
|
|
output = RocketAMF.serialize(obj, 0)
|
||
|
|
output.should == object_fixture('amf0-typed-object.bin')
|
||
|
|
end
|
||
|
|
|
||
|
|
describe "and handling encodings", :if => "".respond_to?(:force_encoding) do
|
||
|
|
it "should support multiple encodings" do
|
||
|
|
shift_str = "\x53\x68\x69\x66\x74\x20\x83\x65\x83\x58\x83\x67".force_encoding("Shift_JIS") # "Shift テスト"
|
||
|
|
utf_str = "\x55\x54\x46\x20\xe3\x83\x86\xe3\x82\xb9\xe3\x83\x88".force_encoding("UTF-8") # "UTF テスト"
|
||
|
|
output = RocketAMF.serialize({:shift => shift_str, :utf => utf_str, :zed => 5}, 0)
|
||
|
|
output.should == object_fixture("amf0-complex-encoded-string.bin")
|
||
|
|
end
|
||
|
|
end
|
||
|
|
end
|
||
|
|
|
||
|
|
describe "AMF3" do
|
||
|
|
describe "simple messages" do
|
||
|
|
it "should serialize a null" do
|
||
|
|
expected = object_fixture("amf3-null.bin")
|
||
|
|
output = RocketAMF.serialize(nil, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize a false" do
|
||
|
|
expected = object_fixture("amf3-false.bin")
|
||
|
|
output = RocketAMF.serialize(false, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize a true" do
|
||
|
|
expected = object_fixture("amf3-true.bin")
|
||
|
|
output = RocketAMF.serialize(true, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize integers" do
|
||
|
|
expected = object_fixture("amf3-max.bin")
|
||
|
|
input = RocketAMF::MAX_INTEGER
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
|
||
|
|
expected = object_fixture("amf3-0.bin")
|
||
|
|
output = RocketAMF.serialize(0, 3)
|
||
|
|
output.should == expected
|
||
|
|
|
||
|
|
expected = object_fixture("amf3-min.bin")
|
||
|
|
input = RocketAMF::MIN_INTEGER
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize large integers" do
|
||
|
|
expected = object_fixture("amf3-large-max.bin")
|
||
|
|
input = RocketAMF::MAX_INTEGER + 1
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
|
||
|
|
expected = object_fixture("amf3-large-min.bin")
|
||
|
|
input = RocketAMF::MIN_INTEGER - 1
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize floats" do
|
||
|
|
expected = object_fixture("amf3-float.bin")
|
||
|
|
input = 3.5
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize BigNums" do
|
||
|
|
expected = object_fixture("amf3-bigNum.bin")
|
||
|
|
input = 2**1000
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize float Numeric conformers" do
|
||
|
|
expected = object_fixture("amf3-float.bin")
|
||
|
|
input = Rational(7, 2) # 3.5
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize a simple string" do
|
||
|
|
expected = object_fixture("amf3-string.bin")
|
||
|
|
input = "String . String"
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize a frozen string" do
|
||
|
|
expected = object_fixture("amf3-string.bin")
|
||
|
|
input = "String . String".freeze
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize a symbol as a string" do
|
||
|
|
expected = object_fixture("amf3-symbol.bin")
|
||
|
|
output = RocketAMF.serialize(:foo, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize Time objects" do
|
||
|
|
expected = object_fixture("amf3-date.bin")
|
||
|
|
input = Time.utc 1970, 1, 1, 0
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize Date objects" do
|
||
|
|
expected = object_fixture("amf3-date.bin")
|
||
|
|
input = Date.civil 1970, 1, 1, 0
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize DateTime objects" do
|
||
|
|
expected = object_fixture("amf3-date.bin")
|
||
|
|
input = DateTime.civil 1970, 1, 1, 0
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
end
|
||
|
|
|
||
|
|
describe "objects" do
|
||
|
|
it "should serialize an unmapped object as a dynamic anonymous object" do
|
||
|
|
class NonMappedObject
|
||
|
|
def another_public_property
|
||
|
|
'a_public_value'
|
||
|
|
end
|
||
|
|
|
||
|
|
attr_accessor :nil_property
|
||
|
|
attr_accessor :property_one
|
||
|
|
attr_writer :read_only_prop
|
||
|
|
|
||
|
|
def method_with_arg arg='foo'
|
||
|
|
arg
|
||
|
|
end
|
||
|
|
end
|
||
|
|
obj = NonMappedObject.new
|
||
|
|
obj.property_one = 'foo'
|
||
|
|
obj.nil_property = nil
|
||
|
|
|
||
|
|
expected = object_fixture("amf3-dynamic-object.bin")
|
||
|
|
input = obj
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize externalizable objects" do
|
||
|
|
a = ExternalizableTest.new
|
||
|
|
a.one = 5
|
||
|
|
a.two = 7
|
||
|
|
b = ExternalizableTest.new
|
||
|
|
b.one = 13
|
||
|
|
b.two = 5
|
||
|
|
obj = [a, b]
|
||
|
|
|
||
|
|
expected = object_fixture("amf3-externalizable.bin")
|
||
|
|
input = obj
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize a hash as a dynamic anonymous object" do
|
||
|
|
hash = {}
|
||
|
|
hash[:answer] = 42
|
||
|
|
hash['foo'] = "bar"
|
||
|
|
|
||
|
|
expected = object_fixture("amf3-hash.bin")
|
||
|
|
input = hash
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize an empty array" do
|
||
|
|
expected = object_fixture("amf3-empty-array.bin")
|
||
|
|
input = []
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize an array of primatives" do
|
||
|
|
expected = object_fixture("amf3-primitive-array.bin")
|
||
|
|
input = [1, 2, 3, 4, 5]
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize an array of mixed objects" do
|
||
|
|
h1 = {:foo_one => "bar_one"}
|
||
|
|
h2 = {:foo_two => ""}
|
||
|
|
class SimpleObj
|
||
|
|
attr_accessor :foo_three
|
||
|
|
end
|
||
|
|
so1 = SimpleObj.new
|
||
|
|
so1.foo_three = 42
|
||
|
|
|
||
|
|
expected = object_fixture("amf3-mixed-array.bin")
|
||
|
|
input = [h1, h2, so1, {}, [h1, h2, so1], [], 42, "", [], "", {}, "bar_one", so1]
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize an array as an array collection" do
|
||
|
|
expected = object_fixture('amf3-array-collection.bin')
|
||
|
|
|
||
|
|
# Test global
|
||
|
|
RocketAMF::ClassMapper.use_array_collection = true
|
||
|
|
input = ["foo", "bar"]
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
RocketAMF::ClassMapper.use_array_collection = false
|
||
|
|
|
||
|
|
# Test override
|
||
|
|
input = ["foo", "bar"]
|
||
|
|
input.is_array_collection = true
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize a complex set of array collections" do
|
||
|
|
RocketAMF::ClassMapper.define {|m| m.map :as => 'org.amf.ASClass', :ruby => 'RubyClass'}
|
||
|
|
expected = object_fixture('amf3-complex-array-collection.bin')
|
||
|
|
|
||
|
|
a = ["foo", "bar"]
|
||
|
|
a.is_array_collection = true
|
||
|
|
obj1 = RubyClass.new
|
||
|
|
obj1.foo = "bar"
|
||
|
|
def obj1.encode_amf serializer
|
||
|
|
serializer.write_object(self, nil, {:class_name => 'org.amf.ASClass', :dynamic => false, :externalizable => false, :members => ["baz", "foo"]})
|
||
|
|
end
|
||
|
|
obj2 = RubyClass.new
|
||
|
|
obj2.foo = "asdf"
|
||
|
|
def obj2.encode_amf serializer
|
||
|
|
serializer.write_object(self, nil, {:class_name => 'org.amf.ASClass', :dynamic => false, :externalizable => false, :members => ["baz", "foo"]})
|
||
|
|
end
|
||
|
|
b = [obj1, obj2]
|
||
|
|
b.is_array_collection = true
|
||
|
|
input = [a, b, b]
|
||
|
|
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize a byte array" do
|
||
|
|
expected = object_fixture("amf3-byte-array.bin")
|
||
|
|
str = "\000\003これtest\100"
|
||
|
|
str.force_encoding("ASCII-8BIT") if str.respond_to?(:force_encoding)
|
||
|
|
input = StringIO.new(str)
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
end
|
||
|
|
|
||
|
|
describe "and implementing the AMF Spec" do
|
||
|
|
it "should keep references of duplicate strings" do
|
||
|
|
class StringCarrier
|
||
|
|
attr_accessor :str
|
||
|
|
end
|
||
|
|
foo = "foo"
|
||
|
|
bar = "str"
|
||
|
|
sc = StringCarrier.new
|
||
|
|
sc.str = foo
|
||
|
|
|
||
|
|
expected = object_fixture("amf3-string-ref.bin")
|
||
|
|
input = [foo, bar, foo, bar, foo, sc]
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should not reference the empty string" do
|
||
|
|
expected = object_fixture("amf3-empty-string-ref.bin")
|
||
|
|
input = ""
|
||
|
|
output = RocketAMF.serialize([input,input], 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should keep references of duplicate dates" do
|
||
|
|
expected = object_fixture("amf3-date-ref.bin")
|
||
|
|
input = Time.utc 1970, 1, 1, 0
|
||
|
|
output = RocketAMF.serialize([input,input], 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should keep reference of duplicate objects" do
|
||
|
|
class SimpleReferenceableObj
|
||
|
|
attr_accessor :foo
|
||
|
|
end
|
||
|
|
obj1 = SimpleReferenceableObj.new
|
||
|
|
obj1.foo = :bar
|
||
|
|
obj2 = SimpleReferenceableObj.new
|
||
|
|
obj2.foo = obj1.foo
|
||
|
|
|
||
|
|
expected = object_fixture("amf3-object-ref.bin")
|
||
|
|
input = [[obj1, obj2], "bar", [obj1, obj2]]
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should keep reference of duplicate object traits" do
|
||
|
|
obj1 = RubyClass.new
|
||
|
|
obj1.foo = "foo"
|
||
|
|
def obj1.encode_amf serializer
|
||
|
|
serializer.write_object(self, nil, {:class_name => 'org.amf.ASClass', :dynamic => false, :externalizable => false, :members => ["baz", "foo"]})
|
||
|
|
end
|
||
|
|
obj2 = RubyClass.new
|
||
|
|
obj2.foo = "bar"
|
||
|
|
def obj2.encode_amf serializer
|
||
|
|
serializer.write_object(self, nil, {:class_name => 'org.amf.ASClass', :dynamic => false, :externalizable => false, :members => ["baz", "foo"]})
|
||
|
|
end
|
||
|
|
input = [obj1, obj2]
|
||
|
|
|
||
|
|
expected = object_fixture("amf3-trait-ref.bin")
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should keep references of duplicate arrays" do
|
||
|
|
a = [1,2,3]
|
||
|
|
b = %w{ a b c }
|
||
|
|
|
||
|
|
expected = object_fixture("amf3-array-ref.bin")
|
||
|
|
input = [a, b, a, b]
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should not keep references of duplicate empty arrays unless the object_id matches" do
|
||
|
|
a = []
|
||
|
|
b = []
|
||
|
|
a.should == b
|
||
|
|
a.object_id.should_not == b.object_id
|
||
|
|
|
||
|
|
expected = object_fixture("amf3-empty-array-ref.bin")
|
||
|
|
input = [a,b,a,b]
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should keep references of duplicate byte arrays" do
|
||
|
|
b = StringIO.new "ASDF"
|
||
|
|
|
||
|
|
expected = object_fixture("amf3-byte-array-ref.bin")
|
||
|
|
input = [b, b]
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should serialize a deep object graph with circular references" do
|
||
|
|
class GraphMember
|
||
|
|
attr_accessor :children
|
||
|
|
attr_accessor :parent
|
||
|
|
|
||
|
|
def initialize
|
||
|
|
self.children = []
|
||
|
|
end
|
||
|
|
|
||
|
|
def add_child child
|
||
|
|
children << child
|
||
|
|
child.parent = self
|
||
|
|
child
|
||
|
|
end
|
||
|
|
end
|
||
|
|
|
||
|
|
parent = GraphMember.new
|
||
|
|
level_1_child_1 = parent.add_child GraphMember.new
|
||
|
|
level_1_child_2 = parent.add_child GraphMember.new
|
||
|
|
|
||
|
|
expected = object_fixture("amf3-graph-member.bin")
|
||
|
|
input = parent
|
||
|
|
output = RocketAMF.serialize(input, 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
end
|
||
|
|
|
||
|
|
describe "and handling encodings", :if => "".respond_to?(:force_encoding) do
|
||
|
|
it "should support multiple encodings" do
|
||
|
|
shift_str = "\x53\x68\x69\x66\x74\x20\x83\x65\x83\x58\x83\x67".force_encoding("Shift_JIS") # "Shift テスト"
|
||
|
|
utf_str = "\x55\x54\x46\x20\xe3\x83\x86\xe3\x82\xb9\xe3\x83\x88".force_encoding("UTF-8") # "UTF テスト"
|
||
|
|
output = RocketAMF.serialize([5, shift_str, utf_str, 5], 3)
|
||
|
|
output.should == object_fixture("amf3-complex-encoded-string-array.bin")
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should keep references of duplicate strings with different encodings" do
|
||
|
|
# String is "this is a テスト"
|
||
|
|
shift_str = "\x74\x68\x69\x73\x20\x69\x73\x20\x61\x20\x83\x65\x83\x58\x83\x67".force_encoding("Shift_JIS")
|
||
|
|
utf_str = "\x74\x68\x69\x73\x20\x69\x73\x20\x61\x20\xe3\x83\x86\xe3\x82\xb9\xe3\x83\x88".force_encoding("UTF-8")
|
||
|
|
|
||
|
|
expected = object_fixture("amf3-encoded-string-ref.bin")
|
||
|
|
output = RocketAMF.serialize([shift_str, utf_str], 3)
|
||
|
|
output.should == expected
|
||
|
|
end
|
||
|
|
|
||
|
|
it "should handle inappropriate UTF-8 characters in byte arrays" do
|
||
|
|
str = "\xff\xff\xff".force_encoding("ASCII-8BIT")
|
||
|
|
str.freeze # For added amusement
|
||
|
|
output = RocketAMF.serialize(StringIO.new(str), 3)
|
||
|
|
output.should == "\x0c\x07\xff\xff\xff".force_encoding("ASCII-8BIT")
|
||
|
|
end
|
||
|
|
end
|
||
|
|
end
|
||
|
|
end
|