import { PropSchema, serialize, deserialize, custom, setDefaultModelSchema, getDefaultModelSchema, primitive, SKIP } from "serializr"; import { Field } from "../../fields/NewDoc"; export namespace SerializationHelper { let serializing: number = 0; export function IsSerializing() { return serializing > 0; } export function Serialize(obj: Field): any { if (!obj) { return null; } 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) { return null; } 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: Function): void; withFields(fields: string[]): Function; } export function Deserializable(name: string): DeserializableOpts; export function Deserializable(constructor: Function): void; export function Deserializable(constructor: Function | string): DeserializableOpts | void { function addToMap(name: string, ctor: Function) { 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: Function) => { 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) ); }