2024-10-04 16:03:16 +02:00
< p align = "center" >
< a href = "https://code.zeptotech.net/Sharkitek/Core" >
< picture >
< img alt = "Sharkitek logo" width = "200" src = "https://code.zeptotech.net/Sharkitek/Core/raw/branch/main/logo.svg" / >
< / picture >
< / a >
< / p >
< h1 align = "center" >
Sharkitek
< / h1 >
2022-07-27 11:34:52 +02:00
2024-10-04 16:03:16 +02:00
< h4 align = "center" >
< a href = "https://code.zeptotech.net/Sharkitek/Core" > Documentation< / a > |
< a href = "https://code.zeptotech.net/Sharkitek/Core" > Website< / a >
< / h4 >
< p align = "center" >
TypeScript library for well-designed model architectures
< / p >
< p align = "center" >
2025-03-30 13:08:59 +02:00
< img alt = "Tests status" src = "https://code.zeptotech.net/Sharkitek/Core/badges/workflows/test.yaml/badge.svg" / >
< a href = "https://bundlephobia.com/package/ @sharkitek/core " target = "_blank" >
< img alt = "Bundle size" src = "https://badgen.net/bundlephobia/minzip/ @sharkitek/core " />
< / a >
2025-03-30 12:51:04 +02:00
< a href = "https://www.npmjs.com/package/ @sharkitek/core " target = "_blank" >
2025-03-30 13:08:59 +02:00
< img alt = "Latest release" src = "https://badgen.net/npm/v/ @sharkitek/core " />
2025-03-30 12:51:04 +02:00
< / a >
2025-03-30 13:08:59 +02:00
< a href = "https://bundlephobia.com/package/ @sharkitek/core " target = "_blank" >
< img alt = "Bundle size" src = "https://badgen.net/bundlephobia/dependency-count/ @sharkitek/core " />
< / a >
< img alt = "Latest release" src = "https://badgen.net/npm/types/ @sharkitek/core " />
2024-10-04 16:03:16 +02:00
< / p >
2024-10-04 14:58:00 +02:00
2022-08-01 14:48:07 +02:00
## Introduction
2025-03-30 12:51:04 +02:00
Sharkitek is a lightweight Javascript / TypeScript library designed to ease development of models.
2022-08-01 14:48:07 +02:00
2025-03-30 13:16:00 +02:00
```shell
yarn add @sharkitek/core
```
2022-11-01 19:13:21 +01:00
With Sharkitek, you define the architecture of your models by specifying their properties and their types.
2025-03-30 12:51:04 +02:00
Then, you can use the defined methods like `serialize` , `parse` , `patch` or `serializeDiff` .
2022-08-01 14:48:07 +02:00
```typescript
2025-03-30 12:51:04 +02:00
class Example
2022-08-01 14:48:07 +02:00
{
2025-03-30 12:51:04 +02:00
static model = defineModel({
Class: Example,
properties: {
id: s.property.numeric(),
name: s.property.string(),
},
identifier: "id",
});
id: number;
name: string;
2022-08-01 14:48:07 +02:00
}
```
2025-03-30 12:51:04 +02:00
## Quick start
**Note**: by convention, we define our models in a `model` static variable in the model's class. It is a good way to keep your model declaration near the actual class, and its usage will be more natural.
2022-08-01 14:48:07 +02:00
2025-03-30 12:51:04 +02:00
### Model definition
2022-08-01 14:48:07 +02:00
```typescript
/**
* A person.
*/
2025-03-30 12:51:04 +02:00
class Person
2022-08-01 14:48:07 +02:00
{
2025-03-30 12:51:04 +02:00
static model = defineModel({
Class: Person,
properties: {
id: s.property.numeric(),
name: s.property.string(),
email: s.property.string(),
createdAt: s.property.date(),
active: s.property.boolean(),
},
identifier: "id",
});
id: number;
name: string;
email: string;
createdAt: Date;
2022-11-01 19:13:21 +01:00
active: boolean = true;
2022-08-01 14:48:07 +02:00
}
```
```typescript
/**
* An article.
*/
2025-03-30 12:51:04 +02:00
class Article
2022-08-01 14:48:07 +02:00
{
2025-03-30 12:51:04 +02:00
static model = defineModel({
Class: Article,
properties: {
id: s.property.numeric(),
title: s.property.string(),
authors: s.property.array(s.property.model(Person)),
text: s.property.string(),
evaluation: s.property.decimal(),
tags: s.property.array(
s.property.object({
name: s.property.string(),
})
),
},
identifier: "id",
});
2022-11-01 19:13:21 +01:00
id: number;
title: string;
2025-03-30 12:51:04 +02:00
authors: Person[] = [];
2022-11-01 19:13:21 +01:00
text: string;
evaluation: number;
2024-10-04 14:58:00 +02:00
tags: {
name: string;
}[];
2022-08-01 14:48:07 +02:00
}
```
2025-03-30 12:51:04 +02:00
```typescript
/**
* A model with composite keys.
*/
class CompositeKeys
{
static model = defineModel({
Class: CompositeKeys,
properties: {
id1: s.property.numeric(),
id2: s.property.string(),
},
identifier: ["id1", "id2"],
});
id1: number;
id2: string;
}
```
### Model functions
#### Serialization
```typescript
const instance = new Person();
instance.id = 1;
instance.createdAt = new Date();
instance.name = "John Doe";
instance.email = "john@doe .test";
instance.active = true;
const serialized = Person.model.model(instance).serialize();
console.log(serialized); // { id: 1, createdAt: "YYYY-MM-DDTHH:mm:ss.sssZ", name: "John Doe", email: "john@doe .test", active: true }
```
#### Deserialization
```typescript
const instance = Person.model.parse({
id: 1,
createdAt: "2011-10-05T14:48:00.000Z",
name: "John Doe",
email: "john@doe .test",
active: true,
});
console.log(instance instanceof Person); // true
console.log(instance.createdAt instanceof Date); // true
```
#### Patch
```typescript
const instance = Person.model.parse({
id: 1,
createdAt: "2011-10-05T14:48:00.000Z",
name: "John Doe",
email: "john@doe .test",
active: true,
});
instance.name = "Johnny";
// Patch serialized only changed properties and the identifier.
console.log(Person.model.model(instance).patch()); // { id: 1, name: "Johnny" }
// If you run it one more time, already patched properties will not be included again.
console.log(Person.model.model(instance).patch()); // { id: 1 }
```
#### Identifier
```typescript
const instance = new CompositeKeys();
instance.id1 = 5;
instance.id2 = "foo";
const instanceIdentifier = CompositeKeys.model.model(instance).getIdentifier();
console.log(instanceIdentifier); // [5, "foo"]
```
2022-08-01 14:48:07 +02:00
## API
### Types
Types are defined by a class extending `Type` .
Sharkitek defines some basic types by default, in these classes:
2025-03-30 12:51:04 +02:00
- `BooleanType` : boolean value in the model, boolean value in the serialized object.
2022-08-01 14:48:07 +02:00
- `StringType` : string in the model, string in the serialized object.
- `NumericType` : number in the model, number in the serialized object.
- `DecimalType` : number in the model, formatted string in the serialized object.
2022-09-18 19:47:38 +02:00
- `DateType` : date in the model, ISO formatted date in the serialized object.
2022-08-01 14:48:07 +02:00
- `ArrayType` : array in the model, array in the serialized object.
2024-10-04 14:58:00 +02:00
- `ObjectType` : object in the model, object in the serialized object.
2022-08-01 14:48:07 +02:00
- `ModelType` : instance of a specific class in the model, object in the serialized object.
2024-10-04 14:58:00 +02:00
When you are defining a property of a Sharkitek model, you must provide its type by instantiating one of these classes.
2022-08-01 14:48:07 +02:00
```typescript
2025-03-30 12:51:04 +02:00
class Example
2022-08-01 14:48:07 +02:00
{
2025-03-30 12:51:04 +02:00
static model = defineModel({
Class: Example,
properties: {
foo: s.property.define(new StringType()),
},
});
2022-11-01 19:13:21 +01:00
foo: string;
2022-08-01 14:48:07 +02:00
}
```
2025-03-30 12:51:04 +02:00
To ease the use of these classes and reduce read complexity, properties of each type are easily definable with a function for each type.
2022-08-01 14:48:07 +02:00
2025-03-30 12:51:04 +02:00
- `BooleanType` => `s.property.boolean`
2024-10-04 14:58:00 +02:00
- `StringType` => `s.property.string`
- `NumericType` => `s.property.numeric`
- `DecimalType` => `s.property.decimal`
- `DateType` => `s.property.date`
- `ArrayType` => `s.property.array`
- `ObjectType` => `s.property.object`
- `ModelType` => `s.property.model`
2022-08-01 14:48:07 +02:00
2025-03-30 12:51:04 +02:00
Type implementers should provide a corresponding function for each defined type. They can even provide multiple functions or constants with predefined parameters. For example, we could define `s.property.stringArray()` which would be similar to `s.property.array(s.property.string())` .
2022-08-01 14:48:07 +02:00
```typescript
2025-03-30 12:51:04 +02:00
class Example
2022-08-01 14:48:07 +02:00
{
2025-03-30 12:51:04 +02:00
static model = defineModel({
Class: Example,
properties: {
foo: s.property.string(),
},
});
2024-10-04 14:58:00 +02:00
foo: string;
2022-08-01 14:48:07 +02:00
}
```
### Models
2025-03-30 12:51:04 +02:00
#### `model(instance)`
Get a model class (which has all the sharkitek models' functions) from a model instance.
```typescript
const model = definedModel.model(modelInstance);
```
2022-08-01 14:48:07 +02:00
#### `serialize()`
Serialize the model.
Example:
```typescript
2025-03-30 12:51:04 +02:00
const serializedObject = definedModel.model(modelInstance).serialize();
2022-08-01 14:48:07 +02:00
```
2025-03-30 12:51:04 +02:00
#### `parse(serializedObject)`
2022-08-01 14:48:07 +02:00
Deserialize the model.
Example:
```typescript
2025-03-30 12:51:04 +02:00
const modelInstance = definedModel.parse({
2022-08-01 14:48:07 +02:00
id: 5,
title: "Hello World!",
users: [
{
id: 6,
name: "TEST",
},
],
});
```
#### `serializeDiff()`
Serialize the difference between current model state and original one.
Example:
```typescript
2025-03-30 12:51:04 +02:00
const modelInstance = definedModel.parse({
2022-08-01 14:48:07 +02:00
id: 5,
title: "Hello World!",
users: [
{
id: 6,
name: "TEST",
},
],
});
2025-03-30 12:51:04 +02:00
modelInstance.title = "A new title for a new world";
2022-08-01 14:48:07 +02:00
2025-03-30 12:51:04 +02:00
const result = definedModel.model(modelInstance).serializeDiff();
2022-08-01 14:48:07 +02:00
// if `id` is defined as the model identifier:
// result = { id: 5, title: "A new title for a new world" }
// if `id` is not defined as the model identifier:
// result = { title: "A new title for a new world" }
```
#### `resetDiff()`
Set current properties values as original values.
Example:
```typescript
2025-03-30 12:51:04 +02:00
const modelInstance = definedModel.parse({
2022-08-01 14:48:07 +02:00
id: 5,
title: "Hello World!",
users: [
{
id: 6,
name: "TEST",
},
],
});
2025-03-30 12:51:04 +02:00
modelInstance.title = "A new title for a new world";
2022-08-01 14:48:07 +02:00
2025-03-30 12:51:04 +02:00
definedModel.model(modelInstance).resetDiff();
2022-08-01 14:48:07 +02:00
2025-03-30 12:51:04 +02:00
const result = definedModel.model(modelInstance).serializeDiff();
2022-08-01 14:48:07 +02:00
// if `id` is defined as the model identifier:
// result = { id: 5 }
// if `id` is not defined as the model identifier:
// result = {}
```
2024-10-05 17:36:03 +02:00
#### `patch()`
2022-08-01 14:48:07 +02:00
Get difference between original values and current ones, then reset it.
Similar to call `serializeDiff()` then `resetDiff()` .
```typescript
2025-03-30 12:51:04 +02:00
const modelInstance = definedModel.parse({
2022-08-01 14:48:07 +02:00
id: 5,
title: "Hello World!",
users: [
{
id: 6,
name: "TEST",
},
],
});
2025-03-30 12:51:04 +02:00
modelInstance.title = "A new title for a new world";
2022-08-01 14:48:07 +02:00
2025-03-30 12:51:04 +02:00
const result = definedModel.model(modelInstance).patch();
2022-08-01 14:48:07 +02:00
// if `id` is defined as the model identifier:
// result = { id: 5, title: "A new title for a new world" }
// if `id` is not defined as the model identifier:
// result = { title: "A new title for a new world" }
```