@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 +64 -274
- package/entity/decorator/definition.js +18 -18
- package/entity/decorator/definition.spec.js +35 -47
- package/options.d.ts +3 -4
- package/options.js +12 -18
- package/options.spec.js +8 -31
- package/package.json +4 -4
- package/querier/abstractSqlQuerier-spec.js +12 -14
- package/querier/decorator/transactional.js +2 -2
- package/querier/decorator/transactional.spec.js +4 -6
- package/test/entityMock.js +3 -3
- package/type/entity.d.ts +15 -20
- package/type/entity.js +1 -1
- package/type/index.d.ts +0 -1
- package/type/index.js +1 -2
- package/type/utility.d.ts +1 -1
- package/type/utility.js +1 -1
- package/type/options.d.ts +0 -7
- package/type/options.js +0 -3
package/README.md
CHANGED
|
@@ -1,14 +1,12 @@
|
|
|
1
1
|
# [uql](https://uql.io) · [](https://github.com/rogerpadilla/uql/blob/main/LICENSE) [](https://github.com/rogerpadilla/uql) [](https://coveralls.io/r/rogerpadilla/uql?branch=main) [](https://badge.fury.io/js/%40uql%2Fcore)
|
|
2
2
|
|
|
3
|
-
|
|
3
|
+
Learn more of uql in the website https://uql.io :high_brightness:
|
|
4
4
|
|
|
5
|
-
#
|
|
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
|
-
|
|
10
|
-
|
|
11
|
-

|
|
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
|
-
|
|
36
|
+
2. Install one of the specific packages according to your database:
|
|
52
37
|
|
|
53
|
-
| Database | Package |
|
|
54
|
-
| ------------ | --------------- |
|
|
55
|
-
| `MySQL` | `@uql/mysql` |
|
|
56
|
-
| `
|
|
57
|
-
| `
|
|
58
|
-
| `
|
|
59
|
-
| `
|
|
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
|
-
|
|
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
|
-
|
|
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 {
|
|
71
|
+
import { setDefaultQuerierPool } from '@uql/core';
|
|
87
72
|
import { PgQuerierPool } from '@uql/postgres';
|
|
88
73
|
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
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?:
|
|
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?:
|
|
109
|
+
creatorId?: string;
|
|
123
110
|
}
|
|
124
111
|
|
|
125
112
|
@Entity()
|
|
126
113
|
export class User {
|
|
127
|
-
@Id()
|
|
128
|
-
id?:
|
|
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?:
|
|
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?:
|
|
141
|
+
@Id({ onInsert: uuidv4 })
|
|
142
|
+
id?: string;
|
|
156
143
|
@Field()
|
|
157
144
|
name?: string;
|
|
158
145
|
@Field({ reference: () => MeasureUnitCategory })
|
|
159
|
-
categoryId?:
|
|
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="
|
|
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 {
|
|
210
|
-
import {
|
|
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
|
-
|
|
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
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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
|
-
|
|
139
|
-
|
|
140
|
-
|
|
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
|
-
|
|
147
|
-
|
|
148
|
-
relOpts.
|
|
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
|
|
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.
|
|
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"]}
|