https://github.com/ynixt/typescript-flat-serializer
A typescript library to serialize/deserialize classes to/from string in a flat format. Supports inheritance, circular reference and more
https://github.com/ynixt/typescript-flat-serializer
flat serialization serializer typescript
Last synced: 10 months ago
JSON representation
A typescript library to serialize/deserialize classes to/from string in a flat format. Supports inheritance, circular reference and more
- Host: GitHub
- URL: https://github.com/ynixt/typescript-flat-serializer
- Owner: ynixt
- License: mit
- Created: 2023-06-03T03:21:15.000Z (almost 3 years ago)
- Default Branch: master
- Last Pushed: 2024-03-21T02:12:22.000Z (about 2 years ago)
- Last Synced: 2024-11-13T11:57:32.996Z (over 1 year ago)
- Topics: flat, serialization, serializer, typescript
- Language: TypeScript
- Homepage:
- Size: 224 KB
- Stars: 3
- Watchers: 1
- Forks: 0
- Open Issues: 1
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# typescript-flat-serializer

[](https://coveralls.io/github/ynixt/typescript-flat-serializer?branch=master)
[](https://snyk.io/test/github/ynixt/typescript-flat-serializer?targetFile=package.json)



A typescript library to serialize/deserialize classes to/from string in a flat format. Supports inheritance, circular reference and more
## Summary
1. [Why](#why)
2. [Installation](#installation)
3. [Usage](#usage)
4. [API](#API)
5. [Important notes](#important-notes)
6. [Inspired by](#inspired-by)
## Why
When I was developing a electron app I needed a serializer that fill some requirements to use with
[IPC](https://www.electronjs.org/docs/latest/tutorial/ipc):
- After serialize I wanted that the object keep the methods (right prototype)
- I needed inheritance
- Supports circular reference is always good
## Installation
`npm install typescript-flat-serializer --save`
## Usage
### With decorators
#### Preparation
If you want to use our decorators you also need to set **experimentalDecorators** and **emitDecoratorMetadata**
to true into the tsconfig.json file.
For example:
```json
{
"compilerOptions": {
...
"emitDecoratorMetadata": true,
"experimentalDecorators": true,
...
}
}
```
#### Usage
First let's create some models
```typescript
import { TSFlatCollection, TSFlatObject } from "typescript-flat-serializer";
@TSFlatObject()
export abstract class Animal {
protected constructor(public name: string) {
}
}
// This decorator NEEDS to be placed on every class that you want to serialize.
// Without this decorator the behavior will be like stringify/parse from JSON.
@TSFlatObject()
export class Dog extends Animal {
// This decorator will take care of serialize/deserialize our collection
@TSFlatCollection({collectionType: "set"})
favoriteFoods: Set;
constructor(name: string, public beautiful: boolean, favoriteFoods: Set) {
super(name);
this.favoriteFoods = favoriteFoods;
}
}
@TSFlatObject()
export class Food extends Animal {
constructor(name: string) {
super(name);
}
}
```
Now we only need to serialize/deserialize the animal.
```typescript
import { parse, stringify } from "typescript-flat-serializer";
const foods = new Set([new Food('all')])
const animal: Animal = new Dog('Luffy', true, foods);
// Let's go serialize our animal
const str = stringify(animal)
// value of str: [{"name":"Luffy","beautiful":true,"favoriteFoods":"#_1_#","__class__":"Dog"},["#_2_#"],{"name":"all","__class__":"Food"}]
// And now we can deserialize the animal
const parsedAnimal = parse(str);
```
You can find another examples of utilisation on [tests](https://github.com/ynixt/typescript-flat-serializer/tree/master/test).
### Without decorators
First let's create some models
```typescript
import { registerTSFlat } from "typescript-flat-serializer";
export abstract class Animal {
protected constructor(public name: string) {
registerTSFlat(
{
target: Animal,
}
);
}
}
export class Dog extends Animal {
favoriteFoods: Set;
constructor(name: string, public beautiful: boolean, favoriteFoods: Set) {
super(name);
registerTSFlat(
{
target: Dog,
},
{ collectionName: 'favoriteFoods', options: {collectionType: 'set'} }
)
this.favoriteFoods = favoriteFoods;
}
}
export class Food extends Animal {
constructor(name: string) {
super(name);
registerTSFlat(
{
target: Food,
},
)
}
}
```
Now we only need to serialize/deserialize the animal.
```typescript
import {parse, stringify} from "typescript-flat-serializer";
const foods = new Set([new Food('all')])
const animal: Animal = new Dog('Luffy', true, foods);
// Let's go serialize our animal
const str = stringify(animal)
// value of str: [{"name":"Luffy","beautiful":true,"favoriteFoods":"#_1_#","__class__":"Dog"},["#_2_#"],{"name":"all","__class__":"Food"}]
// And now we can deserialize the animal
const parsedAnimal = parse(str);
```
## API
### Decorators
#### **@TSFlatObject**
Used to make a class serializable.
##### **Example**
```typescript
@TSFlatObject()
export class Dog {
}
```
#### **@TSFlatCollection**
Used do make a Array|Set|Map|Dictionary
##### **Example**
```typescript
@TSFlatObject()
export class Dog {
@TSFlatCollection({collectionType: "map"})
placesVisited: Map
}
```
##### **Parameters**
**collectionType**
Type: [`CollectionTypeString`](#collectionTypeString)
Optional: `false`
Description: The way to specify the type of collection
#### **@TSFlatProperty**
Used modify the serialization/deserialization of a property.
##### **Example**
```typescript
@TSFlatObject()
export class Dog {
@TSFlatCollection({collectionType: "map"})
@TSFlatProperty({
beforeStringify: (m) => {
m.set('Brazil', true);
}
})
placesVisited: Map
}
```
##### **Parameters**
**options**
Type: [`TSFlatPropertyOptions`](#tSFlatPropertyOptions)
Optional: `true`
Description: The option to customize the serialization/deserialization of the target property.
### Methods
#### **registerTSFlat**
Used do register an object and its items.
```
registerTSFlat(objectRegistration: FlatObjectRegistration, ...items: FlatItem[])
```
##### **Parameters**
**objectRegistration**
Type: `FlatObjectRegistration`
Optional: `false`
Description: Information about the class that we want to make serializable.
Type: `FlatItem[]`
Optional: `true`
Description: Items that this class have that we want to make serializable.
---
#### **stringify**
Used to serialize a object.
```
stringify(obj: any, options?: StringifyOptions): string
```
##### **Parameters**
**obj**
Type: `any`
Optional: `false`
Description: The object that will be serialized
**options**
Type: [`StringifyOptions`](#stringifyOptions)
Optional: `true`
Description: Custom options to the serialization
##### **Return**
`string`
---
#### **parse**
Used to deserialize a string into a object.
```
parse(str: string): T
```
##### **Return**
`T`
### Definitions
#### **Types**
##### **FlatObjectRegistration**
```typescript
export type FlatObjectRegistration = {
target: Type,
options?: TSFlatObjectProperties
}
```
##### **FlatItem\**
```typescript
export type FlatItem = FlatPropertyRegistration | FlatCollectionRegistration;
```
##### **FlatPropertyRegistration\**
```typescript
export type FlatPropertyRegistration = {
propertyName: (keyof T), options?: TSFlatPropertyOptions,
}
```
##### **FlatCollectionRegistration\**
```typescript
export type FlatCollectionRegistration = {
collectionName: (keyof T), options: TSFlatCollectionOptions
}
```
##### **CollectionTypeString**
```typescript
export type CollectionTypeString = 'array' | 'dictionary' | 'map' | 'set';
```
##### **TSFlatPropertyOptions**
```typescript
export type PropertyTransformer = (property: any) => any;
export interface TSFlatPropertyMetadata {
beforeStringify?: PropertyTransformer;
afterParse?: PropertyTransformer;
}
export interface TSFlatPropertyOptions extends TSFlatPropertyMetadata {
}
```
#### **stringifyOptions**
```typescript
export type StringifyOptions = {
rFDCOptions?: RFDCOptions
}
export type CustomWayOfCloningObjectMap = Map, (obj: any) => any>;
export type RFDCOptions = {
customWayOfCloningObject?: CustomWayOfCloningObjectMap
}
```
## Important notes
### Cloning
This library before the serialization makes a clone of the object. By default the cloning supports the types:
- Object
- Array
- Number
- String
- null
- Date
- undefined
- Buffer
- TypedArray
- Map
- Set
- Function
- AsyncFunction
- GeneratorFunction
- arguments
To support other type, like DateTime of [Luxon](https://github.com/moment/luxon/), you should do something like that:
```typescript
const customWayOfCloningObject: CustomWayOfCloningObjectMap = new Map();
const rFDCOptions: RFDCOptions = {
customWayOfCloningObject
}
customWayOfCloningObject.set(DateTime, (obj) => DateTime.fromMillis(obj.toMillis()));
const str = stringify(obj, {rFDCOptions});
```
## Inspired by
- [WebReflection/flatted](https://github.com/WebReflection/flatted)
- [GillianPerard/typescript-json-serializer](https://github.com/GillianPerard/typescript-json-serializer)
- [davidmarkclements/rfdc](https://github.com/davidmarkclements/rfdc)