Index: pkg/serialization/test/serialization_test.dart |
diff --git a/pkg/serialization/test/serialization_test.dart b/pkg/serialization/test/serialization_test.dart |
new file mode 100644 |
index 0000000000000000000000000000000000000000..99b96c403bab71fb29949ff7422ce13941e30845 |
--- /dev/null |
+++ b/pkg/serialization/test/serialization_test.dart |
@@ -0,0 +1,816 @@ |
+// Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file |
+// for details. All rights reserved. Use of this source code is governed by a |
+// BSD-style license that can be found in the LICENSE file. |
+ |
+library serialization_test; |
+ |
+import 'dart:convert'; |
+import 'package:unittest/unittest.dart'; |
+import 'package:serialization/serialization.dart'; |
+import 'package:serialization/src/serialization_helpers.dart'; |
+import 'package:serialization/src/mirrors_helpers.dart'; |
+import 'dart:isolate'; |
+ |
+part 'test_models.dart'; |
+ |
+void main() { |
+ var p1 = new Person(); |
+ var a1 = new Address(); |
+ a1.street = 'N 34th'; |
+ a1.city = 'Seattle'; |
+ |
+ var formats = [const InternalMapFormat(), |
+ const SimpleFlatFormat(), const SimpleMapFormat(), |
+ const SimpleJsonFormat(storeRoundTripInfo: true)]; |
+ |
+ test('Basic extraction of a simple object', () { |
+ // TODO(alanknight): Switch these to use literal types. Issue |
+ var s = new Serialization() |
+ ..addRuleFor(Address).configureForMaps(); |
+ Map extracted = states(a1, s).first; |
+ expect(extracted.length, 4); |
+ expect(extracted['street'], 'N 34th'); |
+ expect(extracted['city'], 'Seattle'); |
+ expect(extracted['state'], null); |
+ expect(extracted['zip'], null); |
+ Reader reader = setUpReader(s, extracted); |
+ Address a2 = readBackSimple(s, a1, reader); |
+ expect(a2.street, 'N 34th'); |
+ expect(a2.city, 'Seattle'); |
+ expect(a2.state,null); |
+ expect(a2.zip, null); |
+ }); |
+ |
+ test('Slightly further with a simple object', () { |
+ var p1 = new Person()..name = 'Alice'..address = a1; |
+ var s = new Serialization() |
+ ..addRuleFor(Person).configureForMaps() |
+ ..addRuleFor(Address).configureForMaps(); |
+ // TODO(alanknight): Need a better API for getting to flat state without |
+ // actually writing. |
+ var w = new Writer(s, const InternalMapFormat()); |
+ w.write(p1); |
+ var personRule = s.rules.firstWhere( |
+ (x) => x is BasicRule && x.type == reflect(p1).type); |
+ var flatPerson = w.states[personRule.number].first; |
+ var primStates = w.states.first; |
+ expect(primStates.isEmpty, true); |
+ expect(flatPerson["name"], "Alice"); |
+ var ref = flatPerson["address"]; |
+ expect(ref is Reference, true); |
+ var addressRule = s.rules.firstWhere( |
+ (x) => x is BasicRule && x.type == reflect(a1).type); |
+ expect(ref.ruleNumber, addressRule.number); |
+ expect(ref.objectNumber, 0); |
+ expect(w.states[addressRule.number].first['street'], 'N 34th'); |
+ }); |
+ |
+ test('exclude fields', () { |
+ var s = new Serialization() |
+ ..addRuleFor(Address, |
+ excludeFields: ['state', 'zip']).configureForMaps(); |
+ var extracted = states(a1, s).first; |
+ expect(extracted.length, 2); |
+ expect(extracted['street'], 'N 34th'); |
+ expect(extracted['city'], 'Seattle'); |
+ Reader reader = setUpReader(s, extracted); |
+ Address a2 = readBackSimple(s, a1, reader); |
+ expect(a2.state, null); |
+ expect(a2.city, 'Seattle'); |
+ }); |
+ |
+ test('list', () { |
+ var list = [5, 4, 3, 2, 1]; |
+ var s = new Serialization(); |
+ var extracted = states(list, s).first; |
+ expect(extracted.length, 5); |
+ for (var i = 0; i < 5; i++) { |
+ expect(extracted[i], (5 - i)); |
+ } |
+ Reader reader = setUpReader(s, extracted); |
+ var list2 = readBackSimple(s, list, reader); |
+ expect(list, list2); |
+ }); |
+ |
+ test('different kinds of fields', () { |
+ var x = new Various.Foo("d", "e"); |
+ x.a = "a"; |
+ x.b = "b"; |
+ x._c = "c"; |
+ var s = new Serialization() |
+ ..addRuleFor(Various, |
+ constructor: "Foo", |
+ constructorFields: ["d", "e"]); |
+ var state = states(x, s).first; |
+ expect(state.length, 4); |
+ var expected = "abde"; |
+ for (var i in [0,1,2,3]) { |
+ expect(state[i], expected[i]); |
+ } |
+ Reader reader = setUpReader(s, state); |
+ Various y = readBackSimple(s, x, reader); |
+ expect(x.a, y.a); |
+ expect(x.b, y.b); |
+ expect(x.d, y.d); |
+ expect(x.e, y.e); |
+ expect(y._c, 'default value'); |
+ }); |
+ |
+ test('Stream', () { |
+ // This is an interesting case. The Stream doesn't expose its internal |
+ // collection at all, and sets it in the constructor. So to get it we |
+ // read a private field and then set that via the constructor. That works |
+ // but should we have some kind of large red flag that you're using private |
+ // state. |
+ var stream = new Stream([3,4,5]); |
+ expect((stream..next()).next(), 4); |
+ expect(stream.position, 2); |
+ // The Symbol class does not allow us to create symbols for private |
+ // variables. However, the mirror system uses them. So we get the symbol |
+ // we want from the mirror. |
+ // TODO(alanknight): Either delete this test and decide we shouldn't |
+ // attempt to access private variables or fix this properly. |
+ var _collectionSym = reflect(stream).type.declarations.keys.firstWhere( |
+ (x) => MirrorSystem.getName(x) == "_collection"); |
+ var s = new Serialization() |
+ ..addRuleFor(Stream, |
+ constructorFields: [_collectionSym]); |
+ var state = states(stream, s).first; |
+ // Define names for the variable offsets to make this more readable. |
+ var _collection = 0, position = 1; |
+ expect(state[_collection],[3,4,5]); |
+ expect(state[position], 2); |
+ }); |
+ |
+ test('date', () { |
+ var date = new DateTime.now(); |
+ var utcDate = new DateTime.utc(date.year, date.month, date.day, |
+ date.hour, date.minute, date.second, date.millisecond); |
+ var s = new Serialization(); |
+ var out = s.write([date, utcDate]); |
+ expect(s.selfDescribing, isTrue); |
+ var input = s.read(out); |
+ expect(input.first, date); |
+ expect(input.last, utcDate); |
+ }); |
+ |
+ test('Iteration helpers', () { |
+ var map = {"a" : 1, "b" : 2, "c" : 3}; |
+ var list = [1, 2, 3]; |
+ var set = new Set.from(list); |
+ var m = keysAndValues(map); |
+ var l = keysAndValues(list); |
+ var s = keysAndValues(set); |
+ |
+ m.forEach((key, value) {expect(key.codeUnits[0], value + 96);}); |
+ l.forEach((key, value) {expect(key + 1, value);}); |
+ var index = 0; |
+ var seen = new Set(); |
+ s.forEach((key, value) { |
+ expect(key, index++); |
+ expect(seen.contains(value), isFalse); |
+ seen.add(value); |
+ }); |
+ expect(seen.length, 3); |
+ |
+ var i = 0; |
+ m = values(map); |
+ l = values(list); |
+ s = values(set); |
+ m.forEach((each) {expect(each, ++i);}); |
+ i = 0; |
+ l.forEach((each) {expect(each, ++i);}); |
+ i = 0; |
+ s.forEach((each) {expect(each, ++i);}); |
+ i = 0; |
+ |
+ seen = new Set(); |
+ for (var each in m) { |
+ expect(seen.contains(each), isFalse); |
+ seen.add(each); |
+ } |
+ expect(seen.length, 3); |
+ i = 0; |
+ for (var each in l) { |
+ expect(each, ++i); |
+ } |
+ }); |
+ |
+ Node n1 = new Node("1"), n2 = new Node("2"), n3 = new Node("3"); |
+ n1.children = [n2, n3]; |
+ n2.parent = n1; |
+ n3.parent = n1; |
+ |
+ test('Trace a cyclical structure', () { |
+ var s = new Serialization(); |
+ var trace = new Trace(new Writer(s)); |
+ trace.writer.trace = trace; |
+ trace.trace(n1); |
+ var all = trace.writer.references.keys.toSet(); |
+ expect(all.length, 4); |
+ expect(all.contains(n1), isTrue); |
+ expect(all.contains(n2), isTrue); |
+ expect(all.contains(n3), isTrue); |
+ expect(all.contains(n1.children), isTrue); |
+ }); |
+ |
+ test('Flatten references in a cyclical structure', () { |
+ var s = new Serialization(); |
+ var w = new Writer(s, const InternalMapFormat()); |
+ w.trace = new Trace(w); |
+ w.write(n1); |
+ expect(w.states.length, 7); // prims, lists * 2, basic, symbol, date |
+ var children = 0, name = 1, parent = 2; |
+ var nodeRule = s.rules.firstWhere((x) => x is BasicRule); |
+ List rootNode = w.states[nodeRule.number].where( |
+ (x) => x[name] == "1").toList(); |
+ rootNode = rootNode.first; |
+ expect(rootNode[parent], isNull); |
+ var list = w.states[1].first; |
+ expect(w.stateForReference(rootNode[children]), list); |
+ var parentNode = w.stateForReference(list[0])[parent]; |
+ expect(w.stateForReference(parentNode), rootNode); |
+ }); |
+ |
+ test('round-trip', () { |
+ runRoundTripTest(nodeSerializerReflective); |
+ }); |
+ |
+ test('round-trip with explicit self-description', () { |
+ // We provide a setup function which, when run the second time, |
+ // returns a blank serialization, to make sure it will fail |
+ // the second time. |
+ var s; |
+ oneShotSetup(node) { |
+ if (s == null) { |
+ s = nodeSerializerReflective(node)..selfDescribing = true; |
+ return s; |
+ } else { |
+ s = null; |
+ return new Serialization.blank() |
+ ..namedObjects['Node'] = reflect(new Node('')).type; |
+ } |
+ } |
+ |
+ runRoundTripTest(oneShotSetup); |
+ }); |
+ |
+ test('round-trip ClosureRule', () { |
+ runRoundTripTest(nodeSerializerNonReflective); |
+ }); |
+ |
+ test('round-trip with essential parent', () { |
+ runRoundTripTest(nodeSerializerWithEssentialParent); |
+ }); |
+ |
+ test('round-trip, flat format', () { |
+ runRoundTripTestFlat(nodeSerializerReflective); |
+ }); |
+ |
+ test('round-trip using Maps', () { |
+ runRoundTripTest(nodeSerializerUsingMaps); |
+ }); |
+ |
+ test('round-trip, flat format, using maps', () { |
+ runRoundTripTestFlat(nodeSerializerUsingMaps); |
+ }); |
+ |
+ test('round-trip with Node CustomRule', () { |
+ runRoundTripTestFlat(nodeSerializerCustom); |
+ }); |
+ |
+ test('round-trip with Node CustomRule, to maps', () { |
+ runRoundTripTest(nodeSerializerCustom); |
+ }); |
+ |
+ test('eating your own tail', () { |
+ // Create a meta-serializer, that serializes serializations, then |
+ // use it to serialize a basic serialization, then run a test on the |
+ // the result. |
+ var s = new Serialization.blank() |
+ // Add the rules in a deliberately unusual order. |
+ ..addRuleFor(Node, constructorFields: ['name']) |
+ ..addRule(new ListRule()) |
+ ..addRule(new PrimitiveRule()) |
+ ..selfDescribing = false; |
+ var meta = metaSerialization(); |
+ var metaWithMaps = metaSerializationUsingMaps(); |
+ for (var eachFormat in formats) { |
+ for (var eachMeta in [meta, metaWithMaps]) { |
+ var serialized = eachMeta.write(s, format: eachFormat); |
+ var newSerialization = eachMeta.read(serialized, format: eachFormat, |
+ externals: {"serialization_test.Node" : reflect(new Node('')).type} |
+ ); |
+ runRoundTripTest((x) => newSerialization); |
+ } |
+ } |
+ }); |
+ |
+ test("Verify we're not serializing lists twice if they're essential", () { |
+ Node n1 = new Node("1"), n2 = new Node("2"), n3 = new Node("3"); |
+ n1.children = [n2, n3]; |
+ n2.parent = n1; |
+ n3.parent = n1; |
+ var s = new Serialization() |
+ ..addRuleFor(Node, constructorFields: ["name"]). |
+ setFieldWith("children", (parent, child) => |
+ parent.reflectee.children = child); |
+ var w = new Writer(s); |
+ w.write(n1); |
+ expect(w.rules[2] is ListRuleEssential, isTrue); |
+ expect(w.rules[1] is ListRule, isTrue); |
+ expect(w.states[1].length, 0); |
+ expect(w.states[2].length, 1); |
+ s = new Serialization() |
+ ..addRuleFor(Node, constructorFields: ["name"]); |
+ w = new Writer(s); |
+ w.write(n1); |
+ expect(w.states[1].length, 1); |
+ expect(w.states[2].length, 0); |
+ }); |
+ |
+ test('Identity of equal objects preserved', () { |
+ Node n1 = new NodeEqualByName("foo"), |
+ n2 = new NodeEqualByName("foo"), |
+ n3 = new NodeEqualByName("3"); |
+ n1.children = [n2, n3]; |
+ n2.parent = n1; |
+ n3.parent = n1; |
+ var s = new Serialization() |
+ ..selfDescribing = false |
+ ..addRuleFor(NodeEqualByName, constructorFields: ["name"]); |
+ var m1 = writeAndReadBack(s, null, n1); |
+ var m2 = m1.children.first; |
+ var m3 = m1.children.last; |
+ expect(m1, m2); |
+ expect(identical(m1, m2), isFalse); |
+ expect(m1 == m3, isFalse); |
+ expect(identical(m2.parent, m3.parent), isTrue); |
+ }); |
+ |
+ test("Constant values as fields", () { |
+ var s = new Serialization() |
+ ..selfDescribing = false |
+ ..addRuleFor(Address, |
+ constructor: 'withData', |
+ constructorFields: ["street", "Kirkland", "WA", "98103"], |
+ fields: []); |
+ var out = s.write(a1); |
+ var newAddress = s.read(out); |
+ expect(newAddress.street, a1.street); |
+ expect(newAddress.city, "Kirkland"); |
+ expect(newAddress.state, "WA"); |
+ expect(newAddress.zip, "98103"); |
+ }); |
+ |
+ test("Straight JSON format", () { |
+ var s = new Serialization(); |
+ var writer = s.newWriter(const SimpleJsonFormat()); |
+ var out = JSON.encode(writer.write(a1)); |
+ var reconstituted = JSON.decode(out); |
+ expect(reconstituted.length, 4); |
+ expect(reconstituted[0], "Seattle"); |
+ }); |
+ |
+ test("Straight JSON format, nested objects", () { |
+ var p1 = new Person()..name = 'Alice'..address = a1; |
+ var s = new Serialization()..selfDescribing = false; |
+ var addressRule = s.addRuleFor(Address)..configureForMaps(); |
+ var personRule = s.addRuleFor(Person)..configureForMaps(); |
+ var writer = s.newWriter(const SimpleJsonFormat(storeRoundTripInfo: true)); |
+ var out = JSON.encode(writer.write(p1)); |
+ var reconstituted = JSON.decode(out); |
+ var expected = { |
+ "name" : "Alice", |
+ "rank" : null, |
+ "serialNumber" : null, |
+ "_rule" : personRule.number, |
+ "address" : { |
+ "street" : "N 34th", |
+ "city" : "Seattle", |
+ "state" : null, |
+ "zip" : null, |
+ "_rule" : addressRule.number |
+ } |
+ }; |
+ expect(expected, reconstituted); |
+ }); |
+ |
+ test("Straight JSON format, round-trip", () { |
+ // Note that we can't use the usual round-trip test because it has cycles. |
+ var p1 = new Person()..name = 'Alice'..address = a1; |
+ // Use maps for one rule, lists for the other. |
+ var s = new Serialization() |
+ ..addRuleFor(Address) |
+ ..addRuleFor(Person).configureForMaps(); |
+ var p2 = writeAndReadBack(s, |
+ const SimpleJsonFormat(storeRoundTripInfo: true), p1); |
+ expect(p2.name, "Alice"); |
+ var a2 = p2.address; |
+ expect(a2.street, "N 34th"); |
+ expect(a2.city, "Seattle"); |
+ }); |
+ |
+ test("Straight JSON format, non-string key", () { |
+ // This tests what happens if we have a key that's not a string. That's |
+ // not allowed by json, so we don't actually turn it into a json string, |
+ // but someone might reasonably convert to a json-able structure without |
+ // going through the string representation. |
+ var p1 = new Person()..name = 'Alice'..address = a1; |
+ var s = new Serialization() |
+ ..addRule(new PersonRuleReturningMapWithNonStringKey()); |
+ var p2 = writeAndReadBack(s, |
+ const SimpleJsonFormat(storeRoundTripInfo: true), p1); |
+ expect(p2.name, "Alice"); |
+ expect(p2.address.street, "N 34th"); |
+ }); |
+ |
+ test("Root is a Map", () { |
+ // Note that we can't use the usual round-trip test because it has cycles. |
+ var p1 = new Person()..name = 'Alice'..address = a1; |
+ // Use maps for one rule, lists for the other. |
+ var s = new Serialization() |
+ // Deliberately left as passing instances to test backward-compatibility. |
+ ..addRuleFor(a1) |
+ ..addRuleFor(p1).configureForMaps(); |
+ for (var eachFormat in formats) { |
+ var w = s.newWriter(eachFormat); |
+ var output = w.write({"stuff" : p1}); |
+ var result = s.read(output, format: w.format); |
+ var p2 = result["stuff"]; |
+ expect(p2.name, "Alice"); |
+ var a2 = p2.address; |
+ expect(a2.street, "N 34th"); |
+ expect(a2.city, "Seattle"); |
+ } |
+ }); |
+ |
+ test("Root is a List", () { |
+ var s = new Serialization(); |
+ for (var eachFormat in formats) { |
+ var result = writeAndReadBack(s, eachFormat, [a1]); |
+ var a2 = result.first; |
+ expect(a2.street, "N 34th"); |
+ expect(a2.city, "Seattle"); |
+ } |
+ }); |
+ |
+ test("Root is a simple object", () { |
+ var s = new Serialization(); |
+ for (var eachFormat in formats) { |
+ expect(writeAndReadBack(s, eachFormat, null), null); |
+ expect(writeAndReadBack(s, eachFormat, [null]), [null]); |
+ expect(writeAndReadBack(s, eachFormat, 3), 3); |
+ expect(writeAndReadBack(s, eachFormat, [3]), [3]); |
+ expect(writeAndReadBack(s, eachFormat, "hello"), "hello"); |
+ expect(writeAndReadBack(s, eachFormat, [3]), [3]); |
+ expect(writeAndReadBack(s, eachFormat, {"hello" : "world"}), |
+ {"hello" : "world"}); |
+ expect(writeAndReadBack(s, eachFormat, true), true); |
+ } |
+ }); |
+ |
+ test("Simple JSON format, round-trip with named objects", () { |
+ // Note that we can't use the usual round-trip test because it has cycles. |
+ var p1 = new Person()..name = 'Alice'..address = a1; |
+ // Use maps for one rule, lists for the other. |
+ var s = new Serialization() |
+ ..selfDescribing = false |
+ ..addRule(new NamedObjectRule()) |
+ ..addRuleFor(Address) |
+ ..addRuleFor(Person).configureForMaps() |
+ ..namedObjects["foo"] = a1; |
+ var format = const SimpleJsonFormat(storeRoundTripInfo: true); |
+ var out = s.write(p1, format: format); |
+ var p2 = s.read(out, format: format, externals: {"foo" : 12}); |
+ expect(p2.name, "Alice"); |
+ var a2 = p2.address; |
+ expect(a2, 12); |
+ }); |
+ |
+ test("More complicated Maps", () { |
+ var s = new Serialization()..selfDescribing = false; |
+ var p1 = new Person()..name = 'Alice'..address = a1; |
+ var data = new Map(); |
+ data["simple data"] = 1; |
+ data[p1] = a1; |
+ data[a1] = p1; |
+ for (var eachFormat in formats) { |
+ var output = s.write(data, format: eachFormat); |
+ var input = s.read(output, format: eachFormat); |
+ expect(input["simple data"], data["simple data"]); |
+ var p2 = input.keys.firstWhere((x) => x is Person); |
+ var a2 = input.keys.firstWhere((x) => x is Address); |
+ if (eachFormat is SimpleJsonFormat) { |
+ // JSON doesn't handle cycles, so these won't be identical. |
+ expect(input[p2] is Address, isTrue); |
+ expect(input[a2] is Person, isTrue); |
+ var a3 = input[p2]; |
+ expect(a3.city, a2.city); |
+ expect(a3.state, a2.state); |
+ expect(a3.state, a2.state); |
+ var p3 = input[a2]; |
+ expect(p3.name, p2.name); |
+ expect(p3.rank, p2.rank); |
+ expect(p3.address.city, a2.city); |
+ } else { |
+ expect(input[p2], same(a2)); |
+ expect(input[a2], same(p2)); |
+ } |
+ } |
+ }); |
+ |
+ test("Map with string keys stays that way", () { |
+ var s = new Serialization()..addRuleFor(Person); |
+ var data = {"abc" : 1, "def" : "ghi"}; |
+ data["person"] = new Person()..name = "Foo"; |
+ var output = s.write(data, format: const InternalMapFormat()); |
+ var mapRule = s.rules.firstWhere((x) => x is MapRule); |
+ var map = output["data"][mapRule.number][0]; |
+ expect(map is Map, isTrue); |
+ expect(map["abc"], 1); |
+ expect(map["def"], "ghi"); |
+ expect(map["person"] is Reference, isTrue); |
+ }); |
+ |
+ test("MirrorRule with lookup by qualified name rather than named object", () { |
+ var s = new Serialization()..addRule(new MirrorRule()); |
+ var m = reflectClass(Address); |
+ var output = s.write(m); |
+ var input = s.read(output); |
+ expect(input is ClassMirror, isTrue); |
+ expect(MirrorSystem.getName(input.simpleName), "Address"); |
+ }); |
+ |
+ test('round-trip, default format, pass to isolate', () { |
+ Node n1 = new Node("1"), n2 = new Node("2"), n3 = new Node("3"); |
+ n1.children = [n2, n3]; |
+ n2.parent = n1; |
+ n3.parent = n1; |
+ var s = nodeSerializerReflective(n1); |
+ var output = s.write(n2); |
+ ReceivePort port = new ReceivePort(); |
+ var remote = Isolate.spawn(echo, [output, port.sendPort]); |
+ port.first.then(verify); |
+ }); |
+} |
+ |
+/** |
+ * Verify serialized output that we have passed to an isolate and back. |
+ */ |
+void verify(input) { |
+ var s2 = nodeSerializerReflective(new Node("a")); |
+ var m2 = s2.read(input); |
+ var m1 = m2.parent; |
+ expect(m1 is Node, isTrue); |
+ var children = m1.children; |
+ expect(m1.name,"1"); |
+ var m3 = m1.children.last; |
+ expect(m2.name, "2"); |
+ expect(m3.name, "3"); |
+ expect(m2.parent, m1); |
+ expect(m3.parent, m1); |
+ expect(m1.parent, isNull); |
+} |
+ |
+/****************************************************************************** |
+ * The end of the tests and the beginning of various helper functions to make |
+ * it easier to write the repetitive sections. |
+ ******************************************************************************/ |
+ |
+writeAndReadBack(Serialization s, Format format, object) { |
+ var output = s.write(object, format: format); |
+ return s.read(output, format: format); |
+} |
+ |
+/** Create a Serialization for serializing Serializations. */ |
+Serialization metaSerialization() { |
+ // Make some bogus rule instances so we have something to feed rule creation |
+ // and get their types. If only we had class literals implemented... |
+ var basicRule = new BasicRule(reflect(null).type, '', [], [], []); |
+ |
+ var meta = new Serialization() |
+ ..selfDescribing = false |
+ ..addRuleFor(ListRule) |
+ ..addRuleFor(PrimitiveRule) |
+ // TODO(alanknight): Handle CustomRule as well. |
+ // Note that we're passing in a constant for one of the fields. |
+ ..addRuleFor(BasicRule, |
+ constructorFields: ['type', |
+ 'constructorName', |
+ 'constructorFields', 'regularFields', []], |
+ fields: []) |
+ ..addRuleFor(Serialization, constructor: "blank") |
+ .setFieldWith('rules', |
+ (InstanceMirror s, List rules) { |
+ rules.forEach((x) => s.reflectee.addRule(x)); |
+ }) |
+ ..addRule(new NamedObjectRule()) |
+ ..addRule(new MirrorRule()) |
+ ..addRule(new MapRule()); |
+ return meta; |
+} |
+ |
+Serialization metaSerializationUsingMaps() { |
+ var meta = metaSerialization(); |
+ meta.rules.where((each) => each is BasicRule) |
+ .forEach((x) => x.configureForMaps()); |
+ return meta; |
+} |
+ |
+/** |
+ * Read back a simple object, assumed to be the only one of its class in the |
+ * reader. |
+ */ |
+readBackSimple(Serialization s, object, Reader reader) { |
+ var rule = s.rulesFor(object, null).first; |
+ reader.inflateForRule(rule); |
+ var list2 = reader.allObjectsForRule(rule).first; |
+ return list2; |
+} |
+ |
+/** |
+ * Set up a basic reader with some fake data. Hard-codes the assumption |
+ * of how many rules there are. |
+ */ |
+Reader setUpReader(aSerialization, sampleData) { |
+ var reader = new Reader(aSerialization); |
+ // We're not sure which rule needs the sample data, so put it everywhere |
+ // and trust that the extra will just be ignored. |
+ |
+ var fillValue = [sampleData]; |
+ var data = []; |
+ for (int i = 0; i < 10; i++) { |
+ data.add(fillValue); |
+ } |
+ reader.data = data; |
+ return reader; |
+} |
+ |
+/** Return a serialization for Node objects, using a reflective rule. */ |
+Serialization nodeSerializerReflective(Node n) { |
+ return new Serialization() |
+ ..addRuleFor(Node, constructorFields: ["name"]) |
+ ..namedObjects['Node'] = reflect(new Node('')).type; |
+} |
+ |
+/** |
+ * Return a serialization for Node objects but using Maps for the internal |
+ * representation rather than lists. |
+ */ |
+Serialization nodeSerializerUsingMaps(Node n) { |
+ return new Serialization() |
+ // Get the type using runtimeType to verify that works. |
+ ..addRuleFor(n.runtimeType, constructorFields: ["name"]).configureForMaps() |
+ ..namedObjects['Node'] = reflect(new Node('')).type; |
+} |
+ |
+/** |
+ * Return a serialization for Node objects but using Maps for the internal |
+ * representation rather than lists. |
+ */ |
+Serialization nodeSerializerCustom(Node n) { |
+ return new Serialization() |
+ ..addRule(new NodeRule()); |
+} |
+ |
+/** |
+ * Return a serialization for Node objects where the "parent" instance |
+ * variable is considered essential state. |
+ */ |
+Serialization nodeSerializerWithEssentialParent(Node n) { |
+ // Force the node rule to be first, in order to make a cycle which would |
+ // not cause a problem if we handled the list first, because the list |
+ // considers all of its state non-essential, thus breaking the cycle. |
+ var s = new Serialization.blank() |
+ ..addRuleFor( |
+ Node, |
+ constructor: "parentEssential", |
+ constructorFields: ["parent"]) |
+ ..addDefaultRules() |
+ ..namedObjects['Node'] = reflect(new Node('')).type |
+ ..selfDescribing = false; |
+ return s; |
+} |
+ |
+/** Return a serialization for Node objects using a ClosureToMapRule. */ |
+Serialization nodeSerializerNonReflective(Node n) { |
+ var rule = new ClosureRule( |
+ n.runtimeType, |
+ (o) => {"name" : o.name, "children" : o.children, "parent" : o.parent}, |
+ (map) => new Node(map["name"]), |
+ (object, map) { |
+ object |
+ ..children = map["children"] |
+ ..parent = map["parent"]; |
+ }); |
+ return new Serialization() |
+ ..selfDescribing = false |
+ ..addRule(rule); |
+} |
+ |
+/** |
+ * Run a round-trip test on a simple tree of nodes, using a serialization |
+ * that's returned by the [serializerSetUp] function. |
+ */ |
+void runRoundTripTest(Function serializerSetUp) { |
+ Node n1 = new Node("1"), n2 = new Node("2"), n3 = new Node("3"); |
+ n1.children = [n2, n3]; |
+ n2.parent = n1; |
+ n3.parent = n1; |
+ var s = serializerSetUp(n1); |
+ var output = s.write(n2); |
+ var s2 = serializerSetUp(n1); |
+ var m2 = s2.read(output); |
+ var m1 = m2.parent; |
+ expect(m1 is Node, isTrue); |
+ var children = m1.children; |
+ expect(m1.name,"1"); |
+ var m3 = m1.children.last; |
+ expect(m2.name, "2"); |
+ expect(m3.name, "3"); |
+ expect(m2.parent, m1); |
+ expect(m3.parent, m1); |
+ expect(m1.parent, isNull); |
+} |
+ |
+/** |
+ * Run a round-trip test on a simple of nodes, but using the flat format |
+ * rather than the maps. |
+ */ |
+void runRoundTripTestFlat(serializerSetUp) { |
+ Node n1 = new Node("1"), n2 = new Node("2"), n3 = new Node("3"); |
+ n1.children = [n2, n3]; |
+ n2.parent = n1; |
+ n3.parent = n1; |
+ var s = serializerSetUp(n1); |
+ var output = s.write(n2, format: const SimpleFlatFormat()); |
+ expect(output is List, isTrue); |
+ var s2 = serializerSetUp(n1); |
+ var m2 = s2.read(output, format: const SimpleFlatFormat()); |
+ var m1 = m2.parent; |
+ expect(m1 is Node, isTrue); |
+ var children = m1.children; |
+ expect(m1.name,"1"); |
+ var m3 = m1.children.last; |
+ expect(m2.name, "2"); |
+ expect(m3.name, "3"); |
+ expect(m2.parent, m1); |
+ expect(m3.parent, m1); |
+ expect(m1.parent, isNull); |
+} |
+ |
+/** Extract the state from [object] using the rules in [s] and return it. */ |
+List states(object, Serialization s) { |
+ var rules = s.rulesFor(object, null); |
+ return rules.map((x) => x.extractState(object, doNothing, null)).toList(); |
+} |
+ |
+/** A hard-coded rule for serializing Node instances. */ |
+class NodeRule extends CustomRule { |
+ bool appliesTo(instance, _) => instance.runtimeType == Node; |
+ getState(instance) => [instance.parent, instance.name, instance.children]; |
+ create(state) => new Node(state[1]); |
+ void setState(Node node, state) { |
+ node.parent = state[0]; |
+ node.children = state[2]; |
+ } |
+} |
+ |
+/** |
+ * This is a rather silly rule which stores the address data in a map, |
+ * but inverts the keys and values, so we look up values and find the |
+ * corresponding key. This will lead to maps that aren't allowed in JSON, |
+ * and which have keys that need to be dereferenced. |
+ */ |
+class PersonRuleReturningMapWithNonStringKey extends CustomRule { |
+ appliesTo(instance, _) => instance is Person; |
+ getState(instance) { |
+ return new Map() |
+ ..[instance.name] = "name" |
+ ..[instance.address] = "address"; |
+ } |
+ create(state) => new Person(); |
+ void setState(Person a, state) { |
+ a.name = findValue("name", state); |
+ a.address = findValue("address", state); |
+ } |
+ findValue(String key, Map state) { |
+ var answer; |
+ for (var each in state.keys) { |
+ var value = state[each]; |
+ if (value == key) return each; |
+ } |
+ return null; |
+ } |
+} |
+ |
+/** |
+ * Function used in an isolate to make sure that the output passes through |
+ * isolate serialization properly. |
+ */ |
+void echo(initialMessage) { |
+ var msg = initialMessage[0]; |
+ var reply = initialMessage[1]; |
+ reply.send(msg); |
+} |