diff options
Diffstat (limited to 'src/client/util/SerializationHelper.ts')
| -rw-r--r-- | src/client/util/SerializationHelper.ts | 130 | 
1 files changed, 130 insertions, 0 deletions
| diff --git a/src/client/util/SerializationHelper.ts b/src/client/util/SerializationHelper.ts new file mode 100644 index 000000000..7ded85e43 --- /dev/null +++ b/src/client/util/SerializationHelper.ts @@ -0,0 +1,130 @@ +import { PropSchema, serialize, deserialize, custom, setDefaultModelSchema, getDefaultModelSchema, primitive, SKIP } from "serializr"; +import { Field } from "../../new_fields/Doc"; + +export namespace SerializationHelper { +    let serializing: number = 0; +    export function IsSerializing() { +        return serializing > 0; +    } + +    export function Serialize(obj: Field): any { +        if (obj === undefined || obj === null) { +            return undefined; +        } + +        if (typeof obj !== 'object') { +            return obj; +        } + +        serializing += 1; +        if (!(obj.constructor.name in reverseMap)) { +            throw Error(`type '${obj.constructor.name}' not registered. Make sure you register it using a @Deserializable decorator`); +        } + +        const json = serialize(obj); +        json.__type = reverseMap[obj.constructor.name]; +        serializing -= 1; +        return json; +    } + +    export function Deserialize(obj: any): any { +        if (obj === undefined || obj === null) { +            return undefined; +        } + +        if (typeof obj !== 'object') { +            return obj; +        } + +        serializing += 1; +        if (!obj.__type) { +            throw Error("No property 'type' found in JSON."); +        } + +        if (!(obj.__type in serializationTypes)) { +            throw Error(`type '${obj.__type}' not registered. Make sure you register it using a @Deserializable decorator`); +        } + +        const value = deserialize(serializationTypes[obj.__type], obj); +        serializing -= 1; +        return value; +    } +} + +let serializationTypes: { [name: string]: any } = {}; +let reverseMap: { [ctor: string]: string } = {}; + +export interface DeserializableOpts { +    (constructor: { new(...args: any[]): any }): void; +    withFields(fields: string[]): Function; +} + +export function Deserializable(name: string): DeserializableOpts; +export function Deserializable(constructor: { new(...args: any[]): any }): void; +export function Deserializable(constructor: { new(...args: any[]): any } | string): DeserializableOpts | void { +    function addToMap(name: string, ctor: { new(...args: any[]): any }) { +        const schema = getDefaultModelSchema(ctor) as any; +        if (schema.targetClass !== ctor) { +            const newSchema = { ...schema, factory: () => new ctor() }; +            setDefaultModelSchema(ctor, newSchema); +        } +        if (!(name in serializationTypes)) { +            serializationTypes[name] = ctor; +            reverseMap[ctor.name] = name; +        } else { +            throw new Error(`Name ${name} has already been registered as deserializable`); +        } +    } +    if (typeof constructor === "string") { +        return Object.assign((ctor: { new(...args: any[]): any }) => { +            addToMap(constructor, ctor); +        }, { withFields: Deserializable.withFields }); +    } +    addToMap(constructor.name, constructor); +} + +export namespace Deserializable { +    export function withFields(fields: string[]) { +        return function (constructor: { new(...fields: any[]): any }) { +            Deserializable(constructor); +            let schema = getDefaultModelSchema(constructor); +            if (schema) { +                schema.factory = context => { +                    const args = fields.map(key => context.json[key]); +                    return new constructor(...args); +                }; +                // TODO A modified version of this would let us not reassign fields that we're passing into the constructor later on in deserializing +                // fields.forEach(field => { +                //     if (field in schema.props) { +                //         let propSchema = schema.props[field]; +                //         if (propSchema === false) { +                //             return; +                //         } else if (propSchema === true) { +                //             propSchema = primitive(); +                //         } +                //         schema.props[field] = custom(propSchema.serializer, +                //             () => { +                //                 return SKIP; +                //             }); +                //     } +                // }); +            } else { +                schema = { +                    props: {}, +                    factory: context => { +                        const args = fields.map(key => context.json[key]); +                        return new constructor(...args); +                    } +                }; +                setDefaultModelSchema(constructor, schema); +            } +        }; +    } +} + +export function autoObject(): PropSchema { +    return custom( +        (s) => SerializationHelper.Serialize(s), +        (s) => SerializationHelper.Deserialize(s) +    ); +}
\ No newline at end of file | 
