npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

dcerialize

v3.6.8

Published

Easy serialization through ES7/Typescript annotations

Downloads

577

Readme

Dcerialize

Easy serialization through ES7/Typescript annotations

This is a library to make serializing and deserializing complex JS objects a breeze. It works by applying meta data annotations (as described in ES7 proposal and experimental Typescript feature) to fields in a user defined class. It also aims to be compatible with Newtonsoft (a C# library for serialization).

Concepts

This library works by processing annotations on class types. Annotations are provided for reading (deserializing) and writing (serializing) values to and from json.

Once you have annotated your class types, you can use the Serialize* and Deserialize*functions to serialize and deserialize your data.

Example

    const ship = new Starship();
    /* assume values assigned */
    const json = Serialize(ship, () => Starship);
    /*
     json = {
        remainingFuel: 500.5,
        capt: {
            name: "Sparrow",
            onDuty: true,
            credits: { amount: 500, currency: "galactic" }
        },
        crew: [
            {
                name: "Bill",
                onDuty: true,
                credits: { amount: 0, currency: "galactic" }
            },
            {
                name: "Ben",
                onDuty: false,
                credits: { amount: 1500, currency: "galactic" }
            },
            {
                name: "Bob",
                onDuty: true,
                credits: { amount: 50, currency: "galactic" }
            }
        ],
        planetsVisited: {
            Tatooine: {
                timeVisited: "Mon Feb 05 2018 11:35:42 GMT+0100 (CET)",
                description: "desert"
            },
            Yavin4: {
                timeVisited: "Tue Feb 06 2018 11:35:42 GMT+0100 (CET)",
                description: "jungle"
            },
            Endor: {
                timeVisited: "Wed Feb 07 2018 11:35:42 GMT+0100 (CET)",
                description: "forest"
            }
        },
        cargo: {
            containers: 4,
            contents: ["lots", "of", "stuff"]
        }
     }
    */
    const instance = Deserialize(json, () => Starship);

Details

    class CrewMember {

        //unannotated properties are not serialized or deserialized, they are totally ignored
        localId :number;

        //serialize and deserialize the crew name as a string
        @autoserializeAs(() => String) name : string;

        //serialize the onDuty property as a boolean, don't deserialize it
        @serializeAs(() => Boolean) onDuty : boolean;

        //deserialize the happiness rating as a number, don't serialize it
        @deserializeAs(() => Number) happinessRating : number;

        //we only want to write our credit value, never deserialize it
        //we want to transform the value into a representation our server
        //understands which is not a direct mapping of number values.
        //use a custom serialization function instead of a type.
        @serializeUsing(CreditSerializer) credits : number;
    }

    class PlanetLog {

        // we handle the timeVisited field specially in our callbacks
        // we do not annotate it so that we can customize it ourselves
        timeVisited : Date;

        // serialize and deserialize description as a string
        @autoserializeAs(() => String) description : string;

        // when serializing our planet log we need to convert the timezone
        // of the timeVisited value from local time to galactic time
        // (This could also be done via @serializeUsing(Time.toGalacticTime))
        static onSerialized(instance : PlanetLog, json : IJsonObject) {
            json["timeVisited"] = Time.toGalacticTime(instance.timeVisited);
        }

        // when deserializing our planet log we need to convert the timezone
        // of the timeVisited value from galactic to local time
        // (This could also be done via @deserializeUsing(Time.toLocalTime))
        @onDeserialized
        static someFunction() {
            this.timeVisited = Time.toLocalTime(this.timeVisited);
        }

    }

    class Starship {

        // when writing our fuel value to the server, we have a number but the server expects a string
        // when reading our fuel value from the server, we receive a string but we want a number
        @serializeAs(() => String)
        @deserializeAs(() => Number)
        remainingFuel : number;

        // keys can be customized by providing a second argument to any of the annotations
        @autoserializeAs(() => CrewMember, "capt") captain : CrewMember;

        // serialize and deserialize the crew members as an array
        @autoserializeAsArray(() => CrewMember) : crew : Array<CrewMember>;

        // serialize and deserialize our planet log as an indexable map by planet name
        @autoserializeAsObjectMap(() => Planet) : planetsVisited : Indexable<PlanetLog>;

        // we don't have a specific format for cargo, so just serialize and deserialize it as normal json
        @autoserializeAsJSON() cargo : any;

    }

    // a function to transform our credit amount into a format our server understands
    function CreditSerializer(instance : { credits : number }) : JsonType {
        return { amount: instance.credits, currency: "galactic" };
    }

Annotations

When annotating your classes you can declare which fields get treated as which kinds of values and how they are read and written to and from json format. To specify how fields are written to json, use @serialize* annotations. For writing, use @deserialize*.

Most annotations take a class constructor. For primitives, use String, Number, Boolean, Date, or RegExp. For other types, provide the corresponding type constructor. All annotations take an optional argument customKey which will overwrite the corresponding key in the output. If no customKey is provided, the property key will be the same as defined in the class. For example, if our class has a field called protons but our server sends json with particles instead, we would use "particles" as the customKey value. If no customKey is provided, the property key will be the same as defined in the class.

If you want the same behavior for a property when serializing and deserializing, you can either tag that property with a @serialize* and @deserialize* or you can use @autoserializeXXX which will do this in a single annotation and behave exactly the same as @serialize* and @deserialize*. The only difference in behavior is that @autoserializeUsing() takes an argument of type SerializeAndDeserializeFns instead of a single function argument like it's siblings do.

@serializeAsMap will convert an es6 Map to an map object (ie an object with keys and values). Due to the syntax of JSON, keys can only be int or string. @deserializeAsMap will correctly deserialize into a Map.

Serialization

  • @serializeAs(type : () => ClassConstructor, customKey? : string)
  • @serializeAsObjectMap(type : () => ClassConstructor, customKey? : string)
  • @serializeAsArray(type : () => ClassConstructor, customKey? : string)
  • @serializeUsing(transform : SerializerFn, customKey? : string)
  • @serializeAsJson(customKey? : string)
  • @serializeAsMap(keyType: SerializableType<any>, valueType: SerializableType<any>, constructor?: () => SerializableType<any>, keyName?: string)

Deserialization

  • @deserializeAs(type : () => ClassConstructor, customKey? : string)
  • @deserializeAsArray(type : () => ClassConstructor, customKey? : string, handling?: ArrayHandling)
  • @deserializeAsObjectMap(type : () => ClassConstructor, customKey? : string)
  • @deserializeUsing(transform : DeserializerFn, customKey? : string)
  • @deserializeAsJson(customKey? : string)
  • @deserializeAsMap(keyType: () => SerializableType<any>, valueType: () => SerializableType<any>, constructor?: () => SerializableType<any>, keyName?: string)

Serialization and Deserialization

  • @autoserializeAs(type : () => ClassConstructor, customKey? : string)
  • @autoserializeAsObjectMap(type : () => ClassConstructor, customKey? : string)
  • @autoserializeAsArray(type : () => ClassConstructor, customKey? : string, handling?: ArrayHandling)
  • @autoserializeUsing(transforms : SerializeAndDeserializeFns, customKey? : string)
  • @autoserializeAsJson(customKey? : string)
  • @autoserializeAsMap(keyType: () => SerializableType<any>, valueType: () => SerializableType<any>, constructor?: () => SerializableType<any>, keyName?: string)

Array

There is two way of serializing an array :

Deserialize([1], itIsAnArray(() => Number))
DeserializeAsArray([1], () => Number)

Both are equivalent, but the former allow you to deserialize nested array

Deserialize([[1], [2]], itIsAnArray(itIsAnArray(() => Number)))

Or Set/Map of array

deserializeAsMap({v1: [1], v2: [2]}, () => String, itIsAnArray(itIsAnArray(() => Number)))

There is an optional second argument to itIsAnArray : a constructor used to deserialize into the correct type.

class MyArray<T> extends Array<T> {}
const instance = Deserialize([1], itIsAnArray(() => Number, () => MyArray));
assert(instance instanceof MyArray);

There is an optional third argument to itIsAnArray : An enum that specify how the deserialization is done when deserializing into an existing object.

  • ArrayHandling.Into: deserialize into each cell, grow or shrink array to fit serialized array size.
  • ArrayHandling.ConcatAtTheEnd: keep previous array, add the deserialized one at the end.
  • ArrayHandling.New: empty the array before adding element to it.
class MyArray<T> extends Array<T> {}
const instance = Deserialize([1], itIsAnArray(() => Number, () => MyArray), ArrayHandling.Into);
assert(instance instanceof MyArray);

Types

 type SerializationFn = <T>(target : T) => JsonType;
 type DeserializationFn = <T>(data : JsonType, target? : T, instantiationMethod? : InstantiationMethod) => T
 type SerializeAndDeserializeFns = {
     Serialize: SerializationFn,
     Deserialize: DeserializationFn
 }

Serializing Data to JSON

Calling any of the Serialize* family of methods will convert the input object into json. The output is a plain javascript object that has not had JSON.stringify called on it.

Functions for Serialization

Depending on how your data is structured there are a few options for serialization. You can work with single objects, maps of objects, or arrays of objects.

  • Serialize<T>(target : T, () => ClassConstructor<T>) => JsonObject

        /* takes a single object and serializes it using the provided class type. */
        const ship = new Starship();
        const json = Serialize(ship, () => Starship);
  • SerializeArray<T>(target : Array<T>, () => ClassConstructor<T>) => JsonArray

        /* takes an array of objects and serializes each entry using the provided class type */
        const ships : Array<Starship>;
        const json = SerializeArray(ships, () => Starship);
  • SerializeAsObjectMap<T>(type: ASerializableType<T>, keyName?: string) => JsonObject

        /* takes an indexable object ie `<T>{ [idx: string] : T }` and for each key serializes
         the object using the provided class type. */
        const ships : Indexable<Starship> = {
            ship1: new Starship(),
            ship2: new Starship()
        };
        const json = SerializeObjectMap(ships, () => Starship);
  • SerializeMap<T>(keyType: ASerializableType<any>, valueType: ASerializableType<any>, constructor?: () => IConstructable, keyName?: string) => JsonObject

        /* takes an es6 Map and for each (key, value) pair serializes the pair with the provided key and value type. */
        const ships : Map<string, Starship> = new Map([
                ["ship1", new Starship()],
                ["ship2", new Starship()]
            ]);
        const json = SerializeMap(ships, () => String, () => Starship);
  • SerializeSet<T>( source: T[], type: ASerializableType<T> ) => JsonType[]

        /* takes an es6 Set and for each element serializes
         the object using the provided class type. */
        const stars : new Set([1, 2, 3])
        const json = SerializeSet(stars, () => Number);
  • SerializeJson(target : any) => JsonType

     /* takes any value and serializes it as json, no structure is assumed
        and any serialization annotations on any processed objects are totally ignored. */
    
        const value = {}; /* anything that isn't a function */
        const json = SerializeJson(value);

If a function has a constructor parameter, it can be used to declare a constructor that inherit of the base constructor. For instance:

class MySet<T> extends Set<T> {}
class Test {
    @serializeAsSet(() => Number, () => MySet)
    public value: MySet<number>;
}

Deserializing From JSON

Calling any of the Deserialize* family of methods will convert the input json into an instance of the provided ClassConstructor or a plain JS object if that is preferred (Redux for example, expects plain objects and not instances)

The simplest way to deserialize a piece of JSON is to call Deserialize(json, () => type) on it. This function takes the provided type and pulls out all the properties you've tagged with @deserializeXXX or @autoserializeXXX. It will pump them (recursively) into a new instance of type which is returned. If your type marks a property for deserialization that is itself tagged with deserialization annotations, that property will be hydrated into it's type following the same deserialization algorithm.

Deserializing Into Existing Instances

It is also possible to re-use existing objects when deserializing with Deserialize(json, () => Type, target). You might want to do this so that you can maintain references to things even after updating their properties. This is handled exactly the same way as Deserialize(json, () => Type) except that it takes one additional argument, the object you want to deserialize properties into. If the target instance you provide is null or undefined, this behaves identically to Deserialize(json, () => Type), otherwise the deserialization will always use existing objects as write targets (if they are defined and of the expected type) instead of creating new ones.

const existingInstance = new Type();
const instance = Deserialize(json, () => Type, existingInstance);
expect(existingInstance === instance).toBe(true);

Deserializing Into Plain Objects

The instantiationMethod parameter can be used to change the way in which instances of the input type are created. With InstantiationMethod.New, the constructor will be invoked when a new instance needs to be created. With InstantiationMethod.ObjectCreate, the object will be created without invoking its constructor, which is useful for systems where constructed objects immediately freeze themselves. With InstantiationMethod.None, the deserializeXXX functions will return a plain object instead, which can be useful for systems like Redux that expect / require plain objects and not class instances.

import {Deserialize, Instances} from 'dcerialize';

class Immutable {

    public value : string;

    constructor(value : string) {
        this.value = value;
        Object.freeze(this);
    }

    public getValue() : string {
        return value;
    }

}

Deserialize({value: 'example'}, Immutable, InstantiationMethod.New);          // Error because of Object.freeze
Deserialize({value: 'example'}, Immutable, InstantiationMethod.ObjectCreate); // Immutable {value 'example'}
Deserialize({value: 'example'}, Immutable, InstantiationMethod.None);         // Object {value: 'example'}

The default InstantiationMethod can be changed with SetDefaultInstantiationMethod(instantiationMethod : InstantiationMethod)

Functions

  • Deserialize<T>(json : JsonObject, () => ClassConstructor<T>, target? : T) : T

        /* takes a single object and serializes it using the provided class type. */
    
        const json = {/* some values from server */};
        const existingInstance = new Starship();
        const instance = Deserialize(json, () => Starship); // make a new instance
    
        Deserialize(json, () => Starship, existing); // re-use our existing instance
  • DeserializeArray<T>(json : JsonArray, () => ClassConstructor<T>, target? : Array<T>) : Array<T>

        const json = [
            {/* some values from server */},
            {/* some values from server */},
            {/* some values from server */}
        ];
        const existingInstances = [ new Starship(), new Starship() ];
        const existingArray = [ new Starship() ];
    
        const array = DeserializeArray(json, () => Starship); // make a new array of instances
    
        /* re-use our existing array, if possible use existing instances in array, otherwise create new ones */
        DeserializeArray(json, () => Starship, existingArray);
  • DeserializeObjectMap<T>(json : JsonObject, () => ClassConstructor<T>, target? : Indexable<T>) : Indexable<T>

        const json = {
            ship0: {/* some values from server */},
            ship1: {/* some values from server */},
            ship2: {/* some values from server */}
        };
        const existingMap = {
            ship0: new Starship(),
            ship3: new Starship()
        };
    
        const map = DeserializeObjectMap(json, () => Starship); // make a new map of instances
    
        /* re-use our existing map, in the case of key collision,
           write new property values into existing instance
           otherwise create new ones */
        DeserializeObjectMap(json, () => Starship, existingMap);
  • DeserializeMap<T>(json : JsonObject, () => ClassConstructor<T>, target? : Indexable<T>) : Indexable<T>

        const json = {
            ship0: {/* some values from server */},
            ship1: {/* some values from server */},
            ship2: {/* some values from server */}
        };
        const existingMap = {
            ship0: new Starship(),
            ship3: new Starship()
        };
    
        const map = DeserializeMap(json, () => String, () => Starship); // make a new map of instances
    
        /* re-use our existing map, in the case of key collision,
           write new property values into existing instance
           otherwise create new ones */
        DeserializeMap(json, () => String, () => Starship, existingMap);
  • DeserializeJson(json : JsonType, target? : any) : any

     /* takes any value and deserializes it from json, no structure is assumed
        and any deserialization annotations on any processed objects are totally ignored. */
    
        const value = { /* anything that isn't a function */ };
        const json = DeserializeJson(value);

onSerialized Callback

A callback can be provided for when a class is serialized. To define the callback, add a static method onSerialized<T>(instance : T, json : IJsonObject) to the class that needs custom post processing. You can either return a new value from this function, or modify the json parameter.

class CrewMember {

    @autoserializeAs(() => String) firstName;
    @autoserializeAs(() => String) lastName;

    static onSerialized(instance : CrewMember, json : IJsonObject) {
        json["employeeId"] = instance.lastName.toUpperCase() + ", " + instance.firstName.toUpperCase();
    }

}

onDeserialized Callback

A callback can be provided for when a class is deserialized. Just add @onDeserialized on a function and it will be called on the instance after its class have been deserialized. You can add only one @onDeserialized callback to a class. The callback are inherited, but you have to call the callback of a parent yourself if the callback is overridden.

class CrewMember {

    @autoserializeAs(() => String) firstName;
    @autoserializeAs(() => String) lastName;

    @onDeserialized
    static callBack() {
        this.firstName = this.firstName.toLowerCase();
        this.lastName = this.lastName.toLowerCase();
    }
}

Inheriting Serialization

Serialization behavior is not inherited by subclasses automatically. To inherit a base class's serialization / deserialization behavior, tag the subclass with @inheritSerialization(() => ParentClass).

import { inheritSerialization } from 'dcerialize';

@inheritSerialization(() => User)
class Admin extends User {

}

Order of serialization

Properties of parents are serialized before those of the children. Inside a class, the order of serialization of property is the order of the decorator: first decorator is (de) serialized in first.

Customizing key transforms

Often your server and your client will have different property naming conventions. For instance, Rails / Ruby generally expects objects to have properties that are under_score_cased while most JS authors prefer camelCase. You can tell Dcerialize to use a certain key transform automatically when serializing and deserializing by calling SetSerializeKeyTransform(fn : (str : string) => string) and SetDeserializeKeyTransform(fn : (str : string) => string). A handful of transform functions are provided in this package or you can define your own function conforming to (key : string) => string.

  • The provided functions are:

    • CamelCase
    • UnderscoreCase
    • SnakeCase
    • DashCase

Note

When using SetDeserializeKeyTransform(fn : (str : string) => string) you need to provide a function that transforms the EXISTING keys to a format that allows indexing of the input object.

//in this example we expect the server to give us upper cased key names
//we need to map our local camel cased key to match the server provided key
//NOT the other way around.
SetDeserializeKeyTransform(function (value : string) : string {
    return value.toUpperCase();
});

class Test {
    @deserializeAs(() => String) value : string;
}

const json = {
    VALUE: "strValue",
};

const instance = Deserialize(json, () => Test);
expect(instance).toEqual({
    value: "strValue"
});

Runtime Typing

consider the following case :

class Living {
    whoIAm(): string{
        return "I am a living being";
    }
}

@inheritSerialization(() => Living)
class Dog extends Living {
    whoIAm(): string{
        return "I am a dog";
    }
}

@inheritSerialization(() => Living)
class Fish extends Living {
    whoIAm(): string{
        return "I am a fish";
    }
}

let animal: Living[] = [new Fish(), new Dog(), new Living()];
let json = SerializeAsArray(animal, () => Living);

At deserialization you end up with a bunch of Living, but no dog nor fish. The runtime serialization will detect at runtime that there are other types in the array (it also work with simple member). To do so, a $type attribute will be added to the serialized object. You have to provide the content of this attribute (this allows you to be compatible with Newtonsoft TypeNameHandling.Objects settings). The Runtime typing is enable as follow :

let animal: Living[] = [new Fish(), new Dog(), new Living()];
RuntimeTypingEnable();
RuntimeTypingSetTypeString(Fish, "It's a fish");
RuntimeTypingSetTypeString(Dog, "and that is a dog");
RuntimeTypingSetTypeString(Living, "every thing is fine as long as there are no collision");

let json = SerializeArray(animal, () => Living);
...
let animal_d = DeserializeArray(json, () => Living);
RuntimeTypingDisable();
RuntimeTypingResetDictionary();

Alternatively, you can decorate a class with the decorator @typeString("It's a fish") to specify the string for a class.

@typeString("every thing is fine as long as there are no collision")
class Living {
    whoIAm(): string{
        return "I am a living being";
    }
}

Note

In the previous case, nothing will enforce a class to be a subclass of Living. The last two line are not mandatory (if you want to serialize/deserialize more). Unfortunately, I didn't found a way to add Runtime Typing for Map (or object used as map). You can do the following (use a class that inherits form Map):

class Satellite {
    @serializeAs(() => String)
    public name: string;
    constructor(nameArg: string) {
        this.name = nameArg;
    }
}
@inheritSerialization(() => Satellite)
class Moon extends Satellite {}
class MyDico extends Map<string, Satellite> {}
class Test0 {
    @serializeAsMap(() => String, () => Satellite, () => MyDico)
    public dico1: MyDico;
}
const s = new Test0();
s.dico1 = new MyDico([["1", new Moon("Europa")], ["2", new Satellite("Adrastea")]]);
s.dico1.set("3" , new Moon("Callisto"));
RuntimeTypingEnable();
RuntimeTypingSetTypeString(Moon, "my Moon type");
RuntimeTypingSetTypeString(Satellite, "my Satellite type");
RuntimeTypingSetTypeString(Test0, "my Test0 type");
RuntimeTypingSetTypeString(MyDico, "my MyDico type");
const json = Serialize(s, () => Test0);
// "{"$type":"my Test0 type","dico1":{"1":{"$type":"my Moon type","name":"Europa"},"2":{"$type":"my Satellite type","name":"Adrastea"},"3":{"$type":"my Moon type","name":"Callisto"},"$type":"my MyDico type"}}"

Default value

@emitDefaultValue

if this decorator has false in it's argument, the variable will not be serialized if it's value is the default. | type | default value | |:-------:|:-------------:| | number | 0 | | string | "" | | boolean | false | | object | null |

@defaultValue

This decorator permits to change the default value. Object are compared by references. It only works with emitDefaultValue set to false.

class Test {
    @emitDefaultValue(false)
    @serializeAs(() => Boolean)
    public valueDefault: boolean = false;

    @emitDefaultValue(false)
    @serializeAs(() => Boolean)
    @defaultValue(true)
    public valueFalse: boolean = false;

    @serializeAs(() => Boolean)
    @defaultValue(true)
    @emitDefaultValue(false)
    public valueTrue: boolean = true;
}

const t = new Test();
const json = Serialize(t, () => Test);
/* { valueNotDefault2: 1 } */

Note

During deserialization, if a member decorated with @emitDefaultValue(false) is not in the parsed json, it will be initialized by its default value. Be careful if an object is the default value, the object will be inserted as it is during serialization. If it appears in multiple place (ie its a sigle object with multiple reference to it).

Reference and circular reference

Without this option, the following happens :

class Test {
    @autoserializeAs(() => Number)
    public value: number = 10;
}

class Test0 {
    @autoserializeAs(() => Test)
    public value0: Test;
    @autoserializeAs(() => Test)
    public value1: Test;
}
const t = new Test();
const t0 = new Test0();
t0.value0 = t0.value1 = t;
const json = Serialize(t0, () => Test0); /* json = {"value0":{"value":10},"value1":{"value":10}} */
const obj = Deserialize(json, () => Test0);
obj.value1 == obj.value0; /* false */

Even if t0.value0 and t0.value1 are the same, they are serialized as to separate object. Event worse, if the references form a loop, the serialization will crash. With SetRefCycleDetection, each object will have its own id, and will be referenced for any other references except the first one.

class Test {
    @deserializeAsJson()
    public value: number = 10;
}

class Test0 {
    @deserializeAsJson()
    public value0: Test;
    @deserializeAsJson()
    public value1: Test;
}
const json = {
    $id: 1,
    value0: { $id: 2, value: 1 },
    value1: { $ref: 2 }
};
RefCycleDetectionEnable();
const instance = Deserialize(json, () => Test0);
RefClean();
RefCycleDetectionDisable();
obj.value1 == obj.value0; /*true*/

You can also override RefCycleDetection with the class decorator isReference.

@isReference(true)
class Test {
    @deserializeAsJson()
    public value: number = 10;
}

class Test0 {
    @deserializeAsJson()
    public value0: Test;
    @deserializeAsJson()
    public value1: Test;
}
const json = {
    value0: { $id: 2, value: 1 },
    value1: { $ref: 2 }
};
RefCycleDetectionEnable();
const instance = Deserialize(json, () => Test0);
RefClean();
RefCycleDetectionDisable();
obj.value1 == obj.value0; /* true */

Note

For architectural code choice, during deserialization, the parser must read the object id before any mention of it's reference. For example the following json would crash during deserialization :

{
    $id: 1,
    value1: { $ref: 2 },
    value2: { $id: 2, value: 1 }
}

Because of the reference of object 2 would be read before its id.

The generated json from SetRefCycleDetection is compatible with PreserveReferencesHandling option form newtonsoft.

Selective serialization

You can serialize only some member and take this decision at runtime using bitmask : affect a bitmask to a member with @serializeBitMask (up to 2^53). Before serialization, you can set a global bitmask using SelectiveSerialization. If and only if the bitmask of the member AND the global bitmask evaluate to true, then the member will be serialized.

{
class Test {
    @serializeBitMask(1)
    @serializeAs(() => Number)
    public v1: number = 1;
    @serializeBitMask(3)
    @serializeAs(() => Number)
    public v2: number = 2;
    @serializeAs(() => Number)
    @serializeBitMask(2)
    public v3: number = 3;
}
SelectiveSerialization(1);
Serialize(s, () => Test); // {v1: 1, v2: 2}
SelectiveSerialization(2);
Serialize(s, () => Test); // {v2: 2, v3: 3}
SelectiveSerialization(3);
Serialize(s, () => Test); // {v1: 1, v2: 2, v3: 3}
}

To reset the selective serialization do a call of SelectiveSerialization() without any parameters.

Serialize and deserialize Infinities and NaN

By default, JSON.stringify(Number.POSITIVE_INFINITY) return null. But we can provide an helper function that will stringify these numbers (Infinities and NaN) to custom string. You can code your own, or use the one provided by this library : stringifyNumber. The call will be JSON.stringify(my object with Infinities, stringifyNumber); The same problem arise during parse. Use JSON.parse('"Infinity"', parseNumber);. These two functions are compatible with the FloatFormatHandling Newtonsoft option.

define your own module to deal with references

you can set your own module to deal with references. see the source ref.ts for an example.

setRefHandler(new RefHandler());

Summary

Decorator

| serialize | deserialize | autoserialize | |-------------------|---------------------|-----------------------| | serializeAs | deserializeAs | autoserializeAs | | serializeAsArray | deserializeAsArray | autoserializeAsArray | | serializeAsJson | deserializeAsJson | autoserializeAsJson | | serializeAsObjectMap | deserializeAsObjectMap | autoserializeAsObjectMap | | serializeAsMap | deserializeAsMap | autoserializeAsMap | | serializeAsSet | deserializeAsSet | autoserializeAsSet | | serializeUsing | deserializeUsing | autoserializeUsing |

Other decorators

  • inheritSerialization
  • serializeBitMask
  • emitDefaultValue
  • defaultValue
  • onDeserialized
  • typeString

Setting Accessor

  • SetRefCycleDetection
  • RefClean
  • SetDefaultInstantiationMethod
  • SetDeserializeKeyTransform
  • SetSerializeKeyTransform
  • RuntimeTypingResetDictionary
  • RuntimeTypingSetTypeString
  • RuntimeTypingEnable
  • RuntimeTypingDisable
  • SelectiveSerialization

Callback

  • onSerialized

Good to know

  • You can serialize and deserialize an enum using @serializeAsJson. They will be converted using their integer representation.
  • To reset the selective serialization do SelectiveSerialization().
  • You need to specify the type of each member that you want to (de)serialize. Use String, Number, Boolean, Date, or RegExp for primitives types
  • Serialization return an object that can be stringified (use JSON.stringify after a call to Serialize).
  • Deserialization expect an object (use JSON.parse before a call to Deserialize).
  • You must use the @inheritSerialization if you want to serialize object with inheritance.
  • Use RefClean if you want that $id start to one again.
  • You don't need to call RuntimeTypingDisable after a serialization if you want to use it again.
  • @serializeAsArray expect a non array type (ie if it's an array of Boolean, you should give Boolean as parameter). Same goes for @serializeAsObjectMap, @serializeAsMap and @serializeAsSet.
  • @serializeAsMap works on es6 Map object.
  • if you create a loop between files by adding decorator, YOU MUST set the option @emitDecoratorMetadata to false in your tsconfig.json. Otherwise your project will fail to load.

Contribute

Please follow this link for detailed explanations on how to contribute to the project