@uql/core 0.4.81 → 0.4.84

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.
Files changed (49) hide show
  1. package/README.md +36 -51
  2. package/dialect/abstractSqlDialect-spec.d.ts +49 -0
  3. package/dialect/abstractSqlDialect-spec.js +765 -0
  4. package/dialect/mysqlDialect.spec.d.ts +6 -0
  5. package/dialect/mysqlDialect.spec.js +33 -0
  6. package/dialect/postgresDialect.spec.d.ts +1 -0
  7. package/dialect/postgresDialect.spec.js +139 -0
  8. package/dialect/sqliteDialect.spec.d.ts +1 -0
  9. package/dialect/sqliteDialect.spec.js +12 -0
  10. package/entity/decorator/definition.js +18 -18
  11. package/entity/decorator/definition.spec.d.ts +1 -0
  12. package/entity/decorator/definition.spec.js +725 -0
  13. package/entity/decorator/relation.spec.d.ts +1 -0
  14. package/entity/decorator/relation.spec.js +55 -0
  15. package/options.spec.d.ts +1 -0
  16. package/options.spec.js +50 -0
  17. package/package.json +3 -3
  18. package/querier/abstractQuerier-it.d.ts +41 -0
  19. package/querier/abstractQuerier-it.js +410 -0
  20. package/querier/abstractSqlQuerier-it.d.ts +9 -0
  21. package/querier/abstractSqlQuerier-it.js +19 -0
  22. package/querier/abstractSqlQuerier-spec.d.ts +53 -0
  23. package/querier/abstractSqlQuerier-spec.js +607 -0
  24. package/querier/decorator/injectQuerier.spec.d.ts +1 -0
  25. package/querier/decorator/injectQuerier.spec.js +108 -0
  26. package/querier/decorator/transactional.spec.d.ts +1 -0
  27. package/querier/decorator/transactional.spec.js +241 -0
  28. package/repository/genericRepository.spec.d.ts +1 -0
  29. package/repository/genericRepository.spec.js +86 -0
  30. package/test/entityMock.d.ts +180 -0
  31. package/test/entityMock.js +451 -0
  32. package/test/index.d.ts +3 -0
  33. package/test/index.js +7 -0
  34. package/test/it.util.d.ts +4 -0
  35. package/test/it.util.js +60 -0
  36. package/test/spec.util.d.ts +13 -0
  37. package/test/spec.util.js +54 -0
  38. package/type/entity.d.ts +15 -20
  39. package/type/entity.js +1 -1
  40. package/type/utility.d.ts +1 -1
  41. package/type/utility.js +1 -1
  42. package/util/dialect.util.spec.d.ts +1 -0
  43. package/util/dialect.util.spec.js +23 -0
  44. package/util/object.util.spec.d.ts +1 -0
  45. package/util/object.util.spec.js +26 -0
  46. package/util/sql.util.spec.d.ts +1 -0
  47. package/util/sql.util.spec.js +164 -0
  48. package/util/string.util.spec.d.ts +1 -0
  49. package/util/string.util.spec.js +26 -0
@@ -0,0 +1,6 @@
1
+ import { AbstractSqlDialectSpec } from './abstractSqlDialect-spec';
2
+ export declare class MySqlDialectSpec extends AbstractSqlDialectSpec {
3
+ constructor();
4
+ shouldBeginTransaction(): void;
5
+ shouldFind$text(): void;
6
+ }
@@ -0,0 +1,33 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.MySqlDialectSpec = void 0;
4
+ const test_1 = require("@uql/core/test");
5
+ const abstractSqlDialect_spec_1 = require("./abstractSqlDialect-spec");
6
+ const mysqlDialect_1 = require("./mysqlDialect");
7
+ class MySqlDialectSpec extends abstractSqlDialect_spec_1.AbstractSqlDialectSpec {
8
+ constructor() {
9
+ super(new mysqlDialect_1.MySqlDialect());
10
+ }
11
+ shouldBeginTransaction() {
12
+ expect(this.dialect.beginTransactionCommand).toBe('START TRANSACTION');
13
+ }
14
+ shouldFind$text() {
15
+ expect(this.dialect.find(test_1.Item, {
16
+ $project: ['id'],
17
+ $filter: { $text: { $fields: ['name', 'description'], $value: 'some text' }, creatorId: 1 },
18
+ $limit: 30,
19
+ })).toBe("SELECT `id` FROM `Item` WHERE MATCH(`name`, `description`) AGAINST('some text') AND `creatorId` = 1 LIMIT 30");
20
+ expect(this.dialect.find(test_1.User, {
21
+ $project: { id: true },
22
+ $filter: {
23
+ $text: { $fields: ['name'], $value: 'something' },
24
+ name: { $ne: 'other unwanted' },
25
+ creatorId: 1,
26
+ },
27
+ $limit: 10,
28
+ })).toBe("SELECT `id` FROM `User` WHERE MATCH(`name`) AGAINST('something') AND `name` <> 'other unwanted' AND `creatorId` = 1 LIMIT 10");
29
+ }
30
+ }
31
+ exports.MySqlDialectSpec = MySqlDialectSpec;
32
+ (0, test_1.createSpec)(new MySqlDialectSpec());
33
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibXlzcWxEaWFsZWN0LnNwZWMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZGlhbGVjdC9teXNxbERpYWxlY3Quc3BlYy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSx5Q0FBd0Q7QUFDeEQsdUVBQW1FO0FBQ25FLGlEQUE4QztBQUU5QyxNQUFhLGdCQUFpQixTQUFRLGdEQUFzQjtJQUMxRDtRQUNFLEtBQUssQ0FBQyxJQUFJLDJCQUFZLEVBQUUsQ0FBQyxDQUFDO0lBQzVCLENBQUM7SUFFUSxzQkFBc0I7UUFDN0IsTUFBTSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsdUJBQXVCLENBQUMsQ0FBQyxJQUFJLENBQUMsbUJBQW1CLENBQUMsQ0FBQztJQUN6RSxDQUFDO0lBRVEsZUFBZTtRQUN0QixNQUFNLENBQ0osSUFBSSxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsV0FBSSxFQUFFO1lBQ3RCLFFBQVEsRUFBRSxDQUFDLElBQUksQ0FBQztZQUNoQixPQUFPLEVBQUUsRUFBRSxLQUFLLEVBQUUsRUFBRSxPQUFPLEVBQUUsQ0FBQyxNQUFNLEVBQUUsYUFBYSxDQUFDLEVBQUUsTUFBTSxFQUFFLFdBQVcsRUFBRSxFQUFFLFNBQVMsRUFBRSxDQUFDLEVBQUU7WUFDM0YsTUFBTSxFQUFFLEVBQUU7U0FDWCxDQUFDLENBQ0gsQ0FBQyxJQUFJLENBQUMsOEdBQThHLENBQUMsQ0FBQztRQUV2SCxNQUFNLENBQ0osSUFBSSxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsV0FBSSxFQUFFO1lBQ3RCLFFBQVEsRUFBRSxFQUFFLEVBQUUsRUFBRSxJQUFJLEVBQUU7WUFDdEIsT0FBTyxFQUFFO2dCQUNQLEtBQUssRUFBRSxFQUFFLE9BQU8sRUFBRSxDQUFDLE1BQU0sQ0FBQyxFQUFFLE1BQU0sRUFBRSxXQUFXLEVBQUU7Z0JBQ2pELElBQUksRUFBRSxFQUFFLEdBQUcsRUFBRSxnQkFBZ0IsRUFBRTtnQkFDL0IsU0FBUyxFQUFFLENBQUM7YUFDYjtZQUNELE1BQU0sRUFBRSxFQUFFO1NBQ1gsQ0FBQyxDQUNILENBQUMsSUFBSSxDQUFDLDhIQUE4SCxDQUFDLENBQUM7SUFDekksQ0FBQztDQUNGO0FBOUJELDRDQThCQztBQUVELElBQUEsaUJBQVUsRUFBQyxJQUFJLGdCQUFnQixFQUFFLENBQUMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IGNyZWF0ZVNwZWMsIEl0ZW0sIFVzZXIgfSBmcm9tICdAdXFsL2NvcmUvdGVzdCc7XG5pbXBvcnQgeyBBYnN0cmFjdFNxbERpYWxlY3RTcGVjIH0gZnJvbSAnLi9hYnN0cmFjdFNxbERpYWxlY3Qtc3BlYyc7XG5pbXBvcnQgeyBNeVNxbERpYWxlY3QgfSBmcm9tICcuL215c3FsRGlhbGVjdCc7XG5cbmV4cG9ydCBjbGFzcyBNeVNxbERpYWxlY3RTcGVjIGV4dGVuZHMgQWJzdHJhY3RTcWxEaWFsZWN0U3BlYyB7XG4gIGNvbnN0cnVjdG9yKCkge1xuICAgIHN1cGVyKG5ldyBNeVNxbERpYWxlY3QoKSk7XG4gIH1cblxuICBvdmVycmlkZSBzaG91bGRCZWdpblRyYW5zYWN0aW9uKCkge1xuICAgIGV4cGVjdCh0aGlzLmRpYWxlY3QuYmVnaW5UcmFuc2FjdGlvbkNvbW1hbmQpLnRvQmUoJ1NUQVJUIFRSQU5TQUNUSU9OJyk7XG4gIH1cblxuICBvdmVycmlkZSBzaG91bGRGaW5kJHRleHQoKSB7XG4gICAgZXhwZWN0KFxuICAgICAgdGhpcy5kaWFsZWN0LmZpbmQoSXRlbSwge1xuICAgICAgICAkcHJvamVjdDogWydpZCddLFxuICAgICAgICAkZmlsdGVyOiB7ICR0ZXh0OiB7ICRmaWVsZHM6IFsnbmFtZScsICdkZXNjcmlwdGlvbiddLCAkdmFsdWU6ICdzb21lIHRleHQnIH0sIGNyZWF0b3JJZDogMSB9LFxuICAgICAgICAkbGltaXQ6IDMwLFxuICAgICAgfSlcbiAgICApLnRvQmUoXCJTRUxFQ1QgYGlkYCBGUk9NIGBJdGVtYCBXSEVSRSBNQVRDSChgbmFtZWAsIGBkZXNjcmlwdGlvbmApIEFHQUlOU1QoJ3NvbWUgdGV4dCcpIEFORCBgY3JlYXRvcklkYCA9IDEgTElNSVQgMzBcIik7XG5cbiAgICBleHBlY3QoXG4gICAgICB0aGlzLmRpYWxlY3QuZmluZChVc2VyLCB7XG4gICAgICAgICRwcm9qZWN0OiB7IGlkOiB0cnVlIH0sXG4gICAgICAgICRmaWx0ZXI6IHtcbiAgICAgICAgICAkdGV4dDogeyAkZmllbGRzOiBbJ25hbWUnXSwgJHZhbHVlOiAnc29tZXRoaW5nJyB9LFxuICAgICAgICAgIG5hbWU6IHsgJG5lOiAnb3RoZXIgdW53YW50ZWQnIH0sXG4gICAgICAgICAgY3JlYXRvcklkOiAxLFxuICAgICAgICB9LFxuICAgICAgICAkbGltaXQ6IDEwLFxuICAgICAgfSlcbiAgICApLnRvQmUoXCJTRUxFQ1QgYGlkYCBGUk9NIGBVc2VyYCBXSEVSRSBNQVRDSChgbmFtZWApIEFHQUlOU1QoJ3NvbWV0aGluZycpIEFORCBgbmFtZWAgPD4gJ290aGVyIHVud2FudGVkJyBBTkQgYGNyZWF0b3JJZGAgPSAxIExJTUlUIDEwXCIpO1xuICB9XG59XG5cbmNyZWF0ZVNwZWMobmV3IE15U3FsRGlhbGVjdFNwZWMoKSk7XG4iXX0=
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,139 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const test_1 = require("@uql/core/test");
4
+ const postgresDialect_1 = require("./postgresDialect");
5
+ class PostgresDialectSpec {
6
+ constructor() {
7
+ this.dialect = new postgresDialect_1.PostgresDialect();
8
+ }
9
+ shouldBeValidEscapeCharacter() {
10
+ expect(this.dialect.escapeIdChar).toBe('"');
11
+ }
12
+ shouldBeginTransaction() {
13
+ expect(this.dialect.beginTransactionCommand).toBe('BEGIN TRANSACTION');
14
+ }
15
+ shouldInsertMany() {
16
+ expect(this.dialect.insert(test_1.User, [
17
+ {
18
+ name: 'Some name 1',
19
+ email: 'someemail1@example.com',
20
+ createdAt: 123,
21
+ },
22
+ {
23
+ name: 'Some name 2',
24
+ email: 'someemail2@example.com',
25
+ createdAt: 456,
26
+ },
27
+ {
28
+ name: 'Some name 3',
29
+ email: 'someemail3@example.com',
30
+ createdAt: 789,
31
+ },
32
+ ])).toBe('INSERT INTO "User" ("name", "email", "createdAt") VALUES' +
33
+ " ('Some name 1', 'someemail1@example.com', 123)" +
34
+ ", ('Some name 2', 'someemail2@example.com', 456)" +
35
+ ", ('Some name 3', 'someemail3@example.com', 789)" +
36
+ ' RETURNING "id" "id"');
37
+ }
38
+ shouldInsertOne() {
39
+ expect(this.dialect.insert(test_1.User, {
40
+ name: 'Some Name',
41
+ email: 'someemail@example.com',
42
+ createdAt: 123,
43
+ })).toBe(`INSERT INTO "User" ("name", "email", "createdAt") VALUES ('Some Name', 'someemail@example.com', 123) RETURNING "id" "id"`);
44
+ }
45
+ shouldInsertWithOnInsertId() {
46
+ expect(this.dialect.insert(test_1.TaxCategory, {
47
+ name: 'Some Name',
48
+ createdAt: 123,
49
+ })).toMatch(/^INSERT INTO "TaxCategory" \("name", "createdAt", "pk"\) VALUES \('Some Name', 123, '.+'\) RETURNING "pk" "id"$/);
50
+ }
51
+ shouldFind$istartsWith() {
52
+ expect(this.dialect.find(test_1.User, {
53
+ $project: ['id'],
54
+ $filter: { name: { $istartsWith: 'Some' } },
55
+ $sort: { name: 1, id: -1 },
56
+ $skip: 0,
57
+ $limit: 50,
58
+ })).toBe(`SELECT "id" FROM "User" WHERE "name" ILIKE 'Some%' ORDER BY "name", "id" DESC LIMIT 50 OFFSET 0`);
59
+ expect(this.dialect.find(test_1.User, {
60
+ $project: { id: true },
61
+ $filter: { name: { $istartsWith: 'Some', $ne: 'Something' } },
62
+ $sort: { name: 1, id: -1 },
63
+ $skip: 0,
64
+ $limit: 50,
65
+ })).toBe(`SELECT "id" FROM "User" WHERE ("name" ILIKE 'Some%' AND "name" <> 'Something') ORDER BY "name", "id" DESC LIMIT 50 OFFSET 0`);
66
+ }
67
+ shouldFind$iendsWith() {
68
+ expect(this.dialect.find(test_1.User, {
69
+ $project: ['id'],
70
+ $filter: { name: { $iendsWith: 'Some' } },
71
+ $sort: { name: 1, id: -1 },
72
+ $skip: 0,
73
+ $limit: 50,
74
+ })).toBe(`SELECT "id" FROM "User" WHERE "name" ILIKE '%Some' ORDER BY "name", "id" DESC LIMIT 50 OFFSET 0`);
75
+ expect(this.dialect.find(test_1.User, {
76
+ $project: { id: true },
77
+ $filter: { name: { $iendsWith: 'Some', $ne: 'Something' } },
78
+ $sort: { name: 1, id: -1 },
79
+ $skip: 0,
80
+ $limit: 50,
81
+ })).toBe(`SELECT "id" FROM "User" WHERE ("name" ILIKE '%Some' AND "name" <> 'Something') ORDER BY "name", "id" DESC LIMIT 50 OFFSET 0`);
82
+ }
83
+ shouldFind$iincludes() {
84
+ expect(this.dialect.find(test_1.User, {
85
+ $project: ['id'],
86
+ $filter: { name: { $iincludes: 'Some' } },
87
+ $sort: { name: 1, id: -1 },
88
+ $skip: 0,
89
+ $limit: 50,
90
+ })).toBe(`SELECT "id" FROM "User" WHERE "name" ILIKE '%Some%' ORDER BY "name", "id" DESC LIMIT 50 OFFSET 0`);
91
+ expect(this.dialect.find(test_1.User, {
92
+ $project: { id: true },
93
+ $filter: { name: { $iincludes: 'Some', $ne: 'Something' } },
94
+ $sort: { name: 1, id: -1 },
95
+ $skip: 0,
96
+ $limit: 50,
97
+ })).toBe(`SELECT "id" FROM "User" WHERE ("name" ILIKE '%Some%' AND "name" <> 'Something') ORDER BY "name", "id" DESC LIMIT 50 OFFSET 0`);
98
+ }
99
+ shouldFind$ilike() {
100
+ expect(this.dialect.find(test_1.User, {
101
+ $project: ['id'],
102
+ $filter: { name: { $ilike: 'Some' } },
103
+ $sort: { name: 1, id: -1 },
104
+ $skip: 0,
105
+ $limit: 50,
106
+ })).toBe(`SELECT "id" FROM "User" WHERE "name" ILIKE 'Some' ORDER BY "name", "id" DESC LIMIT 50 OFFSET 0`);
107
+ expect(this.dialect.find(test_1.User, {
108
+ $project: { id: true },
109
+ $filter: { name: { $ilike: 'Some', $ne: 'Something' } },
110
+ $sort: { name: 1, id: -1 },
111
+ $skip: 0,
112
+ $limit: 50,
113
+ })).toBe(`SELECT "id" FROM "User" WHERE ("name" ILIKE 'Some' AND "name" <> 'Something') ORDER BY "name", "id" DESC LIMIT 50 OFFSET 0`);
114
+ }
115
+ shouldFind$regex() {
116
+ expect(this.dialect.find(test_1.User, {
117
+ $project: { id: true },
118
+ $filter: { name: { $regex: '^some' } },
119
+ })).toBe(`SELECT "id" FROM "User" WHERE "name" ~ '^some'`);
120
+ }
121
+ shouldFind$text() {
122
+ expect(this.dialect.find(test_1.Item, {
123
+ $project: { id: true },
124
+ $filter: { $text: { $fields: ['name', 'description'], $value: 'some text' }, code: '1' },
125
+ $limit: 30,
126
+ })).toBe(`SELECT "id" FROM "Item" WHERE to_tsvector("name" || ' ' || "description") @@ to_tsquery('some text') AND "code" = '1' LIMIT 30`);
127
+ expect(this.dialect.find(test_1.User, {
128
+ $project: { id: true },
129
+ $filter: {
130
+ $text: { $fields: ['name'], $value: 'something' },
131
+ name: { $ne: 'other unwanted' },
132
+ creatorId: 1,
133
+ },
134
+ $limit: 10,
135
+ })).toBe(`SELECT "id" FROM "User" WHERE to_tsvector("name") @@ to_tsquery('something') AND "name" <> 'other unwanted' AND "creatorId" = 1 LIMIT 10`);
136
+ }
137
+ }
138
+ (0, test_1.createSpec)(new PostgresDialectSpec());
139
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"postgresDialect.spec.js","sourceRoot":"","sources":["../../src/dialect/postgresDialect.spec.ts"],"names":[],"mappings":"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","sourcesContent":["import { User, Item, createSpec, TaxCategory } from '@uql/core/test';\nimport { PostgresDialect } from './postgresDialect';\n\nclass PostgresDialectSpec {\n  readonly dialect = new PostgresDialect();\n\n  shouldBeValidEscapeCharacter() {\n    expect(this.dialect.escapeIdChar).toBe('\"');\n  }\n\n  shouldBeginTransaction() {\n    expect(this.dialect.beginTransactionCommand).toBe('BEGIN TRANSACTION');\n  }\n\n  shouldInsertMany() {\n    expect(\n      this.dialect.insert(User, [\n        {\n          name: 'Some name 1',\n          email: 'someemail1@example.com',\n          createdAt: 123,\n        },\n        {\n          name: 'Some name 2',\n          email: 'someemail2@example.com',\n          createdAt: 456,\n        },\n        {\n          name: 'Some name 3',\n          email: 'someemail3@example.com',\n          createdAt: 789,\n        },\n      ])\n    ).toBe(\n      'INSERT INTO \"User\" (\"name\", \"email\", \"createdAt\") VALUES' +\n        \" ('Some name 1', 'someemail1@example.com', 123)\" +\n        \", ('Some name 2', 'someemail2@example.com', 456)\" +\n        \", ('Some name 3', 'someemail3@example.com', 789)\" +\n        ' RETURNING \"id\" \"id\"'\n    );\n  }\n\n  shouldInsertOne() {\n    expect(\n      this.dialect.insert(User, {\n        name: 'Some Name',\n        email: 'someemail@example.com',\n        createdAt: 123,\n      })\n    ).toBe(`INSERT INTO \"User\" (\"name\", \"email\", \"createdAt\") VALUES ('Some Name', 'someemail@example.com', 123) RETURNING \"id\" \"id\"`);\n  }\n\n  shouldInsertWithOnInsertId() {\n    expect(\n      this.dialect.insert(TaxCategory, {\n        name: 'Some Name',\n        createdAt: 123,\n      })\n    ).toMatch(/^INSERT INTO \"TaxCategory\" \\(\"name\", \"createdAt\", \"pk\"\\) VALUES \\('Some Name', 123, '.+'\\) RETURNING \"pk\" \"id\"$/);\n  }\n\n  shouldFind$istartsWith() {\n    expect(\n      this.dialect.find(User, {\n        $project: ['id'],\n        $filter: { name: { $istartsWith: 'Some' } },\n        $sort: { name: 1, id: -1 },\n        $skip: 0,\n        $limit: 50,\n      })\n    ).toBe(`SELECT \"id\" FROM \"User\" WHERE \"name\" ILIKE 'Some%' ORDER BY \"name\", \"id\" DESC LIMIT 50 OFFSET 0`);\n\n    expect(\n      this.dialect.find(User, {\n        $project: { id: true },\n        $filter: { name: { $istartsWith: 'Some', $ne: 'Something' } },\n        $sort: { name: 1, id: -1 },\n        $skip: 0,\n        $limit: 50,\n      })\n    ).toBe(`SELECT \"id\" FROM \"User\" WHERE (\"name\" ILIKE 'Some%' AND \"name\" <> 'Something') ORDER BY \"name\", \"id\" DESC LIMIT 50 OFFSET 0`);\n  }\n\n  shouldFind$iendsWith() {\n    expect(\n      this.dialect.find(User, {\n        $project: ['id'],\n        $filter: { name: { $iendsWith: 'Some' } },\n        $sort: { name: 1, id: -1 },\n        $skip: 0,\n        $limit: 50,\n      })\n    ).toBe(`SELECT \"id\" FROM \"User\" WHERE \"name\" ILIKE '%Some' ORDER BY \"name\", \"id\" DESC LIMIT 50 OFFSET 0`);\n\n    expect(\n      this.dialect.find(User, {\n        $project: { id: true },\n        $filter: { name: { $iendsWith: 'Some', $ne: 'Something' } },\n        $sort: { name: 1, id: -1 },\n        $skip: 0,\n        $limit: 50,\n      })\n    ).toBe(`SELECT \"id\" FROM \"User\" WHERE (\"name\" ILIKE '%Some' AND \"name\" <> 'Something') ORDER BY \"name\", \"id\" DESC LIMIT 50 OFFSET 0`);\n  }\n\n  shouldFind$iincludes() {\n    expect(\n      this.dialect.find(User, {\n        $project: ['id'],\n        $filter: { name: { $iincludes: 'Some' } },\n        $sort: { name: 1, id: -1 },\n        $skip: 0,\n        $limit: 50,\n      })\n    ).toBe(`SELECT \"id\" FROM \"User\" WHERE \"name\" ILIKE '%Some%' ORDER BY \"name\", \"id\" DESC LIMIT 50 OFFSET 0`);\n\n    expect(\n      this.dialect.find(User, {\n        $project: { id: true },\n        $filter: { name: { $iincludes: 'Some', $ne: 'Something' } },\n        $sort: { name: 1, id: -1 },\n        $skip: 0,\n        $limit: 50,\n      })\n    ).toBe(`SELECT \"id\" FROM \"User\" WHERE (\"name\" ILIKE '%Some%' AND \"name\" <> 'Something') ORDER BY \"name\", \"id\" DESC LIMIT 50 OFFSET 0`);\n  }\n\n  shouldFind$ilike() {\n    expect(\n      this.dialect.find(User, {\n        $project: ['id'],\n        $filter: { name: { $ilike: 'Some' } },\n        $sort: { name: 1, id: -1 },\n        $skip: 0,\n        $limit: 50,\n      })\n    ).toBe(`SELECT \"id\" FROM \"User\" WHERE \"name\" ILIKE 'Some' ORDER BY \"name\", \"id\" DESC LIMIT 50 OFFSET 0`);\n\n    expect(\n      this.dialect.find(User, {\n        $project: { id: true },\n        $filter: { name: { $ilike: 'Some', $ne: 'Something' } },\n        $sort: { name: 1, id: -1 },\n        $skip: 0,\n        $limit: 50,\n      })\n    ).toBe(`SELECT \"id\" FROM \"User\" WHERE (\"name\" ILIKE 'Some' AND \"name\" <> 'Something') ORDER BY \"name\", \"id\" DESC LIMIT 50 OFFSET 0`);\n  }\n\n  shouldFind$regex() {\n    expect(\n      this.dialect.find(User, {\n        $project: { id: true },\n        $filter: { name: { $regex: '^some' } },\n      })\n    ).toBe(`SELECT \"id\" FROM \"User\" WHERE \"name\" ~ '^some'`);\n  }\n\n  shouldFind$text() {\n    expect(\n      this.dialect.find(Item, {\n        $project: { id: true },\n        $filter: { $text: { $fields: ['name', 'description'], $value: 'some text' }, code: '1' },\n        $limit: 30,\n      })\n    ).toBe(`SELECT \"id\" FROM \"Item\" WHERE to_tsvector(\"name\" || ' ' || \"description\") @@ to_tsquery('some text') AND \"code\" = '1' LIMIT 30`);\n\n    expect(\n      this.dialect.find(User, {\n        $project: { id: true },\n        $filter: {\n          $text: { $fields: ['name'], $value: 'something' },\n          name: { $ne: 'other unwanted' },\n          creatorId: 1,\n        },\n        $limit: 10,\n      })\n    ).toBe(\n      `SELECT \"id\" FROM \"User\" WHERE to_tsvector(\"name\") @@ to_tsquery('something') AND \"name\" <> 'other unwanted' AND \"creatorId\" = 1 LIMIT 10`\n    );\n  }\n}\n\ncreateSpec(new PostgresDialectSpec());\n"]}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,12 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const test_1 = require("@uql/core/test");
4
+ const abstractSqlDialect_spec_1 = require("./abstractSqlDialect-spec");
5
+ const sqliteDialect_1 = require("./sqliteDialect");
6
+ class SqliteDialectSpec extends abstractSqlDialect_spec_1.AbstractSqlDialectSpec {
7
+ constructor() {
8
+ super(new sqliteDialect_1.SqliteDialect());
9
+ }
10
+ }
11
+ (0, test_1.createSpec)(new SqliteDialectSpec());
12
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3FsaXRlRGlhbGVjdC5zcGVjLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2RpYWxlY3Qvc3FsaXRlRGlhbGVjdC5zcGVjLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7O0FBQUEseUNBQXdEO0FBQ3hELHVFQUFtRTtBQUNuRSxtREFBZ0Q7QUFFaEQsTUFBTSxpQkFBa0IsU0FBUSxnREFBc0I7SUFDcEQ7UUFDRSxLQUFLLENBQUMsSUFBSSw2QkFBYSxFQUFFLENBQUMsQ0FBQztJQUM3QixDQUFDO0NBQ0Y7QUFFRCxJQUFBLGlCQUFVLEVBQUMsSUFBSSxpQkFBaUIsRUFBRSxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBjcmVhdGVTcGVjLCBJdGVtLCBVc2VyIH0gZnJvbSAnQHVxbC9jb3JlL3Rlc3QnO1xuaW1wb3J0IHsgQWJzdHJhY3RTcWxEaWFsZWN0U3BlYyB9IGZyb20gJy4vYWJzdHJhY3RTcWxEaWFsZWN0LXNwZWMnO1xuaW1wb3J0IHsgU3FsaXRlRGlhbGVjdCB9IGZyb20gJy4vc3FsaXRlRGlhbGVjdCc7XG5cbmNsYXNzIFNxbGl0ZURpYWxlY3RTcGVjIGV4dGVuZHMgQWJzdHJhY3RTcWxEaWFsZWN0U3BlYyB7XG4gIGNvbnN0cnVjdG9yKCkge1xuICAgIHN1cGVyKG5ldyBTcWxpdGVEaWFsZWN0KCkpO1xuICB9XG59XG5cbmNyZWF0ZVNwZWMobmV3IFNxbGl0ZURpYWxlY3RTcGVjKCkpO1xuIl19
@@ -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":";;;AAAA,4BAA0B;AAC1B,yCAA0E;AAc1E,MAAM,MAAM,GAAG,UAAU,CAAC;AAC1B,MAAM,OAAO,GAAG,4BAA4B,CAAC;AAC7C,MAAM,KAAK,GAAoC,MAAM,CAAC,OAAO,CAAC,IAAI,IAAI,GAAG,EAAE,CAAC;AAC5E,MAAM,CAAC,OAAO,CAAC,GAAG,KAAK,CAAC;AAExB,SAAgB,WAAW,CAAI,MAAe,EAAE,GAAW,EAAE,OAAqB,EAAE;IAClF,MAAM,IAAI,GAAG,UAAU,CAAC,MAAM,CAAC,CAAC;IAChC,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE;QACd,MAAM,IAAI,GAAG,SAAS,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;QACpC,IAAI,GAAG,EAAE,GAAG,IAAI,EAAE,IAAI,EAAE,CAAC;KAC1B;IACD,IAAI,OAAO,IAAI,CAAC,SAAS,KAAK,UAAU,EAAE;QACxC,IAAI,GAAG,EAAE,GAAG,IAAI,EAAE,SAAS,EAAE,EAAE,MAAM,EAAE,IAAI,CAAC,SAAS,EAAE,EAAE,CAAC;KAC3D;IACD,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,GAAG,EAAE,IAAI,EAAE,GAAG,EAAE,GAAG,IAAI,EAAE,EAAE,CAAC;IACtE,OAAO,IAAI,CAAC;AACd,CAAC;AAXD,kCAWC;AAED,SAAgB,QAAQ,CAAI,MAAe,EAAE,GAAW,EAAE,IAAkB;IAC1E,MAAM,IAAI,GAAG,UAAU,CAAC,MAAM,CAAC,CAAC;IAChC,MAAM,EAAE,GAAG,QAAQ,CAAC,IAAI,CAAC,CAAC;IAC1B,IAAI,EAAE,EAAE;QACN,MAAM,SAAS,CAAC,IAAI,MAAM,CAAC,IAAI,+CAA+C,CAAC,CAAC;KACjF;IACD,OAAO,WAAW,CAAC,MAAM,EAAE,GAAG,EAAE,EAAE,GAAG,IAAI,EAAE,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC;AAC3D,CAAC;AAPD,4BAOC;AAED,SAAgB,cAAc,CAAI,MAAe,EAAE,GAAW,EAAE,IAAwB;IACtF,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE;QAChB,MAAM,YAAY,GAAG,eAAe,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;QAClD,IAAI,CAAC,MAAM,GAAG,GAAG,EAAE,CAAC,YAAY,CAAC;KAClC;IACD,MAAM,IAAI,GAAG,UAAU,CAAC,MAAM,CAAC,CAAC;IAChC,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE,GAAG,IAAI,EAAE,CAAC;IAC1D,OAAO,IAAI,CAAC;AACd,CAAC;AARD,wCAQC;AAED,SAAgB,YAAY,CAAI,MAAe,EAAE,OAAsB,EAAE;IACvE,MAAM,IAAI,GAAG,UAAU,CAAC,MAAM,CAAC,CAAC;IAEhC,IAAI,CAAC,IAAA,cAAO,EAAC,IAAI,CAAC,MAAM,CAAC,EAAE;QACzB,MAAM,SAAS,CAAC,IAAI,MAAM,CAAC,IAAI,oBAAoB,CAAC,CAAC;KACtD;IAED,MAAM,YAAY,GAAG,IAAA,cAAO,EAAC,IAAI,CAAC,MAAM,CAAC,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,QAAQ,CAAkB,CAAC;IAEtG,IAAI,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE;QAC3B,MAAM,SAAS,CAAC,IAAI,MAAM,CAAC,IAAI,kDAAkD,CAAC,CAAC;KACpF;IAED,IAAI,IAAI,CAAC,UAAU,EAAE;QACnB,IAAI,CAAC,YAAY,CAAC,MAAM,EAAE;YACxB,MAAM,SAAS,CAAC,IAAI,MAAM,CAAC,IAAI,8DAA8D,CAAC,CAAC;SAChG;QACD,IAAI,CAAC,UAAU,GAAG,YAAY,CAAC,CAAC,CAAC,CAAC;KACnC;IAED,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,IAAI,IAAI,MAAM,CAAC,IAAI,CAAC;IACrC,IAAI,KAAK,GAAwB,MAAM,CAAC,cAAc,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;IAEzE,OAAO,KAAK,CAAC,WAAW,KAAK,MAAM,EAAE;QACnC,MAAM,UAAU,GAAG,UAAU,CAAC,KAAK,CAAC,WAAsB,CAAC,CAAC;QAC5D,UAAU,CAAC,IAAI,EAAE,UAAU,CAAC,CAAC;QAC7B,KAAK,GAAG,MAAM,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC;KACtC;IAED,MAAM,EAAE,GAAG,QAAQ,CAAC,IAAI,CAAC,CAAC;IAC1B,IAAI,CAAC,EAAE,EAAE;QACP,MAAM,SAAS,CAAC,IAAI,MAAM,CAAC,IAAI,0CAA0C,CAAC,CAAC;KAC5E;IACD,IAAI,CAAC,EAAE,GAAG,EAAE,CAAC;IAEb,OAAO,IAAI,CAAC;AACd,CAAC;AApCD,oCAoCC;AAED,SAAgB,WAAW;IACzB,OAAO,CAAC,GAAG,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,CAAC,GAAG,EAAE,GAAG,CAAC,EAAE,EAAE;QACrD,IAAI,GAAG,CAAC,EAAE,EAAE;YACV,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;SACf;QACD,OAAO,GAAG,CAAC;IACb,CAAC,EAAE,EAAE,CAAC,CAAC;AACT,CAAC;AAPD,kCAOC;AAED,SAAS,UAAU,CAAI,MAAe;IACpC,IAAI,IAAI,GAAG,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;IAC7B,IAAI,IAAI,EAAE;QACR,OAAO,IAAI,CAAC;KACb;IACD,IAAI,GAAG,EAAE,MAAM,EAAE,MAAM,EAAE,EAAE,EAAE,SAAS,EAAE,EAAE,EAAE,CAAC;IAC7C,KAAK,CAAC,GAAG,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;IACxB,OAAO,IAAI,CAAC;AACd,CAAC;AAED,SAAgB,OAAO,CAAI,MAAe;IACxC,MAAM,IAAI,GAAG,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;IAC/B,IAAI,CAAC,IAAI,EAAE;QACT,MAAM,SAAS,CAAC,IAAI,MAAM,CAAC,IAAI,oBAAoB,CAAC,CAAC;KACtD;IACD,IAAI,IAAI,CAAC,SAAS,EAAE;QAClB,OAAO,IAAI,CAAC;KACb;IACD,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;IACtB,OAAO,aAAa,CAAC,IAAI,CAAC,CAAC;AAC7B,CAAC;AAVD,0BAUC;AAED,SAAS,aAAa,CAAI,IAAmB;IAC3C,KAAK,MAAM,MAAM,IAAI,IAAI,CAAC,SAAS,EAAE;QACnC,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC,MAAwB,CAAC,CAAC;QAEzD,IAAI,OAAO,CAAC,UAAU,EAAE;YACtB,qCAAqC;YACrC,SAAS;SACV;QAED,IAAI,OAAO,CAAC,QAAQ,EAAE;YACpB,wBAAwB,CAAC,OAAO,CAAC,CAAC;YAClC,SAAS;SACV;QAED,MAAM,SAAS,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;QACnC,MAAM,OAAO,GAAG,UAAU,CAAC,SAAS,CAAC,CAAC;QAEtC,IAAI,OAAO,CAAC,WAAW,KAAK,IAAI,EAAE;YAChC,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC;YACzC,MAAM,SAAS,GAAG,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC;YAClD,MAAM,MAAM,GAAG,IAAA,iBAAU,EAAC,IAAI,CAAC,IAAI,CAAC,GAAG,IAAA,iBAAU,EAAC,MAAM,CAAC,CAAC;YAC1D,MAAM,MAAM,GAAG,IAAA,iBAAU,EAAC,OAAO,CAAC,IAAI,CAAC,GAAG,IAAA,iBAAU,EAAC,SAAS,CAAC,CAAC;YAChE,OAAO,CAAC,UAAU,GAAG;gBACnB,EAAE,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,IAAI,CAAC,EAAE,EAAE;gBACnC,EAAE,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,CAAC,EAAE,EAAE;aACvC,CAAC;SACH;aAAM;YACL,OAAO,CAAC,UAAU,GAAG,CAAC,EAAE,KAAK,EAAE,GAAG,MAAM,IAAI,EAAE,OAAO,EAAE,OAAO,CAAC,EAAE,EAAE,CAAC,CAAC;SACtE;QAED,IAAI,OAAO,CAAC,OAAO,EAAE;YACnB,oBAAoB,CAAC,OAAO,CAAC,OAAO,EAAE,CAAC,CAAC;SACzC;KACF;IAED,OAAO,IAAI,CAAC;AACd,CAAC;AAED,SAAS,wBAAwB,CAAI,OAA2B;IAC9D,MAAM,SAAS,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;IACnC,MAAM,OAAO,GAAG,OAAO,CAAC,SAAS,CAAC,CAAC;IACnC,OAAO,CAAC,QAAQ,GAAG,sBAAsB,CAAC,OAAO,CAAC,CAAC;IAEnD,+FAA+F;IAC/F,MAAM,EAAE,WAAW,EAAE,UAAU,EAAE,OAAO,EAAE,GAAG,OAAO,CAAC,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;IACjF,IAAI,WAAW,KAAK,IAAI,IAAI,WAAW,KAAK,IAAI,EAAE;QAChD,OAAO,CAAC,UAAU,GAAG,UAAU,CAAC,GAAG,CAAC,CAAC,EAAE,KAAK,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC,CAAC;YAC3D,KAAK,EAAE,OAAO;YACd,OAAO,EAAE,KAAK;SACf,CAAC,CAAC,CAAC;KACL;SAAM;QACL,OAAO,CAAC,UAAU,GAAG,UAAU,CAAC,KAAK,EAAE,CAAC,OAAO,EAAE,CAAC;QAClD,OAAO,CAAC,OAAO,GAAG,OAAO,CAAC;KAC3B;AACH,CAAC;AAED,SAAS,oBAAoB,CAAI,MAAe;IAC9C,MAAM,IAAI,GAAG,UAAU,CAAC,MAAM,CAAC,CAAC;IAChC,IAAI,CAAC,SAAS,GAAG,IAAA,cAAO,EAAC,IAAI,CAAC,MAAM,CAAC,CAAC,MAAM,CAAC,CAAC,SAAS,EAAE,GAAG,EAAE,EAAE;QAC9D,MAAM,EAAE,SAAS,EAAE,GAAG,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;QACvC,IAAI,SAAS,EAAE;YACb,MAAM,eAAe,GAAI,SAA8B,CAAC,MAAM,CAAC;YAC/D,MAAM,SAAS,GAAG,eAAe,EAAE,CAAC;YACpC,MAAM,OAAO,GAAG,UAAU,CAAC,SAAS,CAAC,CAAC;YACtC,MAAM,MAAM,GAAG,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC;YAChD,MAAM,OAAO,GAAoB;gBAC/B,WAAW,EAAE,IAAI;gBACjB,MAAM,EAAE,eAAe;gBACvB,UAAU,EAAE,CAAC,EAAE,KAAK,EAAE,GAAG,EAAE,OAAO,EAAE,OAAO,CAAC,EAAE,EAAE,CAAC;aAClD,CAAC;YACF,SAAS,CAAC,MAAM,CAAC,GAAG,OAAO,CAAC;SAC7B;QACD,OAAO,SAAS,CAAC;IACnB,CAAC,EAAE,EAAE,CAAC,CAAC;AACT,CAAC;AAED,SAAS,sBAAsB,CAAI,OAA2B;IAC5D,IAAI,OAAO,OAAO,CAAC,QAAQ,KAAK,UAAU,EAAE;QAC1C,MAAM,SAAS,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;QACnC,MAAM,OAAO,GAAG,UAAU,CAAC,SAAS,CAAC,CAAC;QACtC,MAAM,MAAM,GAAG,iBAAiB,CAAC,OAAO,CAAC,CAAC;QAC1C,OAAO,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;KACjC;IACD,OAAO,OAAO,CAAC,QAAQ,CAAC;AAC1B,CAAC;AAED,SAAS,iBAAiB,CAAI,IAAmB;IAC/C,OAAO,IAAA,cAAO,EAAC,IAAI,CAAC,SAAS,CAAC,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;QACjD,GAAG,CAAC,GAAG,CAAC,GAAG,GAAG,CAAC;QACf,OAAO,GAAG,CAAC;IACb,CAAC,EAAE,EAAuB,CAAC,CAAC;AAC9B,CAAC;AAED,SAAS,QAAQ,CAAI,IAAmB;IACtC,MAAM,EAAE,GAAG,IAAA,cAAO,EAAC,IAAI,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,IAAI,CAAC,CAAC;IACtE,OAAO,EAAc,CAAC;AACxB,CAAC;AAED,SAAS,UAAU,CAAI,MAAqB,EAAE,MAAqB;IACjE,MAAM,YAAY,GAAG,EAAE,GAAG,MAAM,CAAC,MAAM,EAAE,CAAC;IAC1C,MAAM,QAAQ,GAAG,QAAQ,CAAC,MAAM,CAAC,CAAC;IAClC,IAAI,QAAQ,EAAE;QACZ,MAAM,QAAQ,GAAG,QAAQ,CAAC,MAAM,CAAC,CAAC;QAClC,IAAI,QAAQ,EAAE;YACZ,OAAO,YAAY,CAAC,QAAQ,CAAC,CAAC;SAC/B;KACF;IACD,MAAM,CAAC,MAAM,GAAG,EAAE,GAAG,YAAY,EAAE,GAAG,MAAM,CAAC,MAAM,EAAE,CAAC;IACtD,MAAM,CAAC,SAAS,GAAG,EAAE,GAAG,MAAM,CAAC,SAAS,EAAE,GAAG,MAAM,CAAC,SAAS,EAAE,CAAC;AAClE,CAAC;AAED,SAAS,SAAS,CAAI,MAAe,EAAE,GAAW;IAChD,OAAO,OAAO,CAAC,WAAW,CAAC,aAAa,EAAE,MAAM,CAAC,SAAS,EAAE,GAAG,CAAC,CAAC;AACnE,CAAC;AAED,SAAS,eAAe,CAAI,MAAe,EAAE,GAAW;IACtD,MAAM,YAAY,GAAG,SAAS,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;IAC5C,MAAM,WAAW,GAAG,iBAAiB,CAAC,YAAY,CAAC,CAAC;IACpD,IAAI,CAAC,WAAW,EAAE;QAChB,MAAM,SAAS,CAAC,IAAI,MAAM,CAAC,IAAI,IAAI,GAAG,+CAA+C,YAAY,EAAE,IAAI,GAAG,CAAC,CAAC;KAC7G;IACD,OAAO,YAAY,CAAC;AACtB,CAAC;AAED,SAAgB,iBAAiB,CAAC,IAAS;IACzC,OAAO,CACL,OAAO,IAAI,KAAK,UAAU;QAC1B,IAAI,KAAK,OAAO;QAChB,IAAI,KAAK,MAAM;QACf,IAAI,KAAK,MAAM;QACf,IAAI,KAAK,MAAM;QACf,IAAI,KAAK,IAAI;QACb,IAAI,KAAK,MAAM;QACf,IAAI,KAAK,MAAM,CAChB,CAAC;AACJ,CAAC;AAXD,8CAWC","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 {\n  RelationOptions,\n  FieldOptions,\n  EntityOptions,\n  EntityMeta,\n  Type,\n  RelationKeyMap,\n  RelationKey,\n  RelationCardinality,\n  Key,\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  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"]}
@@ -0,0 +1 @@
1
+ export {};