/*
* Copyright (C) 2005 Joe Walnes.
* Copyright (C) 2006, 2007 XStream Committers.
* All rights reserved.
*
* The software in this package is published under the terms of the BSD
* style license a copy of which has been included with this distribution in
* the LICENSE.txt file.
*
* Created on 02. February 2005 by Joe Walnes
*/
package com.thoughtworks.acceptance;
import com.thoughtworks.xstream.testutil.CallLog;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
import java.io.ObjectInputValidation;
import java.io.ObjectOutputStream;
import java.io.Serializable;
public class SerializationCallbackOrderTest extends AbstractAcceptanceTest {
// static so it can be accessed by objects under test, without them needing a reference back to the testcase
private static CallLog log = new CallLog();
protected void setUp() throws Exception {
super.setUp();
log.reset();
}
// --- Sample class hiearchy
public static class Base implements Serializable{
private void writeObject(ObjectOutputStream out) throws IOException {
log.actual("Base.writeObject() start");
out.defaultWriteObject();
log.actual("Base.writeObject() end");
}
private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException {
log.actual("Base.readObject() start");
in.defaultReadObject();
log.actual("Base.readObject() end");
}
private Object writeReplace() {
log.actual("Base.writeReplace()");
return this;
}
private Object readResolve() {
log.actual("Base.readResolve()");
return this;
}
}
public static class Child extends Base implements Serializable{
private void writeObject(ObjectOutputStream out) throws IOException {
log.actual("Child.writeObject() start");
out.defaultWriteObject();
log.actual("Child.writeObject() end");
}
private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException {
log.actual("Child.readObject() start");
in.defaultReadObject();
log.actual("Child.readObject() end");
}
private Object writeReplace() {
log.actual("Child.writeReplace()");
return this;
}
private Object readResolve() {
log.actual("Child.readResolve()");
return this;
}
}
// --- Convenience wrappers around Java Object Serialization
private byte[] javaSerialize(Object object) throws IOException {
ByteArrayOutputStream bytes = new ByteArrayOutputStream();
ObjectOutputStream objectOutputStream = new ObjectOutputStream(bytes);
objectOutputStream.writeObject(object);
objectOutputStream.close();
return bytes.toByteArray();
}
private Object javaDeserialize(byte[] data) throws IOException, ClassNotFoundException {
ObjectInputStream objectInputStream = new ObjectInputStream(new ByteArrayInputStream(data));
return objectInputStream.readObject();
}
// --- Tests
public void testJavaSerialization() throws IOException {
// expectations
log.expect("Child.writeReplace()");
log.expect("Base.writeObject() start");
log.expect("Base.writeObject() end");
log.expect("Child.writeObject() start");
log.expect("Child.writeObject() end");
// execute
javaSerialize(new Child());
// verify
log.verify();
}
public void testXStreamSerialization() {
// expectations
log.expect("Child.writeReplace()");
log.expect("Base.writeObject() start");
log.expect("Base.writeObject() end");
log.expect("Child.writeObject() start");
log.expect("Child.writeObject() end");
// execute
xstream.toXML(new Child());
// verify
log.verify();
}
public void testJavaDeserialization() throws IOException, ClassNotFoundException {
// setup
byte[] data = javaSerialize(new Child());
log.reset();
// expectations
log.expect("Base.readObject() start");
log.expect("Base.readObject() end");
log.expect("Child.readObject() start");
log.expect("Child.readObject() end");
log.expect("Child.readResolve()");
// execute
javaDeserialize(data);
// verify
log.verify();
}
public void testXStreamDeserialization() {
// setup
String data = xstream.toXML(new Child());
log.reset();
// expectations
log.expect("Base.readObject() start");
log.expect("Base.readObject() end");
log.expect("Child.readObject() start");
log.expect("Child.readObject() end");
log.expect("Child.readResolve()");
// execute
xstream.fromXML(data);
// verify
log.verify();
}
public static class ParentNotTransient implements Serializable {
public int somethingNotTransient;
public ParentNotTransient(int somethingNotTransient) {
this.somethingNotTransient = somethingNotTransient;
}
}
public static class ChildWithTransient extends ParentNotTransient implements Serializable {
public transient int somethingTransient;
public ChildWithTransient(int somethingNotTransient, int somethingTransient) {
super(somethingNotTransient);
this.somethingTransient = somethingTransient;
}
private void readObject(ObjectInputStream s) throws IOException, ClassNotFoundException {
s.defaultReadObject();
somethingTransient = 99999;
}
}
public void testCallsReadObjectEvenWithoutNonTransientFields() {
xstream.alias("parent", ParentNotTransient.class);
xstream.alias("child", ChildWithTransient.class);
Object in = new ChildWithTransient(10, 22222);
String expectedXml = ""
+ "<child serialization=\"custom\">\n"
+ " <parent>\n"
+ " <default>\n"
+ " <somethingNotTransient>10</somethingNotTransient>\n"
+ " </default>\n"
+ " </parent>\n"
+ " <child>\n"
+ " <default/>\n"
+ " </child>\n"
+ "</child>";
String xml = xstream.toXML(in);
assertEquals(expectedXml, xml);
ChildWithTransient childWithTransient = (ChildWithTransient) xstream.fromXML(xml);
assertEquals(10, childWithTransient.somethingNotTransient);
assertEquals(99999, childWithTransient.somethingTransient);
}
public static class SomethingThatValidates implements Serializable {
private void readObject(ObjectInputStream s) throws IOException {
final int LOW_PRIORITY = -5;
final int MEDIUM_PRIORITY = 0;
final int HIGH_PRIORITY = 5;
s.registerValidation(new ObjectInputValidation() {
public void validateObject() {
log.actual("validateObject() medium priority 1");
}
}, MEDIUM_PRIORITY);
s.registerValidation(new ObjectInputValidation() {
public void validateObject() {
log.actual("validateObject() high priority");
}
}, HIGH_PRIORITY);
s.registerValidation(new ObjectInputValidation() {
public void validateObject() {
log.actual("validateObject() low priority");
}
}, LOW_PRIORITY);
s.registerValidation(new ObjectInputValidation() {
public void validateObject() {
log.actual("validateObject() medium priority 2");
}
}, MEDIUM_PRIORITY);
}
private Object readResolve() {
log.actual("readResolve()");
return this;
}
}
public void testJavaSerializationValidatesObjectIsCalledInPriorityOrder() throws IOException, ClassNotFoundException {
// expect
log.expect("readResolve()");
log.expect("validateObject() high priority");
log.expect("validateObject() medium priority 2");
log.expect("validateObject() medium priority 1");
log.expect("validateObject() low priority");
// execute
javaDeserialize(javaSerialize(new SomethingThatValidates()));
// verify
log.verify();
}
public void testXStreamSerializationValidatesObjectIsCalledInPriorityOrder() {
// expect
log.expect("readResolve()");
log.expect("validateObject() high priority");
log.expect("validateObject() medium priority 2");
log.expect("validateObject() medium priority 1");
log.expect("validateObject() low priority");
// execute
xstream.fromXML(xstream.toXML(new SomethingThatValidates()));
// verify
log.verify();
}
public static class UnserializableParent {
public int x;
public UnserializableParent() {
x = 5;
}
}
public static class CustomSerializableChild extends UnserializableParent implements Serializable {
public int y;
public CustomSerializableChild() {
y = 10;
}
private void writeObject(ObjectOutputStream stream) throws IOException {
log.actual("Child.writeObject() start");
stream.defaultWriteObject();
log.actual("Child.writeObject() end");
}
private void readObject(ObjectInputStream stream)
throws IOException, ClassNotFoundException {
log.actual("Child.readObject() start");
stream.defaultReadObject();
log.actual("Child.readObject() end");
}
private Object writeReplace() {
log.actual("Child.writeReplace()");
return this;
}
private Object readResolve() {
log.actual("Child.readResolve()");
return this;
}
}
public void testFieldsOfUnserializableParentsArePreserved() {
xstream.alias("parent", UnserializableParent.class);
xstream.alias("child", CustomSerializableChild.class);
CustomSerializableChild child = new CustomSerializableChild();
String expected = ""
+ "<child serialization=\"custom\">\n"
+ " <unserializable-parents>\n"
+ " <x>5</x>\n"
+ " </unserializable-parents>\n"
+ " <child>\n"
+ " <default>\n"
+ " <y>10</y>\n"
+ " </default>\n"
+ " </child>\n"
+ "</child>";
CustomSerializableChild serialized =(CustomSerializableChild)assertBothWays(child, expected);
assertEquals(5, serialized.x);
assertEquals(10, serialized.y);
}
public static class SerializableGrandChild extends CustomSerializableChild implements Serializable {
public int z;
public SerializableGrandChild() {
super();
z = 42;
}
}
public void testUnserializableParentsAreWrittenOnlyOnce() {
xstream.alias("parent", UnserializableParent.class);
xstream.alias("child", CustomSerializableChild.class);
xstream.alias("grandchild", SerializableGrandChild.class);
SerializableGrandChild grandChild = new SerializableGrandChild();
String expected = ""
+ "<grandchild serialization=\"custom\">\n"
+ " <unserializable-parents>\n"
+ " <x>5</x>\n"
+ " </unserializable-parents>\n"
+ " <child>\n"
+ " <default>\n"
+ " <y>10</y>\n"
+ " </default>\n"
+ " </child>\n"
+ " <grandchild>\n"
+ " <default>\n"
+ " <z>42</z>\n"
+ " </default>\n"
+ " </grandchild>\n"
+ "</grandchild>";
SerializableGrandChild serialized =(SerializableGrandChild)assertBothWays(grandChild, expected);
assertEquals(5, serialized.x);
assertEquals(10, serialized.y);
assertEquals(42, serialized.z);
}
public void testXStreamSerializationForObjectsWithUnserializableParents() {
// expectations
log.expect("Child.writeReplace()");
log.expect("Child.writeObject() start");
log.expect("Child.writeObject() end");
// execute
xstream.toXML(new CustomSerializableChild());
// verify
log.verify();
}
public void testXStreamDeserializationForObjectsWithUnserializableParents() {
// setup
String data = xstream.toXML(new CustomSerializableChild());
log.reset();
// expectations
log.expect("Child.readObject() start");
log.expect("Child.readObject() end");
log.expect("Child.readResolve()");
// execute
xstream.fromXML(data);
// verify
log.verify();
}
}