mirror of
https://github.com/twentyhq/twenty
synced 2026-04-21 21:47:38 +00:00
# Introduction
In this PR we're introducing mainly two branded type signatures for both
`JsonbProperty` entities properties and `SerializedRelation` (jsonb
serialized property storing another entity id).
Allowing to dynamically map over them later in order to build universal
`jsonb` `serialized` relations.
## `JsonbProperty`
A branded wrapper type that marks entity properties stored as PostgreSQL
JSONB columns. It adds a phantom brand `__JsonbPropertyBrand__` to
object types while leaving primitives unchanged. The branded key is
optional and typed as never, also omitted when transpiled to
`UniversalFlat`
**Should be used at entities lvl only:**
```typescript
@Column({ type: 'jsonb', nullable: false })
gridPosition: JsonbProperty<GridPosition>;
@Column({ nullable: false, type: 'jsonb', default: [] })
publishedVersions: JsonbProperty<string[]>;
```
## `SerializedRelation`
A branded string type that marks foreign key IDs stored inside JSONB
objects. These are entity references serialized within a JSONB column
rather than being a regular database foreign key.
**Usage in jsonb property generic***
```ts
type FieldMetadataRelationSettings = {
relationType: RelationType;
onDelete?: RelationOnDeleteAction;
joinColumnName?: string | null;
junctionTargetFieldId?: SerializedRelation;
};
```
## `FormatJsonbSerializedRelation<T>`
A transformation type that processes JSONB properties for universal
entity mapping. It:
1. Detects properties with the `JsonbProperty` brand
2. Finds `SerializedRelation` properties
3. Renames them from `*Id` to `*UniversalIdentifier`
4. Removes the brand from the output type ( optional though )
```typescript
// Input: JsonbProperty<{ targetFieldMetadataId: SerializedRelation }>
// Output: { targetFieldMetadataUniversalIdentifier: SerializedRelation }
```
## Result
An example of the dynamic type mapping, through a type-test example
```ts
type SettingsTestCase = UniversalFlatFieldMetadata<
| FieldMetadataType.RELATION
| FieldMetadataType.NUMBER
| FieldMetadataType.TEXT
>['settings']
type SettingsExpectedResult =
| {
relationType: RelationType;
onDelete?: RelationOnDeleteAction | undefined;
joinColumnName?: string | null | undefined;
junctionTargetFieldUniversalIdentifier?: SerializedRelation | undefined;
}
| {
dataType?: NumberDataType | undefined;
decimals?: number | undefined;
type?: FieldNumberVariant | undefined;
}
| {
displayedMaxRows?: number | undefined;
}
| null;
type Assertions = [
Expect<Equal<SettingsTestCase, SettingsExpectedResult>>,
]
```
## Remarks
- Removed duplicated twenty-server and twenty-shared typed
- Removed class validator instances for default value that were not used
at runtime, we will refactor that to add validation across all entities
following a same pattern
127 lines
3.7 KiB
TypeScript
127 lines
3.7 KiB
TypeScript
import {
|
|
Check,
|
|
Column,
|
|
CreateDateColumn,
|
|
DeleteDateColumn,
|
|
Entity,
|
|
Index,
|
|
JoinColumn,
|
|
ManyToOne,
|
|
OneToMany,
|
|
PrimaryGeneratedColumn,
|
|
Relation,
|
|
UpdateDateColumn,
|
|
} from 'typeorm';
|
|
|
|
import { type JsonbProperty } from 'src/engine/workspace-manager/workspace-migration/universal-flat-entity/types/jsonb-property.type';
|
|
import { CronTriggerEntity } from 'src/engine/metadata-modules/cron-trigger/entities/cron-trigger.entity';
|
|
import { DatabaseEventTriggerEntity } from 'src/engine/metadata-modules/database-event-trigger/entities/database-event-trigger.entity';
|
|
import { RouteTriggerEntity } from 'src/engine/metadata-modules/route-trigger/route-trigger.entity';
|
|
import { ServerlessFunctionLayerEntity } from 'src/engine/metadata-modules/serverless-function-layer/serverless-function-layer.entity';
|
|
import { SyncableEntity } from 'src/engine/workspace-manager/types/syncable-entity.interface';
|
|
|
|
const DEFAULT_SERVERLESS_TIMEOUT_SECONDS = 300; // 5 minutes
|
|
|
|
export enum ServerlessFunctionRuntime {
|
|
NODE18 = 'nodejs18.x',
|
|
NODE22 = 'nodejs22.x',
|
|
}
|
|
|
|
export const DEFAULT_SOURCE_HANDLER_PATH = 'src/index.ts';
|
|
export const DEFAULT_BUILT_HANDLER_PATH = 'index.mjs';
|
|
export const DEFAULT_HANDLER_NAME = 'main';
|
|
|
|
@Entity('serverlessFunction')
|
|
@Index('IDX_SERVERLESS_FUNCTION_ID_DELETED_AT', ['id', 'deletedAt'])
|
|
@Index('IDX_SERVERLESS_FUNCTION_LAYER_ID', ['serverlessFunctionLayerId'])
|
|
export class ServerlessFunctionEntity
|
|
extends SyncableEntity
|
|
implements Required<ServerlessFunctionEntity>
|
|
{
|
|
@PrimaryGeneratedColumn('uuid')
|
|
id: string;
|
|
|
|
@Column({ nullable: false })
|
|
name: string;
|
|
|
|
@Column({ nullable: false, default: DEFAULT_SOURCE_HANDLER_PATH })
|
|
sourceHandlerPath: string;
|
|
|
|
@Column({ nullable: false, default: DEFAULT_BUILT_HANDLER_PATH })
|
|
builtHandlerPath: string;
|
|
|
|
@Column({ nullable: false, default: DEFAULT_HANDLER_NAME })
|
|
handlerName: string;
|
|
|
|
@Column({ nullable: true, type: 'varchar' })
|
|
description: string | null;
|
|
|
|
@Column({ nullable: true, type: 'varchar' })
|
|
latestVersion: string | null;
|
|
|
|
@Column({ nullable: false, type: 'jsonb', default: [] })
|
|
publishedVersions: JsonbProperty<string[]>;
|
|
|
|
@Column({ nullable: false, default: ServerlessFunctionRuntime.NODE22 })
|
|
runtime: ServerlessFunctionRuntime;
|
|
|
|
@Column({ nullable: false, default: DEFAULT_SERVERLESS_TIMEOUT_SECONDS })
|
|
@Check(`"timeoutSeconds" >= 1 AND "timeoutSeconds" <= 900`)
|
|
timeoutSeconds: number;
|
|
|
|
@Column({ nullable: true, type: 'text' })
|
|
checksum: string | null;
|
|
|
|
@Column({ nullable: true, type: 'jsonb' })
|
|
toolInputSchema: JsonbProperty<object> | null;
|
|
|
|
@Column({ nullable: false, default: false })
|
|
isTool: boolean;
|
|
|
|
@Column({ nullable: false, type: 'uuid' })
|
|
serverlessFunctionLayerId: string;
|
|
|
|
@ManyToOne(
|
|
() => ServerlessFunctionLayerEntity,
|
|
(serverlessFunctionLayer) => serverlessFunctionLayer.serverlessFunctions,
|
|
{ nullable: false },
|
|
)
|
|
@JoinColumn({ name: 'serverlessFunctionLayerId' })
|
|
serverlessFunctionLayer: Relation<ServerlessFunctionLayerEntity>;
|
|
|
|
@OneToMany(
|
|
() => CronTriggerEntity,
|
|
(cronTrigger) => cronTrigger.serverlessFunction,
|
|
{
|
|
cascade: true,
|
|
},
|
|
)
|
|
cronTriggers: CronTriggerEntity[];
|
|
|
|
@OneToMany(
|
|
() => DatabaseEventTriggerEntity,
|
|
(databaseEventTrigger) => databaseEventTrigger.serverlessFunction,
|
|
{
|
|
cascade: true,
|
|
},
|
|
)
|
|
databaseEventTriggers: DatabaseEventTriggerEntity[];
|
|
|
|
@OneToMany(
|
|
() => RouteTriggerEntity,
|
|
(routeTrigger) => routeTrigger.serverlessFunction,
|
|
{
|
|
cascade: true,
|
|
},
|
|
)
|
|
routeTriggers: RouteTriggerEntity[];
|
|
|
|
@CreateDateColumn({ type: 'timestamptz' })
|
|
createdAt: Date;
|
|
|
|
@UpdateDateColumn({ type: 'timestamptz' })
|
|
updatedAt: Date;
|
|
|
|
@DeleteDateColumn({ type: 'timestamptz' })
|
|
deletedAt: Date | null;
|
|
}
|