typesystem icon indicating copy to clipboard operation
typesystem copied to clipboard

Serialization of reference fields in composite fields

Open florimondmanca opened this issue 6 years ago • 1 comments

Hey,

I was fiddling with composite data types, in particular with Reference to implement nested schemas, and there seems to be an issue with how arrays and objects of references are serialized when calling dict() on the parent class.

Consider the following schemas:

import typesystem as ts

class Child(ts.Schema):
    pass

class ParentA(ts.Schema):
    child = ts.Reference(to=Child)

class ParentB(ts.Schema):
    children = ts.Array(ts.Reference(Child))

class ParentC(ts.Schema):
    children = ts.Object(properties=ts.Reference(Child))

Test cases with what should happen IMO:

pa = ParentA(child=Child())
assert dict(pa) == {"child": {}}  # OK

pb = ParentB(children=[Child()])
assert dict(pb) == {"children": [{}]}  # FAIL
# {"children": [Child()]}

pc = ParentC(children=[Child()])
assert dict(pc) == {"children": {"john": {}}}  # FAIL
# {"children": {"john": Child()}}

This causes issues when passing the result of dict(pb) or dict(pc) to, say, json.dumps(), because it gets Child instances which are not JSON-serializable.

Am I simply abusing Array and Object? Is there another way of implementing composite fields with nested schemas?

florimondmanca avatar Jul 09 '19 19:07 florimondmanca

FWIW, I implemented a workaround in the form of this helper:

def _serialize_nested(value: typing.Any) -> dict:
    if isinstance(value, ts.Schema):
        return _serialize_nested(dict(value))
    if isinstance(value, list):
        return [_serialize_nested(item) for item in value]
    if isinstance(value, dict):
        return {key: _serialize_nested(val) for key, val in value.items()}
    return value

Usage:

pb = ParentB(children=[Child()])
assert _serialize_nested(pb) == {"children": [{}]}  # OK

florimondmanca avatar Jul 09 '19 21:07 florimondmanca