@uql/core 0.4.82 → 0.4.85

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -1,14 +1,12 @@
1
1
  # [uql](https://uql.io) · [![license](https://img.shields.io/badge/license-MIT-blue.svg)](https://github.com/rogerpadilla/uql/blob/main/LICENSE) [![tests](https://github.com/rogerpadilla/uql/actions/workflows/tests.yml/badge.svg)](https://github.com/rogerpadilla/uql) [![coverage status](https://coveralls.io/repos/rogerpadilla/uql/badge.svg?branch=main)](https://coveralls.io/r/rogerpadilla/uql?branch=main) [![npm version](https://badge.fury.io/js/%40uql%2Fcore.svg)](https://badge.fury.io/js/%40uql%2Fcore)
2
2
 
3
- [Learn how to use uql in your own project](https://uql.io).
3
+ Learn more of uql in the website https://uql.io :high_brightness:
4
4
 
5
- # Getting Started
5
+ # Quick Start
6
6
 
7
- `uql` is a flexible and efficient `ORM`, with declarative `JSON` syntax and smart type-safety.
7
+ `uql` is a flexible and efficient `ORM`, with declarative `JSON` syntax and really smart type-safety.
8
8
 
9
- Given it is just a small library with serializable `JSON` syntax, the queries can be written in the client (web/mobile) and send to the backend, or just use `uql` directly in the backend, or even use it in a mobile app with an embedded database.
10
-
11
- ![autocomplete demo](https://uql.io/code.gif)
9
+ The `uql` queries can be safely written in the frontend (browser/mobile) and sent to the backend; or only use `uql` in the backend, or even in a mobile app with an embedded database (like `sqlite`).
12
10
 
13
11
  ## <a name="features"></a> Features
14
12
 
@@ -18,21 +16,8 @@ Given it is just a small library with serializable `JSON` syntax, the queries ca
18
16
  - `$project`, `$filter`, `$sort`, `$limit` works at multiple levels (including deep relations and their fields).
19
17
  - declarative and imperative `transactions`.
20
18
  - `soft-delete`, `virtual fields`, `repositories`, `connection pooling`.
21
- - different kinds of `relations` between entities.
22
19
  - transparent support for `inheritance` patterns between entities.
23
20
  - supports `Postgres`, `MySQL`, `MariaDB`, `SQLite`, `MongoDB` (beta).
24
- - plugins for frameworks: `express` (more coming).
25
-
26
- ## Table of Contents
27
-
28
- 1. [Installation](#installation)
29
- 2. [Configuration](#configuration)
30
- 3. [Entities](#entities)
31
- 4. [Declarative Transactions](#declarative-transactions)
32
- 5. [Imperative Transactions](#imperative-transactions)
33
- 6. [Generate REST APIs with Express](#express)
34
- 7. [Consume REST APIs from the Frontend](#client)
35
- 8. [FAQs](#faq)
36
21
 
37
22
  ## <a name="installation"></a> Installation
38
23
 
@@ -48,62 +33,64 @@ Given it is just a small library with serializable `JSON` syntax, the queries ca
48
33
  yarn add @uql/core
49
34
  ```
50
35
 
51
- 1. Install one of the following packages according to your database:
36
+ 2. Install one of the specific packages according to your database:
52
37
 
53
- | Database | Package |
54
- | ------------ | --------------- |
55
- | `MySQL` | `@uql/mysql` |
56
- | `MariaDB` | `@uql/maria` |
57
- | `PostgreSQL` | `@uql/postgres` |
58
- | `SQLite` | `@uql/sqlite` |
59
- | `MongoDB` | `@uql/mongo` |
38
+ | Database | Package |
39
+ | ------------ | --------------- |
40
+ | `MySQL` | `@uql/mysql` |
41
+ | `PostgreSQL` | `@uql/postgres` |
42
+ | `MariaDB` | `@uql/maria` |
43
+ | `MongoDB` | `@uql/mongo` |
44
+ | `SQLite` | `@uql/sqlite` |
60
45
 
61
- E.g. for `PostgreSQL`
46
+ E.g. for `PostgreSQL`
62
47
 
63
- ```sh
64
- npm install @uql/postgres --save
65
- ```
48
+ ```sh
49
+ npm install @uql/postgres --save
50
+ ```
66
51
 
67
- or with _yarn_
52
+ or with _yarn_
68
53
 
69
- ```sh
70
- yarn add @uql/postgres
71
- ```
54
+ ```sh
55
+ yarn add @uql/postgres
56
+ ```
72
57
 
73
- 1. Additionally, your `tsconfig.json` needs the following flags:
58
+ 3. Additionally, your `tsconfig.json` may need the following flags:
74
59
 
75
- ```json
76
- "target": "es6", // or a more recent ecmascript version.
77
- "experimentalDecorators": true,
78
- "emitDecoratorMetadata": true
79
- ```
60
+ ```json
61
+ "target": "es6", // or a more recent ecmascript version.
62
+ "experimentalDecorators": true,
63
+ "emitDecoratorMetadata": true
64
+ ```
80
65
 
81
66
  ## <a name="configuration"></a> Configuration
82
67
 
83
- Initialization should be done once (e.g. in one of the bootstrap files of your app).
68
+ A default querier-pool can be set in any of the bootstrap files of your app (e.g. in the `server.ts`).
84
69
 
85
70
  ```ts
86
- import { setOptions } from '@uql/core';
71
+ import { setDefaultQuerierPool } from '@uql/core';
87
72
  import { PgQuerierPool } from '@uql/postgres';
88
73
 
89
- setOptions({
90
- querierPool: new PgQuerierPool(
91
- {
92
- host: 'localhost',
93
- user: 'theUser',
94
- password: 'thePassword',
95
- database: 'theDatabase',
96
- },
97
- console.log
98
- ),
99
- });
74
+ const querierPool = new PgQuerierPool(
75
+ {
76
+ host: 'localhost',
77
+ user: 'theUser',
78
+ password: 'thePassword',
79
+ database: 'theDatabase',
80
+ },
81
+ // a logger can optionally be passed so the SQL queries are logged
82
+ console.log
83
+ );
84
+
85
+ setDefaultQuerierPool(querierPool);
100
86
  ```
101
87
 
102
- ## <a name="entities"></a> Entities
88
+ ## <a name="definition-of-entities"></a> Definition of Entities
103
89
 
104
90
  Take any dump class (aka DTO) and annotate it with the decorators from `'@uql/core/entity'`.
105
91
 
106
92
  ```ts
93
+ import { v4 as uuidv4 } from 'uuid';
107
94
  import { Field, ManyToOne, Id, OneToMany, Entity, OneToOne, ManyToMany } from '@uql/core/entity';
108
95
 
109
96
  @Entity()
@@ -111,21 +98,21 @@ export class Profile {
111
98
  /**
112
99
  * primary key
113
100
  */
114
- @Id()
115
- id?: number;
101
+ @Id({ onInsert: uuidv4 })
102
+ id?: string;
116
103
  @Field()
117
104
  picture?: string;
118
105
  /**
119
106
  * foreign-keys are really simple to specify.
120
107
  */
121
108
  @Field({ reference: () => User })
122
- creatorId?: number;
109
+ creatorId?: string;
123
110
  }
124
111
 
125
112
  @Entity()
126
113
  export class User {
127
- @Id()
128
- id?: number;
114
+ @Id({ onInsert: uuidv4 })
115
+ id?: string;
129
116
  @Field()
130
117
  name?: string;
131
118
  @Field()
@@ -141,8 +128,8 @@ export class User {
141
128
 
142
129
  @Entity()
143
130
  export class MeasureUnitCategory {
144
- @Id()
145
- id?: number;
131
+ @Id({ onInsert: uuidv4 })
132
+ id?: string;
146
133
  @Field()
147
134
  name?: string;
148
135
  @OneToMany({ entity: () => MeasureUnit, mappedBy: (measureUnit) => measureUnit.category })
@@ -151,232 +138,35 @@ export class MeasureUnitCategory {
151
138
 
152
139
  @Entity()
153
140
  export class MeasureUnit {
154
- @Id()
155
- id?: number;
141
+ @Id({ onInsert: uuidv4 })
142
+ id?: string;
156
143
  @Field()
157
144
  name?: string;
158
145
  @Field({ reference: () => MeasureUnitCategory })
159
- categoryId?: number;
146
+ categoryId?: string;
160
147
  @ManyToOne({ cascade: 'persist' })
161
148
  category?: MeasureUnitCategory;
162
149
  }
163
-
164
- @Entity()
165
- export class Item {
166
- @Id()
167
- id?: number;
168
- @Field()
169
- name?: string;
170
- @Field()
171
- description?: string;
172
- @Field()
173
- code?: string;
174
- @ManyToMany({ entity: () => Tag, through: () => ItemTag, cascade: true })
175
- tags?: Tag[];
176
- }
177
-
178
- @Entity()
179
- export class Tag {
180
- @Id()
181
- id?: number;
182
- @Field()
183
- name?: string;
184
- @ManyToMany({ entity: () => Item, mappedBy: (item) => item.tags })
185
- items?: Item[];
186
- }
187
-
188
- @Entity()
189
- export class ItemTag {
190
- @Id()
191
- id?: number;
192
- @Field({ reference: () => Item })
193
- itemId?: number;
194
- @Field({ reference: () => Tag })
195
- tagId?: number;
196
- }
197
150
  ```
198
151
 
199
- ## <a name="declarative-transactions"></a> Declarative Transactions
200
-
201
- Both, _declarative_ and _imperative_ transactions are supported, with the former you can just describe the scope of your transactions, with the later you have more flexibility (hence more responsibility).
202
-
203
- To use Declarative Transactions (using the `@Transactional` decorator):
204
-
205
- 1. take any service class, annotate the wanted function with the `@Transactional` decorator.
206
- 2. inject the querier instance by decorating one of the function's arguments with `@InjectQuerier`.
152
+ ## <a name="creation-of-queries"></a> Creation of Queries
207
153
 
208
154
  ```ts
209
- import { Querier } from '@uql/core/type';
210
- import { Transactional, InjectQuerier } from '@uql/core/querier';
211
-
212
- class ConfirmationService {
213
- @Transactional()
214
- async confirm(confirmation: Confirmation, @InjectQuerier() querier?: Querier): Promise<void> {
215
- if (confirmation.type === 'register') {
216
- await querier.insertOne(User, {
217
- name: confirmation.name,
218
- email: confirmation.email,
219
- password: confirmation.password,
220
- });
221
- } else {
222
- await querier.updateOneById(User, confirmation.creatorId, {
223
- password: confirmation.password,
224
- });
225
- }
226
- await querier.updateOneById(Confirmation, confirmation.id, { status: 1 });
227
- }
228
- }
155
+ import { getQuerier, getRepository } from '@uql/core';
156
+ import { User } from './entity';
229
157
 
230
- export const confirmationService = new ConfirmationService();
231
-
232
- /**
233
- * then you could just import the constant `confirmationService` in another file,
234
- * and when you call `confirmAction` function, all the operations there
235
- * will (automatically) run inside a single transaction.
236
- */
237
- await confirmationService.confirmAction(data);
238
- ```
158
+ const querier = await getQuerier();
239
159
 
240
- ## <a name="imperative-transactions"></a> Imperative Transactions
241
-
242
- `uql` supports both, _declarative_ and _imperative_ transactions, with the former you can just describe the scope of your transactions, with the later you have more flexibility (hence more responsibility).
243
-
244
- To use Imperative Transactions:
245
-
246
- 1. obtain the `querier` object with `await getQuerier()`.
247
- 2. run the transaction with `await querier.transaction(callback)`.
248
- 3. perform the different operations using the same `querier` (or `repositories`) inside your `callback` function.
160
+ const users = await querier.findMany(User, {
161
+ $project: { id: true, email: true, profile: ['id', 'picture'] },
162
+ $filter: { email: { $iendsWith: '@google.com' } },
163
+ $sort: { createdAt: -1 },
164
+ $limit: 100,
165
+ });
249
166
 
250
- ```ts
251
- import { getQuerier } from '@uql/core';
252
-
253
- async function confirm(confirmation: Confirmation): Promise<void> {
254
- const querier = await getQuerier();
255
- await querier.transaction(async () => {
256
- if (confirmation.action === 'signup') {
257
- await querier.insertOne(User, {
258
- name: confirmation.name,
259
- email: confirmation.email,
260
- password: confirmation.password,
261
- });
262
- } else {
263
- await querier.updateOneById(User, confirmation.creatorId, {
264
- password: confirmation.password,
265
- });
266
- }
267
- await querier.updateOneById(Confirmation, confirmation.id, { status: 1 });
268
- });
269
- }
167
+ await querier.release();
270
168
  ```
271
169
 
272
170
  ---
273
171
 
274
- That &#9650; can also be implemented as this &#9660; (for more granular control):
275
-
276
- ```ts
277
- async function confirm(confirmation: Confirmation): Promise<void> {
278
- const querier = await getQuerier();
279
- try {
280
- await querier.beginTransaction();
281
- if (confirmation.action === 'signup') {
282
- await querier.insertOne(User, {
283
- name: confirmation.name,
284
- email: confirmation.email,
285
- password: confirmation.password,
286
- });
287
- } else {
288
- await querier.updateOneById(User, confirmation.creatorId, {
289
- password: confirmation.password,
290
- });
291
- }
292
- await querier.updateOneById(Confirmation, confirmation.id, { status: 1 });
293
- await querier.commitTransaction();
294
- } catch (error) {
295
- await querier.rollbackTransaction();
296
- throw error;
297
- } finally {
298
- await querier.release();
299
- }
300
- }
301
- ```
302
-
303
- ## <a name="express"></a> Autogenerate REST APIs with Express
304
-
305
- A `express` plugin is provided to automatically generate REST APIs for your entities.
306
-
307
- 1. Install express plugin in your server project:
308
-
309
- ```sh
310
- npm install @uql/express --save
311
- ```
312
-
313
- or with _yarn_
314
-
315
- ```sh
316
- yarn add @uql/express
317
- ```
318
-
319
- 1. Initialize the `express` middleware in your server code to generate REST APIs for your entities
320
-
321
- ```ts
322
- import * as express from 'express';
323
- import { augmentFilter } from '@uql/core/util';
324
- import { Query, QueryFilter, EntityMeta } from '@uql/core/type';
325
- import { querierMiddleware } from '@uql/express';
326
-
327
- const app = express();
328
-
329
- app
330
- // ...
331
- .use(
332
- '/api',
333
-
334
- // this will generate REST APIs for the entities.
335
- querierMiddleware({
336
- // all entities will be automatically exposed unless
337
- // 'include' or 'exclude' options are provided.
338
- exclude: [Confirmation],
339
-
340
- // `augmentQuery` callback allows to extend all then queries that are requested to the API,
341
- // so it is a good place to add additional filters to the queries,
342
- // e.g. for multi tenant apps.
343
- augmentQuery: <E>(meta: EntityMeta<E>, qm: Query<E>, req: express.Request): Query<E> => {
344
- // ensure the user can only see the data that belongs to his company.
345
- qm.$filter = augmentFilter(meta, qm.$filter, { companyId: req.identity.companyId } as QueryFilter<E>);
346
- return qm;
347
- },
348
- })
349
- );
350
- ```
351
-
352
- ## <a name="client"></a> Easily call the generated REST APIs from the Client
353
-
354
- A client plugin (for browser/mobile) is provided to easily consume the REST APIs from the frontend.
355
-
356
- 1. Install client plugin in your frontend project:
357
-
358
- ```sh
359
- npm install @uql/client --save
360
- ```
361
-
362
- or with _yarn_
363
-
364
- ```sh
365
- yarn add @uql/client
366
- ```
367
-
368
- 1. Use the client to call the `uql` CRUD API
369
-
370
- ```ts
371
- import { getRepository } from '@uql/client';
372
-
373
- // 'User' is an entity class.
374
- const userRepository = getRepository(User);
375
-
376
- const users = await userRepository.findMany({
377
- $project: { email: true, profile: ['picture'] },
378
- $filter: { email: { $endsWith: '@domain.com' } },
379
- $sort: { createdAt: -1 },
380
- $limit: 100,
381
- });
382
- ```
172
+ See more in https://uql.io :high_brightness:
@@ -13,9 +13,6 @@ function defineField(entity, key, opts = {}) {
13
13
  const type = inferType(entity, key);
14
14
  opts = { ...opts, type };
15
15
  }
16
- if (typeof opts.reference === 'function') {
17
- opts = { ...opts, reference: { entity: opts.reference } };
18
- }
19
16
  meta.fields[key] = { ...meta.fields[key], ...{ name: key, ...opts } };
20
17
  return meta;
21
18
  }
@@ -135,31 +132,32 @@ function fillInverseSideRelations(relOpts) {
135
132
  const relEntity = relOpts.entity();
136
133
  const relMeta = getMeta(relEntity);
137
134
  relOpts.mappedBy = getMappedByRelationKey(relOpts);
138
- // reversing references here makes the SQL generation simpler (no need to check for `mappedBy`)
139
- const { cardinality, references, through } = relMeta.relations[relOpts.mappedBy];
140
- if (cardinality === '11' || cardinality === 'm1') {
141
- relOpts.references = references.map(({ local, foreign }) => ({
142
- local: foreign,
143
- foreign: local,
144
- }));
135
+ if (relMeta.fields[relOpts.mappedBy]) {
136
+ relOpts.references = [{ local: relMeta.id, foreign: relOpts.mappedBy }];
137
+ return;
145
138
  }
146
- else {
147
- relOpts.references = references.slice().reverse();
148
- relOpts.through = through;
139
+ const mappedByRelation = relMeta.relations[relOpts.mappedBy];
140
+ if (relOpts.cardinality === 'm1' || relOpts.cardinality === 'mm') {
141
+ relOpts.references = mappedByRelation.references.slice().reverse();
142
+ relOpts.through = mappedByRelation.through;
143
+ return;
149
144
  }
145
+ relOpts.references = mappedByRelation.references.map(({ local, foreign }) => ({
146
+ local: foreign,
147
+ foreign: local,
148
+ }));
150
149
  }
151
150
  function fillThroughRelations(entity) {
152
151
  const meta = ensureMeta(entity);
153
152
  meta.relations = (0, util_1.getKeys)(meta.fields).reduce((relations, key) => {
154
153
  const { reference } = meta.fields[key];
155
154
  if (reference) {
156
- const relEntityGetter = reference.entity;
157
- const relEntity = relEntityGetter();
155
+ const relEntity = reference();
158
156
  const relMeta = ensureMeta(relEntity);
159
157
  const relKey = key.slice(0, -relMeta.id.length);
160
158
  const relOpts = {
159
+ entity: reference,
161
160
  cardinality: 'm1',
162
- entity: relEntityGetter,
163
161
  references: [{ local: key, foreign: relMeta.id }],
164
162
  };
165
163
  relations[relKey] = relOpts;
@@ -177,7 +175,9 @@ function getMappedByRelationKey(relOpts) {
177
175
  return relOpts.mappedBy;
178
176
  }
179
177
  function getRelationKeyMap(meta) {
180
- return (0, util_1.getKeys)(meta.relations).reduce((acc, key) => {
178
+ return (0, util_1.getKeys)(meta.fields)
179
+ .concat((0, util_1.getKeys)(meta.relations))
180
+ .reduce((acc, key) => {
181
181
  acc[key] = key;
182
182
  return acc;
183
183
  }, {});
@@ -220,4 +220,4 @@ function isValidEntityType(type) {
220
220
  type !== Object);
221
221
  }
222
222
  exports.isValidEntityType = isValidEntityType;
223
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"definition.js","sourceRoot":"","sources":["../../../src/entity/decorator/definition.ts"],"names":[],"mappings":"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","sourcesContent":["import 'reflect-metadata';\nimport { hasKeys, lowerFirst, getKeys, upperFirst } from '@uql/core/util';\nimport {\n  RelationOptions,\n  FieldOptions,\n  EntityOptions,\n  EntityMeta,\n  Type,\n  RelationKeyMap,\n  ReferenceOptions,\n  RelationKey,\n  FieldKey,\n  IdKey,\n} from '@uql/core/type';\n\nconst holder = globalThis;\nconst metaKey = '@uql/core/entity/decorator';\nconst metas: Map<Type<any>, EntityMeta<any>> = holder[metaKey] ?? new Map();\nholder[metaKey] = metas;\n\nexport function defineField<E>(entity: Type<E>, key: string, opts: FieldOptions = {}): EntityMeta<E> {\n  const meta = ensureMeta(entity);\n  if (!opts.type) {\n    const type = inferType(entity, key);\n    opts = { ...opts, type };\n  }\n  if (typeof opts.reference === 'function') {\n    opts = { ...opts, reference: { entity: opts.reference } };\n  }\n  meta.fields[key] = { ...meta.fields[key], ...{ name: key, ...opts } };\n  return meta;\n}\n\nexport function defineId<E>(entity: Type<E>, key: string, opts: FieldOptions): EntityMeta<E> {\n  const meta = ensureMeta(entity);\n  const id = getIdKey(meta);\n  if (id) {\n    throw TypeError(`'${entity.name}' must have a single field decorated with @Id`);\n  }\n  return defineField(entity, key, { ...opts, isId: true });\n}\n\nexport function defineRelation<E>(entity: Type<E>, key: string, opts: RelationOptions<E>): EntityMeta<E> {\n  if (!opts.entity) {\n    const inferredType = inferEntityType(entity, key);\n    opts.entity = () => inferredType;\n  }\n  const meta = ensureMeta(entity);\n  meta.relations[key] = { ...meta.relations[key], ...opts };\n  return meta;\n}\n\nexport function defineEntity<E>(entity: Type<E>, opts: EntityOptions = {}): EntityMeta<E> {\n  const meta = ensureMeta(entity);\n\n  if (!hasKeys(meta.fields)) {\n    throw TypeError(`'${entity.name}' must have fields`);\n  }\n\n  const onDeleteKeys = getKeys(meta.fields).filter((key) => meta.fields[key].onDelete) as FieldKey<E>[];\n\n  if (onDeleteKeys.length > 1) {\n    throw TypeError(`'${entity.name}' must have one field with 'onDelete' as maximum`);\n  }\n\n  if (opts.softDelete) {\n    if (!onDeleteKeys.length) {\n      throw TypeError(`'${entity.name}' must have one field with 'onDelete' to enable 'softDelete'`);\n    }\n    meta.softDelete = onDeleteKeys[0];\n  }\n\n  meta.name = opts.name ?? entity.name;\n  let proto: FunctionConstructor = Object.getPrototypeOf(entity.prototype);\n\n  while (proto.constructor !== Object) {\n    const parentMeta = ensureMeta(proto.constructor as Type<E>);\n    extendMeta(meta, parentMeta);\n    proto = Object.getPrototypeOf(proto);\n  }\n\n  const id = getIdKey(meta);\n  if (!id) {\n    throw TypeError(`'${entity.name}' must have one field decorated with @Id`);\n  }\n  meta.id = id;\n\n  return meta;\n}\n\nexport function getEntities(): Type<any>[] {\n  return [...metas.entries()].reduce((acc, [key, val]) => {\n    if (val.id) {\n      acc.push(key);\n    }\n    return acc;\n  }, []);\n}\n\nfunction ensureMeta<E>(entity: Type<E>): EntityMeta<E> {\n  let meta = metas.get(entity);\n  if (meta) {\n    return meta;\n  }\n  meta = { entity, fields: {}, relations: {} };\n  metas.set(entity, meta);\n  return meta;\n}\n\nexport function getMeta<E>(entity: Type<E>): EntityMeta<E> {\n  const meta = metas.get(entity);\n  if (!meta) {\n    throw TypeError(`'${entity.name}' is not an entity`);\n  }\n  if (meta.processed) {\n    return meta;\n  }\n  meta.processed = true;\n  return fillRelations(meta);\n}\n\nfunction fillRelations<E>(meta: EntityMeta<E>): EntityMeta<E> {\n  for (const relKey in meta.relations) {\n    const relOpts = meta.relations[relKey as RelationKey<E>];\n\n    if (relOpts.references) {\n      // references were manually specified\n      continue;\n    }\n\n    if (relOpts.mappedBy) {\n      fillInverseSideRelations(relOpts);\n      continue;\n    }\n\n    const relEntity = relOpts.entity();\n    const relMeta = ensureMeta(relEntity);\n\n    if (relOpts.cardinality === 'mm') {\n      const idName = meta.fields[meta.id].name;\n      const relIdName = relMeta.fields[relMeta.id].name;\n      const source = lowerFirst(meta.name) + upperFirst(idName);\n      const target = lowerFirst(relMeta.name) + upperFirst(relIdName);\n      relOpts.references = [\n        { local: source, foreign: meta.id },\n        { local: target, foreign: relMeta.id },\n      ];\n    } else {\n      relOpts.references = [{ local: `${relKey}Id`, foreign: relMeta.id }];\n    }\n\n    if (relOpts.through) {\n      fillThroughRelations(relOpts.through());\n    }\n  }\n\n  return meta;\n}\n\nfunction fillInverseSideRelations<E>(relOpts: RelationOptions<E>): void {\n  const relEntity = relOpts.entity();\n  const relMeta = getMeta(relEntity);\n  relOpts.mappedBy = getMappedByRelationKey(relOpts);\n\n  // reversing references here makes the SQL generation simpler (no need to check for `mappedBy`)\n  const { cardinality, references, through } = relMeta.relations[relOpts.mappedBy];\n  if (cardinality === '11' || cardinality === 'm1') {\n    relOpts.references = references.map(({ local, foreign }) => ({\n      local: foreign,\n      foreign: local,\n    }));\n  } else {\n    relOpts.references = references.slice().reverse();\n    relOpts.through = through;\n  }\n}\n\nfunction fillThroughRelations<E>(entity: Type<E>): void {\n  const meta = ensureMeta(entity);\n  meta.relations = getKeys(meta.fields).reduce((relations, key) => {\n    const { reference } = meta.fields[key];\n    if (reference) {\n      const relEntityGetter = (reference as ReferenceOptions).entity;\n      const relEntity = relEntityGetter();\n      const relMeta = ensureMeta(relEntity);\n      const relKey = key.slice(0, -relMeta.id.length);\n      const relOpts: RelationOptions = {\n        cardinality: 'm1',\n        entity: relEntityGetter,\n        references: [{ local: key, foreign: relMeta.id }],\n      };\n      relations[relKey] = relOpts;\n    }\n    return relations;\n  }, {});\n}\n\nfunction getMappedByRelationKey<E>(relOpts: RelationOptions<E>): RelationKey<E> {\n  if (typeof relOpts.mappedBy === 'function') {\n    const relEntity = relOpts.entity();\n    const relMeta = ensureMeta(relEntity);\n    const keyMap = getRelationKeyMap(relMeta);\n    return relOpts.mappedBy(keyMap);\n  }\n  return relOpts.mappedBy;\n}\n\nfunction getRelationKeyMap<E>(meta: EntityMeta<E>): RelationKeyMap<E> {\n  return getKeys(meta.relations).reduce((acc, key) => {\n    acc[key] = key;\n    return acc;\n  }, {} as RelationKeyMap<E>);\n}\n\nfunction getIdKey<E>(meta: EntityMeta<E>): IdKey<E> {\n  const id = getKeys(meta.fields).find((key) => meta.fields[key]?.isId);\n  return id as IdKey<E>;\n}\n\nfunction extendMeta<E>(target: EntityMeta<E>, source: EntityMeta<E>): void {\n  const sourceFields = { ...source.fields };\n  const targetId = getIdKey(target);\n  if (targetId) {\n    const sourceId = getIdKey(source);\n    if (sourceId) {\n      delete sourceFields[sourceId];\n    }\n  }\n  target.fields = { ...sourceFields, ...target.fields };\n  target.relations = { ...source.relations, ...target.relations };\n}\n\nfunction inferType<E>(entity: Type<E>, key: string): any {\n  return Reflect.getMetadata('design:type', entity.prototype, key);\n}\n\nfunction inferEntityType<E>(entity: Type<E>, key: string): Type<any> {\n  const inferredType = inferType(entity, key);\n  const isValidType = isValidEntityType(inferredType);\n  if (!isValidType) {\n    throw TypeError(`'${entity.name}.${key}' type was auto-inferred with invalid type '${inferredType?.name}'`);\n  }\n  return inferredType;\n}\n\nexport function isValidEntityType(type: any): type is Type<any> {\n  return (\n    typeof type === 'function' &&\n    type !== Boolean &&\n    type !== String &&\n    type !== Number &&\n    type !== BigInt &&\n    type !== Date &&\n    type !== Symbol &&\n    type !== Object\n  );\n}\n"]}
223
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"definition.js","sourceRoot":"","sources":["../../../src/entity/decorator/definition.ts"],"names":[],"mappings":"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","sourcesContent":["import 'reflect-metadata';\nimport { hasKeys, lowerFirst, getKeys, upperFirst } from '@uql/core/util';\nimport { RelationOptions, FieldOptions, EntityOptions, EntityMeta, Type, RelationKeyMap, RelationKey, Key, FieldKey, IdKey } from '@uql/core/type';\n\nconst holder = globalThis;\nconst metaKey = '@uql/core/entity/decorator';\nconst metas: Map<Type<any>, EntityMeta<any>> = holder[metaKey] ?? new Map();\nholder[metaKey] = metas;\n\nexport function defineField<E>(entity: Type<E>, key: string, opts: FieldOptions = {}): EntityMeta<E> {\n  const meta = ensureMeta(entity);\n  if (!opts.type) {\n    const type = inferType(entity, key);\n    opts = { ...opts, type };\n  }\n  meta.fields[key] = { ...meta.fields[key], ...{ name: key, ...opts } };\n  return meta;\n}\n\nexport function defineId<E>(entity: Type<E>, key: string, opts: FieldOptions): EntityMeta<E> {\n  const meta = ensureMeta(entity);\n  const id = getIdKey(meta);\n  if (id) {\n    throw TypeError(`'${entity.name}' must have a single field decorated with @Id`);\n  }\n  return defineField(entity, key, { ...opts, isId: true });\n}\n\nexport function defineRelation<E>(entity: Type<E>, key: string, opts: RelationOptions<E>): EntityMeta<E> {\n  if (!opts.entity) {\n    const inferredType = inferEntityType(entity, key);\n    opts.entity = () => inferredType;\n  }\n  const meta = ensureMeta(entity);\n  meta.relations[key] = { ...meta.relations[key], ...opts };\n  return meta;\n}\n\nexport function defineEntity<E>(entity: Type<E>, opts: EntityOptions = {}): EntityMeta<E> {\n  const meta = ensureMeta(entity);\n\n  if (!hasKeys(meta.fields)) {\n    throw TypeError(`'${entity.name}' must have fields`);\n  }\n\n  const onDeleteKeys = getKeys(meta.fields).filter((key) => meta.fields[key].onDelete) as FieldKey<E>[];\n\n  if (onDeleteKeys.length > 1) {\n    throw TypeError(`'${entity.name}' must have one field with 'onDelete' as maximum`);\n  }\n\n  if (opts.softDelete) {\n    if (!onDeleteKeys.length) {\n      throw TypeError(`'${entity.name}' must have one field with 'onDelete' to enable 'softDelete'`);\n    }\n    meta.softDelete = onDeleteKeys[0];\n  }\n\n  meta.name = opts.name ?? entity.name;\n  let proto: FunctionConstructor = Object.getPrototypeOf(entity.prototype);\n\n  while (proto.constructor !== Object) {\n    const parentMeta = ensureMeta(proto.constructor as Type<E>);\n    extendMeta(meta, parentMeta);\n    proto = Object.getPrototypeOf(proto);\n  }\n\n  const id = getIdKey(meta);\n  if (!id) {\n    throw TypeError(`'${entity.name}' must have one field decorated with @Id`);\n  }\n  meta.id = id;\n\n  return meta;\n}\n\nexport function getEntities(): Type<any>[] {\n  return [...metas.entries()].reduce((acc, [key, val]) => {\n    if (val.id) {\n      acc.push(key);\n    }\n    return acc;\n  }, []);\n}\n\nfunction ensureMeta<E>(entity: Type<E>): EntityMeta<E> {\n  let meta = metas.get(entity);\n  if (meta) {\n    return meta;\n  }\n  meta = { entity, fields: {}, relations: {} };\n  metas.set(entity, meta);\n  return meta;\n}\n\nexport function getMeta<E>(entity: Type<E>): EntityMeta<E> {\n  const meta = metas.get(entity);\n  if (!meta) {\n    throw TypeError(`'${entity.name}' is not an entity`);\n  }\n  if (meta.processed) {\n    return meta;\n  }\n  meta.processed = true;\n  return fillRelations(meta);\n}\n\nfunction fillRelations<E>(meta: EntityMeta<E>): EntityMeta<E> {\n  for (const relKey in meta.relations) {\n    const relOpts = meta.relations[relKey as RelationKey<E>];\n\n    if (relOpts.references) {\n      // references were manually specified\n      continue;\n    }\n\n    if (relOpts.mappedBy) {\n      fillInverseSideRelations(relOpts);\n      continue;\n    }\n\n    const relEntity = relOpts.entity();\n    const relMeta = ensureMeta(relEntity);\n\n    if (relOpts.cardinality === 'mm') {\n      const idName = meta.fields[meta.id].name;\n      const relIdName = relMeta.fields[relMeta.id].name;\n      const source = lowerFirst(meta.name) + upperFirst(idName);\n      const target = lowerFirst(relMeta.name) + upperFirst(relIdName);\n      relOpts.references = [\n        { local: source, foreign: meta.id },\n        { local: target, foreign: relMeta.id },\n      ];\n    } else {\n      relOpts.references = [{ local: `${relKey}Id`, foreign: relMeta.id }];\n    }\n\n    if (relOpts.through) {\n      fillThroughRelations(relOpts.through());\n    }\n  }\n\n  return meta;\n}\n\nfunction fillInverseSideRelations<E>(relOpts: RelationOptions<E>): void {\n  const relEntity = relOpts.entity();\n  const relMeta = getMeta(relEntity);\n  relOpts.mappedBy = getMappedByRelationKey(relOpts);\n\n  if (relMeta.fields[relOpts.mappedBy as FieldKey<E>]) {\n    relOpts.references = [{ local: relMeta.id, foreign: relOpts.mappedBy }];\n    return;\n  }\n\n  const mappedByRelation = relMeta.relations[relOpts.mappedBy as RelationKey<E>];\n\n  if (relOpts.cardinality === 'm1' || relOpts.cardinality === 'mm') {\n    relOpts.references = mappedByRelation.references.slice().reverse();\n    relOpts.through = mappedByRelation.through;\n    return;\n  }\n\n  relOpts.references = mappedByRelation.references.map(({ local, foreign }) => ({\n    local: foreign,\n    foreign: local,\n  }));\n}\n\nfunction fillThroughRelations<E>(entity: Type<E>): void {\n  const meta = ensureMeta(entity);\n  meta.relations = getKeys(meta.fields).reduce((relations, key) => {\n    const { reference } = meta.fields[key as FieldKey<E>];\n    if (reference) {\n      const relEntity = reference();\n      const relMeta = ensureMeta(relEntity);\n      const relKey = key.slice(0, -relMeta.id.length);\n      const relOpts: RelationOptions = {\n        entity: reference,\n        cardinality: 'm1',\n        references: [{ local: key, foreign: relMeta.id }],\n      };\n      relations[relKey] = relOpts;\n    }\n    return relations;\n  }, {});\n}\n\nfunction getMappedByRelationKey<E>(relOpts: RelationOptions<E>): Key<E> {\n  if (typeof relOpts.mappedBy === 'function') {\n    const relEntity = relOpts.entity();\n    const relMeta = ensureMeta(relEntity);\n    const keyMap = getRelationKeyMap(relMeta);\n    return relOpts.mappedBy(keyMap);\n  }\n  return relOpts.mappedBy;\n}\n\nfunction getRelationKeyMap<E>(meta: EntityMeta<E>): RelationKeyMap<E> {\n  return getKeys(meta.fields)\n    .concat(getKeys(meta.relations))\n    .reduce((acc, key) => {\n      acc[key] = key;\n      return acc;\n    }, {} as RelationKeyMap<E>);\n}\n\nfunction getIdKey<E>(meta: EntityMeta<E>): IdKey<E> {\n  const id = getKeys(meta.fields).find((key) => meta.fields[key]?.isId);\n  return id as IdKey<E>;\n}\n\nfunction extendMeta<E>(target: EntityMeta<E>, source: EntityMeta<E>): void {\n  const sourceFields = { ...source.fields };\n  const targetId = getIdKey(target);\n  if (targetId) {\n    const sourceId = getIdKey(source);\n    if (sourceId) {\n      delete sourceFields[sourceId];\n    }\n  }\n  target.fields = { ...sourceFields, ...target.fields };\n  target.relations = { ...source.relations, ...target.relations };\n}\n\nfunction inferType<E>(entity: Type<E>, key: string): any {\n  return Reflect.getMetadata('design:type', entity.prototype, key);\n}\n\nfunction inferEntityType<E>(entity: Type<E>, key: string): Type<any> {\n  const inferredType = inferType(entity, key);\n  const isValidType = isValidEntityType(inferredType);\n  if (!isValidType) {\n    throw TypeError(`'${entity.name}.${key}' type was auto-inferred with invalid type '${inferredType?.name}'`);\n  }\n  return inferredType;\n}\n\nexport function isValidEntityType(type: any): type is Type<any> {\n  return (\n    typeof type === 'function' &&\n    type !== Boolean &&\n    type !== String &&\n    type !== Number &&\n    type !== BigInt &&\n    type !== Date &&\n    type !== Symbol &&\n    type !== Object\n  );\n}\n"]}