@uql/core 0.4.68 → 0.4.74

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 (53) hide show
  1. package/README.md +1 -1
  2. package/dialect/{baseSqlDialect.d.ts → abstractSqlDialect.d.ts} +1 -1
  3. package/dialect/abstractSqlDialect.js +367 -0
  4. package/dialect/index.d.ts +1 -1
  5. package/dialect/index.js +5 -5
  6. package/dialect/mysqlDialect.d.ts +2 -4
  7. package/dialect/mysqlDialect.js +3 -13
  8. package/dialect/postgresDialect.d.ts +2 -2
  9. package/dialect/postgresDialect.js +6 -6
  10. package/dialect/sqliteDialect.d.ts +4 -2
  11. package/dialect/sqliteDialect.js +12 -3
  12. package/entity/decorator/definition.js +21 -21
  13. package/entity/decorator/entity.js +2 -2
  14. package/entity/decorator/field.js +2 -2
  15. package/entity/decorator/id.js +2 -2
  16. package/entity/decorator/index.js +5 -5
  17. package/entity/decorator/relation.js +2 -2
  18. package/entity/index.js +2 -2
  19. package/index.js +2 -2
  20. package/options.js +3 -3
  21. package/package.json +6 -6
  22. package/querier/{baseQuerier.d.ts → abstractQuerier.d.ts} +4 -3
  23. package/querier/abstractQuerier.js +230 -0
  24. package/querier/abstractSqlQuerier.d.ts +27 -0
  25. package/querier/abstractSqlQuerier.js +88 -0
  26. package/querier/decorator/index.js +3 -3
  27. package/querier/decorator/transactional.js +4 -4
  28. package/querier/index.d.ts +2 -2
  29. package/querier/index.js +4 -4
  30. package/repository/{baseRepository.d.ts → genericRepository.d.ts} +1 -1
  31. package/repository/genericRepository.js +50 -0
  32. package/repository/index.d.ts +1 -1
  33. package/repository/index.js +2 -2
  34. package/type/entity.d.ts +14 -13
  35. package/type/entity.js +1 -7
  36. package/type/index.d.ts +1 -0
  37. package/type/index.js +9 -8
  38. package/type/options.d.ts +0 -4
  39. package/type/options.js +1 -1
  40. package/type/querier.d.ts +8 -91
  41. package/type/querier.js +1 -1
  42. package/type/querierPool.d.ts +2 -27
  43. package/type/querierPool.js +1 -1
  44. package/type/universalQuerier.d.ts +96 -0
  45. package/type/universalQuerier.js +3 -0
  46. package/util/dialect.util.js +5 -5
  47. package/util/index.js +6 -6
  48. package/util/sql.util.js +4 -4
  49. package/dialect/baseSqlDialect.js +0 -365
  50. package/querier/baseQuerier.js +0 -230
  51. package/querier/sqlQuerier.d.ts +0 -19
  52. package/querier/sqlQuerier.js +0 -91
  53. package/repository/baseRepository.js +0 -50
@@ -1,365 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.BaseSqlDialect = void 0;
4
- const sqlstring_1 = require("sqlstring");
5
- const entity_1 = require("@uql/core/entity");
6
- const util_1 = require("@uql/core/util");
7
- class BaseSqlDialect {
8
- constructor(beginTransactionCommand, escapeIdChar) {
9
- this.beginTransactionCommand = beginTransactionCommand;
10
- this.escapeIdChar = escapeIdChar;
11
- this.escapeIdRegex = RegExp(escapeIdChar, 'g');
12
- }
13
- criteria(entity, qm, opts = {}) {
14
- var _a;
15
- const meta = entity_1.getMeta(entity);
16
- const prefix = ((_a = opts.prefix) !== null && _a !== void 0 ? _a : (opts.autoPrefix || util_1.isProjectingRelations(meta, qm.$project))) ? meta.name : undefined;
17
- const where = this.where(entity, qm.$filter, { ...opts, prefix });
18
- const group = this.group(entity, qm.$group, { ...opts, prefix });
19
- const having = this.where(entity, qm.$having, { ...opts, prefix, clause: 'HAVING' });
20
- const sort = this.sort(entity, qm.$sort, { ...opts, prefix });
21
- const pager = this.pager(qm);
22
- return where + group + having + sort + pager;
23
- }
24
- projectFields(entity, project, opts = {}) {
25
- const meta = entity_1.getMeta(entity);
26
- const prefix = opts.prefix ? opts.prefix + '.' : '';
27
- const escapedPrefix = this.escapeId(opts.prefix, true, true);
28
- let fields;
29
- if (project) {
30
- if (Array.isArray(project)) {
31
- fields = project;
32
- }
33
- else {
34
- const positiveProjectKeys = util_1.getKeys(project).filter((key) => project[key]);
35
- fields = positiveProjectKeys.length
36
- ? positiveProjectKeys.map((key) => {
37
- const val = project[key];
38
- if (val instanceof util_1.Raw) {
39
- return util_1.raw(val.value, key);
40
- }
41
- return key;
42
- })
43
- : util_1.getKeys(meta.fields).filter((key) => !(key in project));
44
- }
45
- fields = fields.filter((key) => key instanceof util_1.Raw || meta.fields[key]);
46
- if (!fields.length || (opts.prefix && !fields.includes(meta.id))) {
47
- fields = [meta.id, ...fields];
48
- }
49
- }
50
- else {
51
- fields = util_1.getKeys(meta.fields);
52
- }
53
- return fields
54
- .map((key) => {
55
- var _a;
56
- if (key instanceof util_1.Raw) {
57
- return util_1.getRawValue({
58
- value: key,
59
- dialect: this,
60
- prefix,
61
- escapedPrefix,
62
- autoPrefixAlias: opts.autoPrefixAlias,
63
- });
64
- }
65
- const field = meta.fields[key];
66
- if (field.virtual) {
67
- return util_1.getRawValue({
68
- value: field.virtual,
69
- alias: key,
70
- dialect: this,
71
- prefix,
72
- escapedPrefix,
73
- autoPrefixAlias: opts.autoPrefixAlias,
74
- });
75
- }
76
- const name = (_a = field === null || field === void 0 ? void 0 : field.name) !== null && _a !== void 0 ? _a : key;
77
- const fieldPath = `${escapedPrefix}${this.escapeId(name)}`;
78
- return !opts.autoPrefixAlias && (name === key || !field) ? fieldPath : `${fieldPath} ${this.escapeId((prefix + key), true)}`;
79
- })
80
- .join(', ');
81
- }
82
- projectRelations(entity, project = {}, { prefix } = {}) {
83
- const meta = entity_1.getMeta(entity);
84
- const relations = util_1.getProjectRelationKeys(meta, project);
85
- const isProjectArray = Array.isArray(project);
86
- let fields = '';
87
- let tables = '';
88
- for (const key of relations) {
89
- const relOpts = meta.relations[key];
90
- if (relOpts.cardinality === '1m' || relOpts.cardinality === 'mm') {
91
- // '1m' and 'mm' should be resolved in a higher layer because they will need multiple queries
92
- continue;
93
- }
94
- const joinRelAlias = prefix ? prefix + '.' + key : key;
95
- const relEntity = relOpts.entity();
96
- const relProject = project[key];
97
- const relQuery = isProjectArray ? {} : Array.isArray(relProject) ? { $project: relProject } : relProject;
98
- const relColumns = this.projectFields(relEntity, relQuery.$project, {
99
- prefix: joinRelAlias,
100
- autoPrefixAlias: true,
101
- });
102
- fields += ', ' + relColumns;
103
- const { fields: subColumns, tables: subTables } = this.projectRelations(relEntity, relQuery.$project, {
104
- prefix: joinRelAlias,
105
- });
106
- fields += subColumns;
107
- const relMeta = entity_1.getMeta(relEntity);
108
- const relEntityName = this.escapeId(relMeta.name);
109
- const relPath = prefix ? this.escapeId(prefix, true) : this.escapeId(meta.name);
110
- const joinType = relQuery.$required ? 'INNER' : 'LEFT';
111
- const joinAlias = this.escapeId(joinRelAlias, true);
112
- tables += ` ${joinType} JOIN ${relEntityName} ${joinAlias} ON `;
113
- tables += relOpts.references.map((it) => `${joinAlias}.${this.escapeId(it.foreign)} = ${relPath}.${this.escapeId(it.local)}`).join(' AND ');
114
- if (relQuery.$filter) {
115
- const filter = this.where(relEntity, relQuery.$filter, { prefix: key, clause: false });
116
- tables += ` AND ${filter}`;
117
- }
118
- tables += subTables;
119
- }
120
- return { fields, tables };
121
- }
122
- select(entity, qm, opts = {}) {
123
- var _a;
124
- const meta = entity_1.getMeta(entity);
125
- const prefix = ((_a = opts.prefix) !== null && _a !== void 0 ? _a : (opts.autoPrefix || util_1.isProjectingRelations(meta, qm.$project))) ? meta.name : undefined;
126
- const fields = this.projectFields(entity, qm.$project, { prefix });
127
- const { fields: relationFields, tables } = this.projectRelations(entity, qm.$project);
128
- return `SELECT ${fields}${relationFields} FROM ${this.escapeId(meta.name)}${tables}`;
129
- }
130
- where(entity, filter = {}, opts = {}) {
131
- const meta = entity_1.getMeta(entity);
132
- const { usePrecedence, clause = 'WHERE', softDelete } = opts;
133
- filter = util_1.getQueryFilterAsMap(meta, filter);
134
- if (meta.softDelete && (softDelete || softDelete === undefined) && clause !== 'HAVING' && !filter[meta.softDelete]) {
135
- filter[meta.softDelete] = null;
136
- }
137
- const entries = Object.entries(filter);
138
- if (!entries.length) {
139
- return '';
140
- }
141
- const options = { ...opts, usePrecedence: entries.length > 1 };
142
- let sql = entries.map(([key, val]) => this.compare(entity, key, val, options)).join(` AND `);
143
- if (usePrecedence) {
144
- sql = `(${sql})`;
145
- }
146
- return clause ? ` ${clause} ${sql}` : sql;
147
- }
148
- compare(entity, key, val, opts) {
149
- var _a;
150
- const meta = entity_1.getMeta(entity);
151
- if (val instanceof util_1.Raw) {
152
- if (key === '$exists' || key === '$nexists') {
153
- const value = val;
154
- const query = util_1.getRawValue({
155
- value,
156
- dialect: this,
157
- prefix: meta.name,
158
- escapedPrefix: this.escapeId(meta.name, false, true),
159
- });
160
- return `${key === '$exists' ? 'EXISTS' : 'NOT EXISTS'} (${query})`;
161
- }
162
- const comparisonKey = this.getComparisonKey(entity, key, opts);
163
- return `${comparisonKey} = ${val.value}`;
164
- }
165
- if (key === '$text') {
166
- const search = val;
167
- return `${this.escapeId(meta.name)} MATCH ${this.escape(search.$value)}`;
168
- }
169
- if (key === '$and' || key === '$or' || key === '$not' || key === '$nor') {
170
- const negateOperatorMap = {
171
- $not: '$and',
172
- $nor: '$or',
173
- };
174
- const op = (_a = negateOperatorMap[key]) !== null && _a !== void 0 ? _a : key;
175
- const negate = key in negateOperatorMap ? 'NOT ' : '';
176
- const values = val;
177
- const hasManyItems = values.length > 1;
178
- const logicalComparison = values
179
- .map((filterEntry) => {
180
- if (filterEntry instanceof util_1.Raw) {
181
- return util_1.getRawValue({
182
- value: filterEntry,
183
- dialect: this,
184
- prefix: opts.prefix,
185
- escapedPrefix: this.escapeId(opts.prefix, true, true),
186
- });
187
- }
188
- return this.where(entity, filterEntry, {
189
- prefix: opts.prefix,
190
- usePrecedence: hasManyItems && !Array.isArray(filterEntry) && util_1.getKeys(filterEntry).length > 1,
191
- clause: false,
192
- });
193
- })
194
- .join(op === '$or' ? ' OR ' : ' AND ');
195
- return (opts.usePrecedence || negate) && hasManyItems ? `${negate}(${logicalComparison})` : `${negate}${logicalComparison}`;
196
- }
197
- const value = Array.isArray(val) ? { $in: val } : typeof val === 'object' && val !== null ? val : { $eq: val };
198
- const operators = util_1.getKeys(value);
199
- const comparisons = operators.map((op) => this.compareFieldOperator(entity, key, op, value[op], opts)).join(' AND ');
200
- return operators.length > 1 ? `(${comparisons})` : comparisons;
201
- }
202
- compareFieldOperator(entity, key, op, val, opts) {
203
- const comparisonKey = this.getComparisonKey(entity, key, opts);
204
- switch (op) {
205
- case '$eq':
206
- return val === null ? `${comparisonKey} IS NULL` : `${comparisonKey} = ${this.escape(val)}`;
207
- case '$ne':
208
- return val === null ? `${comparisonKey} IS NOT NULL` : `${comparisonKey} <> ${this.escape(val)}`;
209
- case '$not':
210
- return this.compare(entity, '$not', [{ [key]: val }], opts);
211
- case '$gt':
212
- return `${comparisonKey} > ${this.escape(val)}`;
213
- case '$gte':
214
- return `${comparisonKey} >= ${this.escape(val)}`;
215
- case '$lt':
216
- return `${comparisonKey} < ${this.escape(val)}`;
217
- case '$lte':
218
- return `${comparisonKey} <= ${this.escape(val)}`;
219
- case '$startsWith':
220
- return `${comparisonKey} LIKE ${this.escape(`${val}%`)}`;
221
- case '$istartsWith':
222
- return `LOWER(${comparisonKey}) LIKE ${this.escape(val.toLowerCase() + '%')}`;
223
- case '$endsWith':
224
- return `${comparisonKey} LIKE ${this.escape(`%${val}`)}`;
225
- case '$iendsWith':
226
- return `LOWER(${comparisonKey}) LIKE ${this.escape('%' + val.toLowerCase())}`;
227
- case '$includes':
228
- return `${comparisonKey} LIKE ${this.escape(`%${val}%`)}`;
229
- case '$iincludes':
230
- return `LOWER(${comparisonKey}) LIKE ${this.escape('%' + val.toLowerCase() + '%')}`;
231
- case '$ilike':
232
- return `LOWER(${comparisonKey}) LIKE ${this.escape(val.toLowerCase())}`;
233
- case '$like':
234
- return `${comparisonKey} LIKE ${this.escape(val)}`;
235
- case '$in':
236
- return `${comparisonKey} IN (${this.escape(val)})`;
237
- case '$nin':
238
- return `${comparisonKey} NOT IN (${this.escape(val)})`;
239
- case '$regex':
240
- return `${comparisonKey} REGEXP ${this.escape(val)}`;
241
- default:
242
- throw new TypeError(`unknown operator: ${op}`);
243
- }
244
- }
245
- getComparisonKey(entity, key, { prefix } = {}) {
246
- var _a;
247
- const meta = entity_1.getMeta(entity);
248
- const escapedPrefix = this.escapeId(prefix, true, true);
249
- const field = meta.fields[key];
250
- if (field === null || field === void 0 ? void 0 : field.virtual) {
251
- return util_1.getRawValue({
252
- value: field.virtual,
253
- dialect: this,
254
- prefix,
255
- escapedPrefix,
256
- });
257
- }
258
- return escapedPrefix + this.escapeId((_a = field === null || field === void 0 ? void 0 : field.name) !== null && _a !== void 0 ? _a : key);
259
- }
260
- group(entity, fields, opts = {}) {
261
- if (!(fields === null || fields === void 0 ? void 0 : fields.length)) {
262
- return '';
263
- }
264
- const meta = entity_1.getMeta(entity);
265
- const names = fields.map((key) => { var _a, _b; return this.escapeId((_b = (_a = meta.fields[key]) === null || _a === void 0 ? void 0 : _a.name) !== null && _b !== void 0 ? _b : key); }).join(', ');
266
- return ` GROUP BY ${names}`;
267
- }
268
- sort(entity, sort, { prefix } = {}) {
269
- const sortMap = util_1.buildSortMap(sort);
270
- if (!util_1.hasKeys(sortMap)) {
271
- return '';
272
- }
273
- const meta = entity_1.getMeta(entity);
274
- const flattenedSort = util_1.flatObject(sortMap, prefix);
275
- const directionMap = { 1: '', asc: '', '-1': ' DESC', desc: ' DESC' };
276
- const order = Object.entries(flattenedSort)
277
- .map(([key, sort]) => {
278
- var _a, _b;
279
- const name = (_b = (_a = meta.fields[key]) === null || _a === void 0 ? void 0 : _a.name) !== null && _b !== void 0 ? _b : key;
280
- const direction = directionMap[sort];
281
- return this.escapeId(name) + direction;
282
- })
283
- .join(', ');
284
- return ` ORDER BY ${order}`;
285
- }
286
- pager(opts) {
287
- let sql = '';
288
- if (opts.$limit) {
289
- sql += ` LIMIT ${Number(opts.$limit)}`;
290
- }
291
- if (opts.$skip !== undefined) {
292
- sql += ` OFFSET ${Number(opts.$skip)}`;
293
- }
294
- return sql;
295
- }
296
- count(entity, qm, opts) {
297
- const search = {
298
- ...qm,
299
- $project: [util_1.raw('COUNT(*)', 'count')],
300
- };
301
- delete search.$sort;
302
- delete search.$skip;
303
- delete search.$limit;
304
- const select = this.select(entity, search);
305
- const criteria = this.criteria(entity, search, opts);
306
- return select + criteria;
307
- }
308
- find(entity, qm, opts) {
309
- const select = this.select(entity, qm, opts);
310
- const criteria = this.criteria(entity, qm, opts);
311
- return select + criteria;
312
- }
313
- insert(entity, payload) {
314
- const meta = entity_1.getMeta(entity);
315
- payload = util_1.getPersistables(meta, payload, 'onInsert');
316
- const keys = util_1.getKeys(payload[0]);
317
- const columns = keys.map((key) => this.escapeId(meta.fields[key].name));
318
- const values = payload.map((it) => keys.map((key) => this.escape(it[key])).join(', ')).join('), (');
319
- return `INSERT INTO ${this.escapeId(meta.name)} (${columns.join(', ')}) VALUES (${values})`;
320
- }
321
- update(entity, qm, payload, opts) {
322
- const meta = entity_1.getMeta(entity);
323
- payload = util_1.getPersistable(meta, payload, 'onUpdate');
324
- const values = util_1.getKeys(payload)
325
- .map((key) => `${this.escapeId(key)} = ${this.escape(payload[key])}`)
326
- .join(', ');
327
- const criteria = this.criteria(entity, qm, opts);
328
- return `UPDATE ${this.escapeId(meta.name)} SET ${values}${criteria}`;
329
- }
330
- delete(entity, qm, opts = {}) {
331
- const meta = entity_1.getMeta(entity);
332
- if (opts.softDelete || opts.softDelete === undefined) {
333
- if (meta.softDelete) {
334
- const criteria = this.criteria(entity, qm, opts);
335
- const value = meta.fields[meta.softDelete].onDelete();
336
- return `UPDATE ${this.escapeId(meta.name)} SET ${this.escapeId(meta.softDelete)} = ${this.escape(value)}${criteria}`;
337
- }
338
- else if (opts.softDelete) {
339
- throw new TypeError(`'${meta.name}' has not enabled 'softDelete'`);
340
- }
341
- }
342
- const criteria = this.criteria(entity, qm, opts);
343
- return `DELETE FROM ${this.escapeId(meta.name)}${criteria}`;
344
- }
345
- escapeId(val, forbidQualified, addDot) {
346
- if (!val) {
347
- return '';
348
- }
349
- if (!forbidQualified && val.includes('.')) {
350
- return val
351
- .split('.')
352
- .map((it) => this.escapeId(it))
353
- .join('.');
354
- }
355
- // sourced from 'escapeId' function here https://github.com/mysqljs/sqlstring/blob/master/lib/SqlString.js
356
- const escaped = this.escapeIdChar + val.replace(this.escapeIdRegex, this.escapeIdChar + this.escapeIdChar) + this.escapeIdChar;
357
- const suffix = addDot ? '.' : '';
358
- return escaped + suffix;
359
- }
360
- escape(value) {
361
- return sqlstring_1.escape(value);
362
- }
363
- }
364
- exports.BaseSqlDialect = BaseSqlDialect;
365
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"baseSqlDialect.js","sourceRoot":"","sources":["../../src/dialect/baseSqlDialect.ts"],"names":[],"mappings":"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","sourcesContent":["import { escape } from 'sqlstring';\nimport { getMeta } from '@uql/core/entity';\nimport {\n  getPersistable,\n  getProjectRelationKeys,\n  getPersistables,\n  isProjectingRelations,\n  getKeys,\n  hasKeys,\n  buildSortMap,\n  flatObject,\n  getRawValue,\n  raw,\n  Raw,\n  getQueryFilterAsMap,\n} from '@uql/core/util';\n\nimport {\n  QueryFilter,\n  Query,\n  Scalar,\n  QueryFilterFieldOperatorMap,\n  QuerySort,\n  QueryPager,\n  QueryTextSearchOptions,\n  FieldKey,\n  QueryProject,\n  Type,\n  QueryCriteria,\n  QueryProjectArray,\n  QueryOptions,\n  QueryDialect,\n  QueryFilterOptions,\n  QueryComparisonOptions,\n  QueryFilterMap,\n  QuerySearch,\n  QueryProjectOptions,\n  QuerySortDirection,\n  QueryFilterLogical,\n} from '@uql/core/type';\n\nexport abstract class BaseSqlDialect implements QueryDialect {\n  readonly escapeIdRegex: RegExp;\n\n  constructor(readonly beginTransactionCommand: string, readonly escapeIdChar: '`' | '\"') {\n    this.escapeIdRegex = RegExp(escapeIdChar, 'g');\n  }\n\n  criteria<E>(entity: Type<E>, qm: Query<E>, opts: QueryOptions = {}): string {\n    const meta = getMeta(entity);\n    const prefix = opts.prefix ?? (opts.autoPrefix || isProjectingRelations(meta, qm.$project)) ? meta.name : undefined;\n    const where = this.where<E>(entity, qm.$filter, { ...opts, prefix });\n    const group = this.group<E>(entity, qm.$group, { ...opts, prefix });\n    const having = this.where<E>(entity, qm.$having, { ...opts, prefix, clause: 'HAVING' });\n    const sort = this.sort<E>(entity, qm.$sort, { ...opts, prefix });\n    const pager = this.pager(qm);\n    return where + group + having + sort + pager;\n  }\n\n  projectFields<E>(entity: Type<E>, project: QueryProject<E>, opts: QueryProjectOptions = {}): string {\n    const meta = getMeta(entity);\n    const prefix = opts.prefix ? opts.prefix + '.' : '';\n    const escapedPrefix = this.escapeId(opts.prefix, true, true);\n\n    let fields: QueryProjectArray<E>;\n\n    if (project) {\n      if (Array.isArray(project)) {\n        fields = project;\n      } else {\n        const positiveProjectKeys = getKeys(project).filter((key) => project[key]);\n        fields = positiveProjectKeys.length\n          ? positiveProjectKeys.map((key) => {\n              const val = project[key];\n              if (val instanceof Raw) {\n                return raw(val.value, key);\n              }\n              return key as FieldKey<E>;\n            })\n          : (getKeys(meta.fields).filter((key) => !(key in project)) as FieldKey<E>[]);\n      }\n      fields = fields.filter((key) => key instanceof Raw || meta.fields[key as FieldKey<E>]);\n      if (!fields.length || (opts.prefix && !fields.includes(meta.id))) {\n        fields = [meta.id, ...fields];\n      }\n    } else {\n      fields = getKeys(meta.fields) as FieldKey<E>[];\n    }\n\n    return fields\n      .map((key) => {\n        if (key instanceof Raw) {\n          return getRawValue({\n            value: key,\n            dialect: this,\n            prefix,\n            escapedPrefix,\n            autoPrefixAlias: opts.autoPrefixAlias,\n          });\n        }\n\n        const field = meta.fields[key as FieldKey<E>];\n\n        if (field.virtual) {\n          return getRawValue({\n            value: field.virtual,\n            alias: key as string,\n            dialect: this,\n            prefix,\n            escapedPrefix,\n            autoPrefixAlias: opts.autoPrefixAlias,\n          });\n        }\n\n        const name = field?.name ?? (key as FieldKey<E>);\n        const fieldPath = `${escapedPrefix}${this.escapeId(name)}`;\n\n        return !opts.autoPrefixAlias && (name === key || !field) ? fieldPath : `${fieldPath} ${this.escapeId((prefix + key) as FieldKey<E>, true)}`;\n      })\n      .join(', ');\n  }\n\n  projectRelations<E>(entity: Type<E>, project: QueryProject<E> = {}, { prefix }: { prefix?: string } = {}): { fields: string; tables: string } {\n    const meta = getMeta(entity);\n    const relations = getProjectRelationKeys(meta, project);\n    const isProjectArray = Array.isArray(project);\n    let fields = '';\n    let tables = '';\n\n    for (const key of relations) {\n      const relOpts = meta.relations[key];\n\n      if (relOpts.cardinality === '1m' || relOpts.cardinality === 'mm') {\n        // '1m' and 'mm' should be resolved in a higher layer because they will need multiple queries\n        continue;\n      }\n\n      const joinRelAlias = prefix ? prefix + '.' + key : key;\n      const relEntity = relOpts.entity();\n      const relProject = project[key as string];\n      const relQuery = isProjectArray ? {} : Array.isArray(relProject) ? { $project: relProject } : relProject;\n\n      const relColumns = this.projectFields(relEntity, relQuery.$project, {\n        prefix: joinRelAlias,\n        autoPrefixAlias: true,\n      });\n\n      fields += ', ' + relColumns;\n\n      const { fields: subColumns, tables: subTables } = this.projectRelations(relEntity, relQuery.$project, {\n        prefix: joinRelAlias,\n      });\n\n      fields += subColumns;\n\n      const relMeta = getMeta(relEntity);\n      const relEntityName = this.escapeId(relMeta.name);\n      const relPath = prefix ? this.escapeId(prefix, true) : this.escapeId(meta.name);\n      const joinType = relQuery.$required ? 'INNER' : 'LEFT';\n      const joinAlias = this.escapeId(joinRelAlias, true);\n\n      tables += ` ${joinType} JOIN ${relEntityName} ${joinAlias} ON `;\n      tables += relOpts.references.map((it) => `${joinAlias}.${this.escapeId(it.foreign)} = ${relPath}.${this.escapeId(it.local)}`).join(' AND ');\n\n      if (relQuery.$filter) {\n        const filter = this.where(relEntity, relQuery.$filter, { prefix: key, clause: false });\n        tables += ` AND ${filter}`;\n      }\n\n      tables += subTables;\n    }\n\n    return { fields, tables };\n  }\n\n  select<E>(entity: Type<E>, qm: Query<E>, opts: QueryOptions = {}): string {\n    const meta = getMeta(entity);\n    const prefix = opts.prefix ?? (opts.autoPrefix || isProjectingRelations(meta, qm.$project)) ? meta.name : undefined;\n\n    const fields = this.projectFields(entity, qm.$project, { prefix });\n    const { fields: relationFields, tables } = this.projectRelations(entity, qm.$project);\n\n    return `SELECT ${fields}${relationFields} FROM ${this.escapeId(meta.name)}${tables}`;\n  }\n\n  where<E>(entity: Type<E>, filter: QueryFilter<E> = {}, opts: QueryFilterOptions = {}): string {\n    const meta = getMeta(entity);\n    const { usePrecedence, clause = 'WHERE', softDelete } = opts;\n\n    filter = getQueryFilterAsMap(meta, filter);\n\n    if (meta.softDelete && (softDelete || softDelete === undefined) && clause !== 'HAVING' && !filter[meta.softDelete as string]) {\n      filter[meta.softDelete as string] = null;\n    }\n\n    const entries = Object.entries(filter);\n\n    if (!entries.length) {\n      return '';\n    }\n\n    const options = { ...opts, usePrecedence: entries.length > 1 };\n\n    let sql = entries.map(([key, val]) => this.compare(entity, key as keyof QueryFilterMap<E>, val as any, options)).join(` AND `);\n\n    if (usePrecedence) {\n      sql = `(${sql})`;\n    }\n\n    return clause ? ` ${clause} ${sql}` : sql;\n  }\n\n  compare<E, K extends keyof QueryFilterMap<E>>(entity: Type<E>, key: K, val: QueryFilterMap<E>[K], opts?: QueryComparisonOptions): string {\n    const meta = getMeta(entity);\n\n    if (val instanceof Raw) {\n      if (key === '$exists' || key === '$nexists') {\n        const value = val as Raw;\n        const query = getRawValue({\n          value,\n          dialect: this,\n          prefix: meta.name,\n          escapedPrefix: this.escapeId(meta.name, false, true),\n        });\n        return `${key === '$exists' ? 'EXISTS' : 'NOT EXISTS'} (${query})`;\n      }\n      const comparisonKey = this.getComparisonKey(entity, key as FieldKey<E>, opts);\n      return `${comparisonKey} = ${val.value}`;\n    }\n\n    if (key === '$text') {\n      const search = val as QueryTextSearchOptions<E>;\n      return `${this.escapeId(meta.name)} MATCH ${this.escape(search.$value)}`;\n    }\n\n    if (key === '$and' || key === '$or' || key === '$not' || key === '$nor') {\n      const negateOperatorMap = {\n        $not: '$and',\n        $nor: '$or',\n      } as const;\n\n      const op: '$and' | '$or' = negateOperatorMap[key as string] ?? key;\n      const negate = key in negateOperatorMap ? 'NOT ' : '';\n\n      const values = val as QueryFilterLogical<E>;\n      const hasManyItems = values.length > 1;\n      const logicalComparison = values\n        .map((filterEntry) => {\n          if (filterEntry instanceof Raw) {\n            return getRawValue({\n              value: filterEntry,\n              dialect: this,\n              prefix: opts.prefix,\n              escapedPrefix: this.escapeId(opts.prefix, true, true),\n            });\n          }\n          return this.where(entity, filterEntry, {\n            prefix: opts.prefix,\n            usePrecedence: hasManyItems && !Array.isArray(filterEntry) && getKeys(filterEntry).length > 1,\n            clause: false,\n          });\n        })\n        .join(op === '$or' ? ' OR ' : ' AND ');\n\n      return (opts.usePrecedence || negate) && hasManyItems ? `${negate}(${logicalComparison})` : `${negate}${logicalComparison}`;\n    }\n\n    const value = Array.isArray(val) ? { $in: val } : typeof val === 'object' && val !== null ? val : { $eq: val };\n    const operators = getKeys(value) as (keyof QueryFilterFieldOperatorMap<E>)[];\n    const comparisons = operators.map((op) => this.compareFieldOperator(entity, key as FieldKey<E>, op, value[op], opts)).join(' AND ');\n\n    return operators.length > 1 ? `(${comparisons})` : comparisons;\n  }\n\n  compareFieldOperator<E, K extends keyof QueryFilterFieldOperatorMap<E>>(\n    entity: Type<E>,\n    key: FieldKey<E>,\n    op: K,\n    val: QueryFilterFieldOperatorMap<E>[K],\n    opts?: QueryOptions\n  ): string {\n    const comparisonKey = this.getComparisonKey(entity, key, opts);\n    switch (op) {\n      case '$eq':\n        return val === null ? `${comparisonKey} IS NULL` : `${comparisonKey} = ${this.escape(val)}`;\n      case '$ne':\n        return val === null ? `${comparisonKey} IS NOT NULL` : `${comparisonKey} <> ${this.escape(val)}`;\n      case '$not':\n        return this.compare(entity, '$not', [{ [key]: val }] as any, opts);\n      case '$gt':\n        return `${comparisonKey} > ${this.escape(val)}`;\n      case '$gte':\n        return `${comparisonKey} >= ${this.escape(val)}`;\n      case '$lt':\n        return `${comparisonKey} < ${this.escape(val)}`;\n      case '$lte':\n        return `${comparisonKey} <= ${this.escape(val)}`;\n      case '$startsWith':\n        return `${comparisonKey} LIKE ${this.escape(`${val}%`)}`;\n      case '$istartsWith':\n        return `LOWER(${comparisonKey}) LIKE ${this.escape((val as string).toLowerCase() + '%')}`;\n      case '$endsWith':\n        return `${comparisonKey} LIKE ${this.escape(`%${val}`)}`;\n      case '$iendsWith':\n        return `LOWER(${comparisonKey}) LIKE ${this.escape('%' + (val as string).toLowerCase())}`;\n      case '$includes':\n        return `${comparisonKey} LIKE ${this.escape(`%${val}%`)}`;\n      case '$iincludes':\n        return `LOWER(${comparisonKey}) LIKE ${this.escape('%' + (val as string).toLowerCase() + '%')}`;\n      case '$ilike':\n        return `LOWER(${comparisonKey}) LIKE ${this.escape((val as string).toLowerCase())}`;\n      case '$like':\n        return `${comparisonKey} LIKE ${this.escape(val)}`;\n      case '$in':\n        return `${comparisonKey} IN (${this.escape(val)})`;\n      case '$nin':\n        return `${comparisonKey} NOT IN (${this.escape(val)})`;\n      case '$regex':\n        return `${comparisonKey} REGEXP ${this.escape(val)}`;\n      default:\n        throw new TypeError(`unknown operator: ${op}`);\n    }\n  }\n\n  getComparisonKey<E>(entity: Type<E>, key: FieldKey<E>, { prefix }: QueryOptions = {}): Scalar {\n    const meta = getMeta(entity);\n    const escapedPrefix = this.escapeId(prefix, true, true);\n    const field = meta.fields[key];\n\n    if (field?.virtual) {\n      return getRawValue({\n        value: field.virtual,\n        dialect: this,\n        prefix,\n        escapedPrefix,\n      });\n    }\n\n    return escapedPrefix + this.escapeId(field?.name ?? key);\n  }\n\n  group<E>(entity: Type<E>, fields: readonly FieldKey<E>[], opts: QueryOptions = {}): string {\n    if (!fields?.length) {\n      return '';\n    }\n    const meta = getMeta(entity);\n    const names = fields.map((key) => this.escapeId(meta.fields[key]?.name ?? key)).join(', ');\n    return ` GROUP BY ${names}`;\n  }\n\n  sort<E>(entity: Type<E>, sort: QuerySort<E>, { prefix }: QueryOptions = {}): string {\n    const sortMap = buildSortMap(sort);\n    if (!hasKeys(sortMap)) {\n      return '';\n    }\n    const meta = getMeta(entity);\n    const flattenedSort = flatObject(sortMap, prefix);\n    const directionMap = { 1: '', asc: '', '-1': ' DESC', desc: ' DESC' } as const;\n    const order = Object.entries(flattenedSort)\n      .map(([key, sort]) => {\n        const name = meta.fields[key]?.name ?? key;\n        const direction = directionMap[sort as QuerySortDirection];\n        return this.escapeId(name) + direction;\n      })\n      .join(', ');\n    return ` ORDER BY ${order}`;\n  }\n\n  pager(opts: QueryPager): string {\n    let sql = '';\n    if (opts.$limit) {\n      sql += ` LIMIT ${Number(opts.$limit)}`;\n    }\n    if (opts.$skip !== undefined) {\n      sql += ` OFFSET ${Number(opts.$skip)}`;\n    }\n    return sql;\n  }\n\n  count<E>(entity: Type<E>, qm: QuerySearch<E>, opts?: QueryOptions): string {\n    const search: Query<E> = {\n      ...qm,\n      $project: [raw('COUNT(*)', 'count')],\n    };\n\n    delete search.$sort;\n    delete search.$skip;\n    delete search.$limit;\n\n    const select = this.select<E>(entity, search);\n    const criteria = this.criteria(entity, search, opts);\n\n    return select + criteria;\n  }\n\n  find<E>(entity: Type<E>, qm: Query<E>, opts?: QueryOptions): string {\n    const select = this.select<E>(entity, qm, opts);\n    const criteria = this.criteria(entity, qm, opts);\n    return select + criteria;\n  }\n\n  insert<E>(entity: Type<E>, payload: E | E[]): string {\n    const meta = getMeta(entity);\n    payload = getPersistables(meta, payload, 'onInsert');\n    const keys = getKeys(payload[0]);\n    const columns = keys.map((key) => this.escapeId(meta.fields[key].name));\n    const values = payload.map((it) => keys.map((key) => this.escape(it[key])).join(', ')).join('), (');\n    return `INSERT INTO ${this.escapeId(meta.name)} (${columns.join(', ')}) VALUES (${values})`;\n  }\n\n  update<E>(entity: Type<E>, qm: QueryCriteria<E>, payload: E, opts?: QueryOptions): string {\n    const meta = getMeta(entity);\n    payload = getPersistable(meta, payload, 'onUpdate');\n    const values = getKeys(payload)\n      .map((key) => `${this.escapeId(key)} = ${this.escape(payload[key])}`)\n      .join(', ');\n    const criteria = this.criteria(entity, qm, opts);\n    return `UPDATE ${this.escapeId(meta.name)} SET ${values}${criteria}`;\n  }\n\n  delete<E>(entity: Type<E>, qm: QueryCriteria<E>, opts: QueryOptions = {}): string {\n    const meta = getMeta(entity);\n\n    if (opts.softDelete || opts.softDelete === undefined) {\n      if (meta.softDelete) {\n        const criteria = this.criteria(entity, qm, opts);\n        const value = meta.fields[meta.softDelete].onDelete();\n        return `UPDATE ${this.escapeId(meta.name)} SET ${this.escapeId(meta.softDelete)} = ${this.escape(value)}${criteria}`;\n      } else if (opts.softDelete) {\n        throw new TypeError(`'${meta.name}' has not enabled 'softDelete'`);\n      }\n    }\n\n    const criteria = this.criteria(entity, qm, opts);\n\n    return `DELETE FROM ${this.escapeId(meta.name)}${criteria}`;\n  }\n\n  escapeId(val: string, forbidQualified?: boolean, addDot?: boolean): string {\n    if (!val) {\n      return '';\n    }\n\n    if (!forbidQualified && val.includes('.')) {\n      return val\n        .split('.')\n        .map((it) => this.escapeId(it))\n        .join('.');\n    }\n\n    // sourced from 'escapeId' function here https://github.com/mysqljs/sqlstring/blob/master/lib/SqlString.js\n    const escaped = this.escapeIdChar + val.replace(this.escapeIdRegex, this.escapeIdChar + this.escapeIdChar) + this.escapeIdChar;\n\n    const suffix = addDot ? '.' : '';\n\n    return escaped + suffix;\n  }\n\n  escape(value: any): Scalar {\n    return escape(value);\n  }\n}\n"]}