drizzle-orm 0.10.45 → 0.11.0

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 (76) hide show
  1. package/README.md +47 -55
  2. package/builders/aggregators/abstractAggregator.js +1 -1
  3. package/builders/aggregators/selectAggregator.d.ts +2 -0
  4. package/builders/aggregators/selectAggregator.js +24 -3
  5. package/builders/highLvlBuilders/abstractRequestBuilder.d.ts +3 -3
  6. package/builders/highLvlBuilders/abstractRequestBuilder.js +1 -0
  7. package/builders/highLvlBuilders/joins/joinBuilderResponse.d.ts +8 -0
  8. package/builders/highLvlBuilders/joins/joinBuilderResponse.js +16 -0
  9. package/builders/highLvlBuilders/joins/proxies/joinProxies.d.ts +16 -0
  10. package/builders/highLvlBuilders/joins/proxies/joinProxies.js +47 -0
  11. package/builders/highLvlBuilders/joins/selectJoinBuilder.d.ts +43 -0
  12. package/builders/highLvlBuilders/joins/selectJoinBuilder.js +109 -0
  13. package/builders/highLvlBuilders/selectRequestBuilder.d.ts +28 -12
  14. package/builders/highLvlBuilders/selectRequestBuilder.js +45 -4
  15. package/builders/index.d.ts +3 -0
  16. package/builders/index.js +15 -1
  17. package/builders/joinBuilders/builders/abstractJoinBuilder.d.ts +1 -1
  18. package/builders/joinBuilders/builders/selectWithFiveJoins.d.ts +1 -1
  19. package/builders/joinBuilders/builders/selectWithFourJoins.d.ts +1 -1
  20. package/builders/joinBuilders/builders/selectWithJoin.d.ts +1 -1
  21. package/builders/joinBuilders/builders/selectWithThreeJoins.d.ts +1 -1
  22. package/builders/joinBuilders/builders/selectWithTwoJoins.d.ts +1 -1
  23. package/builders/lowLvlBuilders/selects/selectFrom.d.ts +3 -0
  24. package/builders/lowLvlBuilders/selects/selectFrom.js +1 -0
  25. package/builders/lowLvlBuilders/selects/selectJoined.d.ts +3 -0
  26. package/builders/lowLvlBuilders/selects/selectJoined.js +4 -0
  27. package/builders/requestBuilders/where/and.d.ts +1 -4
  28. package/builders/requestBuilders/where/and.js +2 -2
  29. package/builders/requestBuilders/where/const.d.ts +1 -4
  30. package/builders/requestBuilders/where/const.js +1 -1
  31. package/builders/requestBuilders/where/constArray.d.ts +1 -4
  32. package/builders/requestBuilders/where/constArray.js +1 -1
  33. package/builders/requestBuilders/where/eqWhere.d.ts +1 -1
  34. package/builders/requestBuilders/where/eqWhere.js +3 -3
  35. package/builders/requestBuilders/where/greater.d.ts +1 -4
  36. package/builders/requestBuilders/where/greater.js +3 -3
  37. package/builders/requestBuilders/where/greaterEq.d.ts +1 -4
  38. package/builders/requestBuilders/where/greaterEq.js +3 -3
  39. package/builders/requestBuilders/where/in.d.ts +1 -4
  40. package/builders/requestBuilders/where/in.js +3 -3
  41. package/builders/requestBuilders/where/isNotNull.d.ts +1 -4
  42. package/builders/requestBuilders/where/isNotNull.js +2 -2
  43. package/builders/requestBuilders/where/isNull.d.ts +1 -4
  44. package/builders/requestBuilders/where/isNull.js +2 -2
  45. package/builders/requestBuilders/where/less.d.ts +1 -4
  46. package/builders/requestBuilders/where/less.js +3 -3
  47. package/builders/requestBuilders/where/lessEq.d.ts +1 -4
  48. package/builders/requestBuilders/where/lessEq.js +3 -3
  49. package/builders/requestBuilders/where/like.d.ts +1 -4
  50. package/builders/requestBuilders/where/like.js +3 -3
  51. package/builders/requestBuilders/where/notEqWhere.d.ts +1 -4
  52. package/builders/requestBuilders/where/notEqWhere.js +3 -3
  53. package/builders/requestBuilders/where/or.d.ts +1 -4
  54. package/builders/requestBuilders/where/or.js +2 -2
  55. package/builders/requestBuilders/where/rawWhere.d.ts +1 -4
  56. package/builders/requestBuilders/where/rawWhere.js +1 -1
  57. package/builders/requestBuilders/where/static.d.ts +1 -0
  58. package/builders/requestBuilders/where/static.js +3 -1
  59. package/builders/requestBuilders/where/var.d.ts +1 -4
  60. package/builders/requestBuilders/where/var.js +2 -2
  61. package/builders/requestBuilders/where/where.d.ts +1 -4
  62. package/db/dbConnector.js +2 -2
  63. package/db/session.d.ts +5 -2
  64. package/db/session.js +8 -1
  65. package/docs/cases/simple_join.js +5 -5
  66. package/docs/tables/userGroupsTable.d.ts +1 -0
  67. package/docs/tables/userGroupsTable.js +1 -0
  68. package/docs/tables/usersTable.js +1 -1
  69. package/mappers/responseMapper.d.ts +3 -4
  70. package/mappers/responseMapper.js +2 -1
  71. package/package.json +6 -3
  72. package/serializer/serializer.d.ts +15 -4
  73. package/tables/abstractTable.d.ts +3 -4
  74. package/tables/abstractTable.js +1 -0
  75. package/tables/inferTypes.d.ts +6 -2
  76. package/test.js +108 -46
package/README.md CHANGED
@@ -342,79 +342,71 @@ await usersTable.insertMany([{
342
342
  const usersTable = new UsersTable(db);
343
343
  const citiesTable = new CitiesTable(db);
344
344
 
345
- const userWithCities = await citiesTable.select()
346
- .where(eq(citiesTable.id, 1))
347
- .leftJoin(UsersTable,
348
- (city) => city.userId,
349
- (users) => users.id)
350
- .execute();
345
+ const userWithCities = await citiesTable.select()
346
+ .where(eq(citiesTable.id, 1))
347
+ .leftJoin(usersTable, (cities, users) => onEq(cities.userId, users.id))
348
+ .execute();
351
349
 
352
350
  const citiesWithUserObject = userWithCities.map((city, user) => ({ ...city, user }));
353
351
  ```
354
352
 
355
353
  ### Join Many-To-Many Tables
356
- ##### Join User Groups with Users, using many-to-many table and map response to get user object with groups array
357
- ```typescript
358
- const usersWithUserGroups = await usersToUserGroupsTable.select()
359
- .where(eq(userGroupsTable.id, 1))
360
- .leftJoin(UsersTable,
361
- (userToGroup) => userToGroup.userId,
362
- (users) => users.id)
363
- .leftJoin(UsersToUserGroupsTable, UserGroupsTable,
364
- (userToGroup) => userToGroup.groupId,
365
- (users) => users.id)
366
- .execute();
367
-
368
- const userGroupWithUsers = usersWithUserGroups.group({
369
- one: (_, dbUser, dbUserGroup) => dbUser!,
370
- many: (_, dbUser, dbUserGroup) => dbUserGroup!,
371
- });
372
-
373
- const userWithGroups: ExtractModel<UsersTable> & { groups: ExtractModel<UserGroupsTable>[] } = {
374
- ...userGroupWithUsers.one,
375
- groups: userGroupWithUsers.many,
376
- };
377
- ```
378
- ##### Join User Groups with Users, using many-to-many table and map response to get user group object with users array
354
+ ##### Join User Groups with Users, using many-to-many table
379
355
  ```typescript
380
- const usersWithUserGroups = await usersToUserGroupsTable.select()
356
+ const usersWithUserGroups = await usersToUserGroupsTable.select()
381
357
  .where(eq(userGroupsTable.id, 1))
382
- .leftJoin(UsersTable,
383
- (userToGroup) => userToGroup.userId,
384
- (users) => users.id)
385
- .leftJoin(UsersToUserGroupsTable, UserGroupsTable,
386
- (userToGroup) => userToGroup.groupId,
387
- (users) => users.id)
358
+ .leftJoin(usersTable, (usersToUserGroups, users) => onEq(usersToUserGroups.userId, users.id))
359
+ .leftJoin(userGroupsTable, (usersToUserGroups, _users, userGroups) => onEq(usersToUserGroups.groupId, userGroups.id))
388
360
  .execute();
389
-
390
- const userGroupWithUsers = usersWithUserGroups.group({
391
- one: (_, dbUser, dbUserGroup) => dbUserGroup!,
392
- many: (_, dbUser, dbUserGroup) => dbUser!,
393
- });
394
-
395
- const userWithGroups: ExtractModel<UserGroupsTable> & { users: ExtractModel<UsersTable>[] } = {
396
- ...userGroupWithUsers.one,
397
- users: userGroupWithUsers.many,
398
- };
399
361
  ```
400
362
  ### Join using partial field select
401
363
  ##### Join Cities with Users getting only needed fields form request
402
364
  ```typescript
403
- await citiesTable.select({
365
+ await citiesTable.select({
404
366
  id: citiesTable.id,
405
367
  userId: citiesTable.userId,
406
368
  })
407
- .where(eq(citiesTable.id, 1))
408
- .leftJoin(UsersTable,
409
- (city) => city.userId,
410
- (users) => users.id,
411
- {
412
- id: usersTable.id,
413
- })
414
- .execute();
369
+ .where(eq(citiesTable.id, 1))
370
+ .leftJoin(usersTable, (cities, users) => onEq(cities.userId, users.id))
371
+ .execute();
415
372
 
416
373
  const citiesWithUserObject = userWithCities.map((city, user) => ({ ...city, user }));
417
374
  ```
375
+ ### Another join examples with different callback ON statements
376
+ ```typescript
377
+ await citiesTable.select()
378
+ .where(eq(citiesTable.location, 'q'))
379
+ .leftJoin(usersTable, (cities, _users) => eq(cities.id, 13))
380
+ .execute();
381
+ // Join statement generated from query
382
+ // LEFT JOIN users AS users_1
383
+ // ON cities."id"=$1
384
+ // WHERE cities."page"=$2
385
+ //
386
+ // Values: [13, 'q']
387
+
388
+ await citiesTable.select()
389
+ .leftJoin(usersTable, (cities, _users) => and([
390
+ eq(cities.id, 13), notEq(cities.id, 14),
391
+ ]))
392
+ .execute();
393
+ // Join statement generated from query
394
+ // LEFT JOIN users AS users_1
395
+ // ON (cities."id"=$1 and cities."id"!=$2)
396
+ //
397
+ // Values: [13, 14]
398
+
399
+ await citiesTable.select()
400
+ .where(eq(citiesTable.location, 'location'))
401
+ .leftJoin(usersTable, (_cities, _users) => raw('<custom expression after ON statement>'))
402
+ .execute();
403
+ // Join statement generated from query
404
+ // LEFT JOIN users AS users_1
405
+ // ON <custom expression after ON statement>
406
+ // WHERE cities."page"=$1
407
+ //
408
+ // Values: ['location']
409
+ ```
418
410
 
419
411
 
420
412
  ## Migrations
@@ -16,7 +16,7 @@ class Aggregator {
16
16
  selectFields.push('.');
17
17
  selectFields.push(ecranate_1.ecranate(field.getColumnName()));
18
18
  selectFields.push(' AS ');
19
- selectFields.push(ecranate_1.ecranate(`${field.getAlias()}${id ? `_${id}` : ''}`));
19
+ selectFields.push(ecranate_1.ecranate(`${table}_${field.getColumnName()}`));
20
20
  selectFields.push(',');
21
21
  }
22
22
  });
@@ -1,6 +1,7 @@
1
1
  import { AbstractColumn } from '../../columns/column';
2
2
  import ColumnType from '../../columns/types/columnType';
3
3
  import { AbstractTable } from '../../tables';
4
+ import { JoinType } from '../highLvlBuilders/joins/selectJoinBuilder';
4
5
  import Order from '../highLvlBuilders/order';
5
6
  import Join from '../joinBuilders/join';
6
7
  import Expr from '../requestBuilders/where/where';
@@ -25,6 +26,7 @@ export default class SelectAggregator extends Aggregator {
25
26
  orderBy: (column?: AbstractColumn<ColumnType<{}>, boolean, boolean, any> | undefined, order?: Order | undefined) => SelectAggregator;
26
27
  distinct: (column?: AbstractColumn<ColumnType<{}>, boolean, boolean, any> | undefined) => SelectAggregator;
27
28
  appendFrom: (tableName: string) => SelectAggregator;
29
+ join2: (joins: Array<JoinType<AbstractTable<any>>>) => SelectAggregator;
28
30
  join: (joins: {
29
31
  join: Join<any>;
30
32
  partial?: {
@@ -24,10 +24,10 @@ class SelectAggregator extends abstractAggregator_1.default {
24
24
  this._joinCache = {};
25
25
  this.filters = (filters) => {
26
26
  if (filters) {
27
- const queryBuilder = filters.toQuery({ position: 1, tableCache: this._joinCache, session: this._table.db.session() });
27
+ const queryBuilder = filters.toQuery({ position: this._values.length + 1, session: this._table.db.session() });
28
28
  this._filters.push('WHERE ');
29
29
  this._filters.push(queryBuilder.query);
30
- this._values = queryBuilder.values;
30
+ this._values.push(...queryBuilder.values);
31
31
  }
32
32
  return this;
33
33
  };
@@ -64,7 +64,28 @@ class SelectAggregator extends abstractAggregator_1.default {
64
64
  this._from.push('FROM ');
65
65
  this._from.push(tableName);
66
66
  // this._from.push(`${tableName} AS ${tableName}_0`);
67
- this._joinCache[tableName] = tableName;
67
+ // this._joinCache[tableName] = tableName;
68
+ return this;
69
+ };
70
+ this.join2 = (joins) => {
71
+ const tableFrom = this._table.tableName();
72
+ for (const [index, join] of joins.entries()) {
73
+ const { table: tableToName, originalName, onExpression, type, columns, } = join;
74
+ const selectString = this.generateSelectArray(tableToName, Object.values(columns)).join('');
75
+ this._fields.push(', ');
76
+ this._fields.push(selectString);
77
+ this._join.push('\n');
78
+ this._join.push(type);
79
+ this._join.push(' ');
80
+ this._join.push(originalName);
81
+ this._join.push(' ');
82
+ this._join.push(`AS ${tableToName}`);
83
+ this._join.push('\n');
84
+ this._join.push('ON ');
85
+ const joinExpr = onExpression.toQuery({ position: this._values.length + 1, session: this._table.db.session() });
86
+ this._values.push(...joinExpr.values);
87
+ this._join.push(joinExpr.query);
88
+ }
68
89
  return this;
69
90
  };
70
91
  // Add select generator for second table also
@@ -4,9 +4,8 @@ import { ISession } from '../../db/session';
4
4
  import BaseLogger from '../../logger/abstractLogger';
5
5
  import { AbstractTable } from '../../tables';
6
6
  import { ExtractModel } from '../../tables/inferTypes';
7
- export default abstract class TableRequestBuilder<TTable extends AbstractTable<TTable>, TPartial extends {
8
- [name: string]: AbstractColumn<ColumnType<any>, boolean, boolean, TTable>;
9
- } = {}> {
7
+ import { EmptyPartial } from './joins/selectJoinBuilder';
8
+ export default abstract class TableRequestBuilder<TTable extends AbstractTable<TTable>, TPartial extends EmptyPartial<TTable>> {
10
9
  protected _table: TTable;
11
10
  protected _session: ISession;
12
11
  protected _mappedServiceToDb: {
@@ -21,6 +20,7 @@ export default abstract class TableRequestBuilder<TTable extends AbstractTable<T
21
20
  /**
22
21
  * Current function will return an element only if response is of length 1
23
22
  * If there are more or less than 1 element, will throw an Error
23
+ * @deprecated Use `.execute()` or `.all()` instead
24
24
  */
25
25
  findOne: () => Promise<[keyof TPartial] extends [never] ? ExtractModel<TTable> : ExtractModel<TPartial>>;
26
26
  protected abstract _execute(): Promise<Array<[keyof TPartial] extends [never] ? ExtractModel<TTable> : ExtractModel<TPartial>>>;
@@ -9,6 +9,7 @@ class TableRequestBuilder {
9
9
  /**
10
10
  * Current function will return an element only if response is of length 1
11
11
  * If there are more or less than 1 element, will throw an Error
12
+ * @deprecated Use `.execute()` or `.all()` instead
12
13
  */
13
14
  this.findOne = async () => {
14
15
  const executionRes = await this._execute();
@@ -0,0 +1,8 @@
1
+ import AbstractTable from '@/tables/abstractTable';
2
+ import { FullOrPartial } from '@/tables/inferTypes';
3
+ export default class JoinBuilderResponses<TResponses extends FullOrPartial<AbstractTable<any>, any>[]> {
4
+ private responses;
5
+ private responsesType;
6
+ constructor(responses: TResponses[]);
7
+ map: <M>(imac: (...args_0: TResponses) => M) => M[];
8
+ }
@@ -0,0 +1,16 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ class JoinBuilderResponses {
4
+ constructor(responses) {
5
+ this.responses = responses;
6
+ this.map = (imac) => {
7
+ const objects = new Array();
8
+ for (let i = 0; i < this.responses[0].length; i += 1) {
9
+ const responses = this.responses.map((res) => res[i]);
10
+ objects.push(imac(...responses));
11
+ }
12
+ return objects;
13
+ };
14
+ }
15
+ }
16
+ exports.default = JoinBuilderResponses;
@@ -0,0 +1,16 @@
1
+ import AbstractTable from '@/tables/abstractTable';
2
+ import ColumnType from '../../../../columns/types/columnType';
3
+ import { AbstractColumn } from '../../../../columns/column';
4
+ export declare class JoinedColumn<TJoinedTableColumn extends AbstractColumn<ColumnType>> implements ProxyHandler<TJoinedTableColumn> {
5
+ private tableObj;
6
+ private aliasCounter;
7
+ private parentName;
8
+ constructor(tableObj: any, aliasCounter: number);
9
+ get(columnObj: TJoinedTableColumn, prop: string): (() => string) | TJoinedTableColumn[keyof TJoinedTableColumn];
10
+ }
11
+ export declare class JoinedHandler<TJoinedTable extends AbstractTable<TJoinedTable>> implements ProxyHandler<TJoinedTable> {
12
+ private aliasCounter;
13
+ private tableName;
14
+ constructor(aliasCounter: number);
15
+ get(tableObj: TJoinedTable, prop: string | symbol, receiver: any): any;
16
+ }
@@ -0,0 +1,47 @@
1
+ "use strict";
2
+ /* eslint-disable max-classes-per-file */
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.JoinedHandler = exports.JoinedColumn = void 0;
5
+ const column_1 = require("../../../../columns/column");
6
+ class JoinedColumn {
7
+ constructor(tableObj, aliasCounter) {
8
+ this.tableObj = tableObj;
9
+ this.aliasCounter = aliasCounter;
10
+ this.parentName = 'getParentName';
11
+ }
12
+ get(columnObj, prop) {
13
+ if (prop === this.parentName) {
14
+ return () => `${this.tableObj.tableName()}_${this.aliasCounter}`;
15
+ }
16
+ return columnObj[prop];
17
+ }
18
+ }
19
+ exports.JoinedColumn = JoinedColumn;
20
+ class JoinedHandler {
21
+ constructor(aliasCounter) {
22
+ this.aliasCounter = aliasCounter;
23
+ this.tableName = 'tableName';
24
+ }
25
+ get(tableObj, prop, receiver) {
26
+ if (prop === this.tableName) {
27
+ return () => `${tableObj.tableName()}_${this.aliasCounter}`;
28
+ }
29
+ const columnProp = tableObj[prop];
30
+ if (columnProp instanceof column_1.AbstractColumn) {
31
+ return new Proxy(columnProp, new JoinedColumn(tableObj, this.aliasCounter));
32
+ }
33
+ if (prop === 'mapServiceToDb') {
34
+ return () => Object.getOwnPropertyNames(tableObj)
35
+ .reduce((res, fieldName) => {
36
+ const field1 = tableObj[fieldName];
37
+ if (field1 instanceof column_1.AbstractColumn) {
38
+ const field = new Proxy(field1, new JoinedColumn(tableObj, this.aliasCounter));
39
+ res[fieldName] = field;
40
+ }
41
+ return res;
42
+ }, {});
43
+ }
44
+ return tableObj[prop];
45
+ }
46
+ }
47
+ exports.JoinedHandler = JoinedHandler;
@@ -0,0 +1,43 @@
1
+ import { ExtractPartialObjectFromColumns, FullOrPartial, PartialFor } from '../../../tables/inferTypes';
2
+ import AbstractTable from '../../../tables/abstractTable';
3
+ import BaseLogger from '../../../logger/abstractLogger';
4
+ import ColumnType from '../../../columns/types/columnType';
5
+ import { AbstractColumn } from '../../../columns/column';
6
+ import Expr from '../../requestBuilders/where/where';
7
+ import Order from '../order';
8
+ import JoinBuilderResponses from './joinBuilderResponse';
9
+ export interface JoinType<T extends AbstractTable<T>> {
10
+ table: string;
11
+ columns: ExtractPartialObjectFromColumns<T>;
12
+ onExpression: Expr;
13
+ type: string;
14
+ originalName: string;
15
+ }
16
+ export declare type TableIfPartialIsUndefined<TPartial, TJoinedTableMapped> = TPartial extends undefined ? TJoinedTableMapped : TPartial;
17
+ export declare type EmptyPartial<T extends AbstractTable<T>> = PartialFor<T> | undefined;
18
+ export default class JoinBuilder<TJoins extends {
19
+ [name: string]: any;
20
+ }[], TJoinsResponses extends FullOrPartial<AbstractTable<any>, any>[]> {
21
+ private joins;
22
+ private aliasCounter;
23
+ private rootTable;
24
+ private _filter;
25
+ private _props;
26
+ private _orderBy?;
27
+ private _order?;
28
+ private _distinct?;
29
+ private _partial?;
30
+ private _logger?;
31
+ private joinedTables;
32
+ private joinResponses;
33
+ constructor(joins: TJoins, onTableOriginalName: string, aliasCounter: number, onExpression: Expr, onTable: AbstractTable<any>, type: string, rootTable: AbstractTable<any>, _filter: Expr, _props: {
34
+ limit?: number;
35
+ offset?: number;
36
+ }, _orderBy?: AbstractColumn<ColumnType<{}>, boolean, boolean, any> | undefined, _order?: Order | undefined, _distinct?: AbstractColumn<ColumnType<{}>, boolean, boolean, any> | undefined, _partial?: any, _logger?: BaseLogger | undefined);
37
+ innerJoin<TJoinedTable extends AbstractTable<TJoinedTable>, TPartial extends EmptyPartial<TJoinedTable> = undefined>(value: TJoinedTable, callback: (...args: [...TJoins, TableIfPartialIsUndefined<TPartial, ExtractPartialObjectFromColumns<TJoinedTable>>]) => Expr, partial?: TPartial): JoinBuilder<[...TJoins, TableIfPartialIsUndefined<TPartial, ExtractPartialObjectFromColumns<TJoinedTable>>], [...TJoinsResponses, FullOrPartial<TJoinedTable, TPartial>]>;
38
+ leftJoin<TJoinedTable extends AbstractTable<TJoinedTable>, TPartial extends EmptyPartial<TJoinedTable> = undefined>(value: TJoinedTable, callback: (...args: [...TJoins, TableIfPartialIsUndefined<TPartial, ExtractPartialObjectFromColumns<TJoinedTable>>]) => Expr, partial?: TPartial): JoinBuilder<[...TJoins, TableIfPartialIsUndefined<TPartial, ExtractPartialObjectFromColumns<TJoinedTable>>], [...TJoinsResponses, FullOrPartial<TJoinedTable, TPartial>]>;
39
+ rightJoin<TJoinedTable extends AbstractTable<TJoinedTable>, TPartial extends EmptyPartial<TJoinedTable> = undefined>(value: TJoinedTable, callback: (...args: [...TJoins, TableIfPartialIsUndefined<TPartial, ExtractPartialObjectFromColumns<TJoinedTable>>]) => Expr, partial?: TPartial): JoinBuilder<[...TJoins, TableIfPartialIsUndefined<TPartial, ExtractPartialObjectFromColumns<TJoinedTable>>], [...TJoinsResponses, FullOrPartial<TJoinedTable, TPartial>]>;
40
+ fullJoin<TJoinedTable extends AbstractTable<TJoinedTable>, TPartial extends EmptyPartial<TJoinedTable> = undefined>(value: TJoinedTable, callback: (...args: [...TJoins, TableIfPartialIsUndefined<TPartial, ExtractPartialObjectFromColumns<TJoinedTable>>]) => Expr, partial?: TPartial): JoinBuilder<[...TJoins, TableIfPartialIsUndefined<TPartial, ExtractPartialObjectFromColumns<TJoinedTable>>], [...TJoinsResponses, FullOrPartial<TJoinedTable, TPartial>]>;
41
+ execute(): Promise<JoinBuilderResponses<TJoinsResponses>>;
42
+ private join;
43
+ }
@@ -0,0 +1,109 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });
5
+ }) : (function(o, m, k, k2) {
6
+ if (k2 === undefined) k2 = k;
7
+ o[k2] = m[k];
8
+ }));
9
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
10
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
11
+ }) : function(o, v) {
12
+ o["default"] = v;
13
+ });
14
+ var __importStar = (this && this.__importStar) || function (mod) {
15
+ if (mod && mod.__esModule) return mod;
16
+ var result = {};
17
+ if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
18
+ __setModuleDefault(result, mod);
19
+ return result;
20
+ };
21
+ var __importDefault = (this && this.__importDefault) || function (mod) {
22
+ return (mod && mod.__esModule) ? mod : { "default": mod };
23
+ };
24
+ Object.defineProperty(exports, "__esModule", { value: true });
25
+ const responseMapper_1 = __importDefault(require("../../../mappers/responseMapper"));
26
+ const builderError_1 = __importStar(require("../../../errors/builderError"));
27
+ const select_1 = __importDefault(require("../../lowLvlBuilders/selects/select"));
28
+ const joinBuilderResponse_1 = __importDefault(require("./joinBuilderResponse"));
29
+ const joinProxies_1 = require("./proxies/joinProxies");
30
+ class JoinBuilder {
31
+ constructor(joins, onTableOriginalName, aliasCounter, onExpression, onTable, type, rootTable, _filter, _props, _orderBy, _order, _distinct, _partial, _logger) {
32
+ this.joins = joins;
33
+ this.aliasCounter = aliasCounter;
34
+ this.rootTable = rootTable;
35
+ this._filter = _filter;
36
+ this._props = _props;
37
+ this._orderBy = _orderBy;
38
+ this._order = _order;
39
+ this._distinct = _distinct;
40
+ this._partial = _partial;
41
+ this._logger = _logger;
42
+ this.joinedTables = [];
43
+ this.joinedTables.push({
44
+ table: onTable.tableName(),
45
+ columns: joins[1],
46
+ originalName: onTableOriginalName,
47
+ onExpression,
48
+ type,
49
+ });
50
+ }
51
+ innerJoin(value, callback, partial) {
52
+ return this.join(value, callback, 'INNER JOIN', partial);
53
+ }
54
+ leftJoin(value, callback, partial) {
55
+ return this.join(value, callback, 'LEFT JOIN', partial);
56
+ }
57
+ rightJoin(value, callback, partial) {
58
+ return this.join(value, callback, 'RIGHT JOIN', partial);
59
+ }
60
+ fullJoin(value, callback, partial) {
61
+ return this.join(value, callback, 'FULL JOIN', partial);
62
+ }
63
+ async execute() {
64
+ const queryBuilder = select_1.default
65
+ .from(this.rootTable, this._partial)
66
+ .distinct(this._distinct)
67
+ .joined2(this.joinedTables)
68
+ .limit(this._props.limit)
69
+ .offset(this._props.offset)
70
+ .filteredBy(this._filter)
71
+ .orderBy(this._orderBy, this._order);
72
+ let query = '';
73
+ let values = [];
74
+ try {
75
+ const builderResult = queryBuilder.build();
76
+ query = builderResult.query;
77
+ values = builderResult.values;
78
+ }
79
+ catch (e) {
80
+ throw new builderError_1.default(builderError_1.BuilderType.JOINED_SELECT, this.rootTable.tableName(), Object.values(this.rootTable.mapServiceToDb()), e, this.rootTable.db.session(), this._filter);
81
+ }
82
+ if (this._logger) {
83
+ this._logger.info(`Selecting from ${this.rootTable.tableName()} using query:\n ${query}`);
84
+ this._logger.info(`Values for query:\n ${values}`);
85
+ }
86
+ const result = await this.rootTable.db.session().execute(query, values);
87
+ const mappedResponse = this.joins.map((joinedTable) => responseMapper_1.default.map(joinedTable, result));
88
+ return new joinBuilderResponse_1.default(mappedResponse);
89
+ }
90
+ join(value, callback, joinType, partial) {
91
+ this.aliasCounter += 1;
92
+ const valueAsProxy = new Proxy(value, new joinProxies_1.JoinedHandler(this.aliasCounter));
93
+ if (partial) {
94
+ for (const key of Object.keys(partial)) {
95
+ // eslint-disable-next-line no-param-reassign
96
+ partial[key] = new Proxy(partial[key], new joinProxies_1.JoinedColumn(value, this.aliasCounter));
97
+ }
98
+ }
99
+ const obj = partial ? partial
100
+ : valueAsProxy.mapServiceToDb();
101
+ const onExpression = callback(...this.joins, obj);
102
+ this.joins.push(partial !== null && partial !== void 0 ? partial : valueAsProxy.mapServiceToDb());
103
+ this.joinedTables.push({
104
+ table: valueAsProxy.tableName(), columns: obj, originalName: value.tableName(), onExpression, type: joinType,
105
+ });
106
+ return this;
107
+ }
108
+ }
109
+ exports.default = JoinBuilder;
@@ -4,14 +4,13 @@ import DB from '../../db/db';
4
4
  import { ISession } from '../../db/session';
5
5
  import BaseLogger from '../../logger/abstractLogger';
6
6
  import { AbstractTable } from '../../tables';
7
- import { ExtractModel, PartialFor } from '../../tables/inferTypes';
7
+ import { ExtractModel, ExtractPartialObjectFromColumns, FullOrPartial, PartialFor } from '../../tables/inferTypes';
8
8
  import SelectTRBWithJoin from '../joinBuilders/builders/selectWithJoin';
9
9
  import Expr from '../requestBuilders/where/where';
10
10
  import TableRequestBuilder from './abstractRequestBuilder';
11
+ import JoinBuilder, { EmptyPartial, TableIfPartialIsUndefined } from './joins/selectJoinBuilder';
11
12
  import Order from './order';
12
- export default class SelectTRB<TTable extends AbstractTable<TTable>, TPartial extends {
13
- [name: string]: AbstractColumn<ColumnType<any>, boolean, boolean, TTable>;
14
- } = {}> extends TableRequestBuilder<TTable, TPartial> {
13
+ export default class SelectTRB<TTable extends AbstractTable<TTable>, TPartial extends EmptyPartial<TTable> = undefined> extends TableRequestBuilder<TTable, TPartial> {
15
14
  protected _filter: Expr;
16
15
  private props;
17
16
  private __orderBy?;
@@ -25,23 +24,40 @@ export default class SelectTRB<TTable extends AbstractTable<TTable>, TPartial ex
25
24
  limit?: number;
26
25
  offset?: number;
27
26
  }, table: AbstractTable<TTable>, logger?: BaseLogger, partial?: TPartial);
27
+ private join;
28
+ innerJoin<TJoinedTable extends AbstractTable<TJoinedTable>, TJoinedPartial extends EmptyPartial<TJoinedTable> = undefined>(value: TJoinedTable, callback: (...args: [TableIfPartialIsUndefined<TPartial, ExtractPartialObjectFromColumns<TTable>>, TableIfPartialIsUndefined<TJoinedPartial, ExtractPartialObjectFromColumns<TJoinedTable>>]) => Expr, partial?: TJoinedPartial): JoinBuilder<[TableIfPartialIsUndefined<TPartial, ExtractPartialObjectFromColumns<TTable>>, TableIfPartialIsUndefined<TJoinedPartial, ExtractPartialObjectFromColumns<TJoinedTable>>], [FullOrPartial<TTable, TPartial>, FullOrPartial<TJoinedTable, TJoinedPartial>]>;
29
+ leftJoin<TJoinedTable extends AbstractTable<TJoinedTable>, TJoinedPartial extends EmptyPartial<TJoinedTable> = undefined>(value: TJoinedTable, callback: (...args: [TableIfPartialIsUndefined<TPartial, ExtractPartialObjectFromColumns<TTable>>, TableIfPartialIsUndefined<TJoinedPartial, ExtractPartialObjectFromColumns<TJoinedTable>>]) => Expr, partial?: TJoinedPartial): JoinBuilder<[TableIfPartialIsUndefined<TPartial, ExtractPartialObjectFromColumns<TTable>>, TableIfPartialIsUndefined<TJoinedPartial, ExtractPartialObjectFromColumns<TJoinedTable>>], [FullOrPartial<TTable, TPartial>, FullOrPartial<TJoinedTable, TJoinedPartial>]>;
30
+ rightJoin<TJoinedTable extends AbstractTable<TJoinedTable>, TJoinedPartial extends EmptyPartial<TJoinedTable> = undefined>(value: TJoinedTable, callback: (...args: [TableIfPartialIsUndefined<TPartial, ExtractPartialObjectFromColumns<TTable>>, TableIfPartialIsUndefined<TJoinedPartial, ExtractPartialObjectFromColumns<TJoinedTable>>]) => Expr, partial?: TJoinedPartial): JoinBuilder<[TableIfPartialIsUndefined<TPartial, ExtractPartialObjectFromColumns<TTable>>, TableIfPartialIsUndefined<TJoinedPartial, ExtractPartialObjectFromColumns<TJoinedTable>>], [FullOrPartial<TTable, TPartial>, FullOrPartial<TJoinedTable, TJoinedPartial>]>;
31
+ fullJoin<TJoinedTable extends AbstractTable<TJoinedTable>, TJoinedPartial extends EmptyPartial<TJoinedTable> = undefined>(value: TJoinedTable, callback: (...args: [TableIfPartialIsUndefined<TPartial, ExtractPartialObjectFromColumns<TTable>>, TableIfPartialIsUndefined<TJoinedPartial, ExtractPartialObjectFromColumns<TJoinedTable>>]) => Expr, partial?: TJoinedPartial): JoinBuilder<[TableIfPartialIsUndefined<TPartial, ExtractPartialObjectFromColumns<TTable>>, TableIfPartialIsUndefined<TJoinedPartial, ExtractPartialObjectFromColumns<TJoinedTable>>], [FullOrPartial<TTable, TPartial>, FullOrPartial<TJoinedTable, TJoinedPartial>]>;
28
32
  where: (expr: Expr) => SelectTRB<TTable, TPartial>;
29
33
  orderBy<TColumnType extends ColumnType>(callback: (table: TTable) => AbstractColumn<TColumnType, boolean, boolean>, order: Order): SelectTRB<TTable, TPartial>;
30
34
  distinct: (column: AbstractColumn<ColumnType<any>, boolean, boolean>) => SelectTRB<TTable, TPartial>;
31
35
  limit: (limit: number) => SelectTRB<TTable, TPartial>;
32
36
  offset: (offset: number) => SelectTRB<TTable, TPartial>;
33
- innerJoin<TColumn extends ColumnType, TToColumn extends ColumnType, IToTable extends AbstractTable<IToTable>, IToPartial extends PartialFor<IToTable> = {}>(table: {
37
+ /**
38
+ * @deprecated Since version 0.11.0. Will be deleted in version 0.12.0. Use {@link innerJoinV2()} instead
39
+ */
40
+ innerJoinV1<TColumn extends ColumnType, TToColumn extends ColumnType, IToTable extends AbstractTable<IToTable>, IToPartial extends PartialFor<IToTable>>(table: {
34
41
  new (db: DB): IToTable;
35
- }, from: (table: TTable) => AbstractColumn<TColumn, boolean, boolean>, to: (table: IToTable) => AbstractColumn<TToColumn, boolean, boolean>, partial?: IToPartial): SelectTRBWithJoin<TTable, IToTable, TPartial, IToPartial>;
36
- leftJoin<TColumn extends ColumnType<any>, IToColumn extends ColumnType<any>, IToTable extends AbstractTable<IToTable>, IToPartial extends PartialFor<IToTable> = {}>(table: {
42
+ }, from: (table: TTable) => AbstractColumn<TColumn, boolean, boolean>, to: (table: IToTable) => AbstractColumn<TToColumn, boolean, boolean>, partial?: IToPartial): SelectTRBWithJoin<TTable, IToTable, PartialFor<TTable>, IToPartial>;
43
+ /**
44
+ * @deprecated Since version 0.11.0. Will be deleted in version 0.12.0. Use {@link leftJoin()} instead
45
+ */
46
+ leftJoinV1<TColumn extends ColumnType<any>, IToColumn extends ColumnType<any>, IToTable extends AbstractTable<IToTable>, IToPartial extends PartialFor<IToTable> = {}>(table: {
37
47
  new (db: DB): IToTable;
38
- }, from: (table: TTable) => AbstractColumn<TColumn, boolean, boolean, TTable>, to: (table: IToTable) => AbstractColumn<IToColumn, boolean, boolean, IToTable>, partial?: IToPartial): SelectTRBWithJoin<TTable, IToTable, TPartial, IToPartial>;
39
- rightJoin<TColumn extends ColumnType, TToColumn extends ColumnType, IToTable extends AbstractTable<IToTable>, IToPartial extends PartialFor<IToTable> = {}>(table: {
48
+ }, from: (table: TTable) => AbstractColumn<TColumn, boolean, boolean, TTable>, to: (table: IToTable) => AbstractColumn<IToColumn, boolean, boolean, IToTable>, partial?: IToPartial): SelectTRBWithJoin<TTable, IToTable, PartialFor<TTable>, IToPartial>;
49
+ /**
50
+ * @deprecated Since version 0.11.0. Will be deleted in version 0.12.0. Use {@link rightJoin()} instead
51
+ */
52
+ rightJoinV1<TColumn extends ColumnType, TToColumn extends ColumnType, IToTable extends AbstractTable<IToTable>, IToPartial extends PartialFor<IToTable> = {}>(table: {
40
53
  new (db: DB): IToTable;
41
- }, from: (table: TTable) => AbstractColumn<TColumn, boolean, boolean>, to: (table: IToTable) => AbstractColumn<TToColumn, boolean, boolean>, partial?: IToPartial): SelectTRBWithJoin<TTable, IToTable, TPartial, IToPartial>;
42
- fullJoin<TColumn extends ColumnType, TToColumn extends ColumnType, IToTable extends AbstractTable<IToTable>, IToPartial extends PartialFor<IToTable> = {}>(table: {
54
+ }, from: (table: TTable) => AbstractColumn<TColumn, boolean, boolean>, to: (table: IToTable) => AbstractColumn<TToColumn, boolean, boolean>, partial?: IToPartial): SelectTRBWithJoin<TTable, IToTable, PartialFor<TTable>, IToPartial>;
55
+ /**
56
+ * @deprecated Since version 0.11.0. Will be deleted in version 0.12.0. Use {@link fullJoin()} instead
57
+ */
58
+ fullJoinV1<TColumn extends ColumnType, TToColumn extends ColumnType, IToTable extends AbstractTable<IToTable>, IToPartial extends PartialFor<IToTable> = {}>(table: {
43
59
  new (db: DB): IToTable;
44
- }, from: (table: TTable) => AbstractColumn<TColumn, boolean, boolean>, to: (table: IToTable) => AbstractColumn<TToColumn, boolean, boolean>, partial?: IToPartial): SelectTRBWithJoin<TTable, IToTable, TPartial, IToPartial>;
60
+ }, from: (table: TTable) => AbstractColumn<TColumn, boolean, boolean>, to: (table: IToTable) => AbstractColumn<TToColumn, boolean, boolean>, partial?: IToPartial): SelectTRBWithJoin<TTable, IToTable, PartialFor<TTable>, IToPartial>;
45
61
  execute: () => Promise<([keyof TPartial] extends [never] ? ExtractModel<TTable> : ExtractModel<TPartial>)[]>;
46
62
  protected _execute: () => Promise<Array<[keyof TPartial] extends [never] ? ExtractModel<TTable> : ExtractModel<TPartial>>>;
47
63
  }
@@ -28,6 +28,8 @@ const responseMapper_1 = __importDefault(require("../../mappers/responseMapper")
28
28
  const selectWithJoin_1 = __importDefault(require("../joinBuilders/builders/selectWithJoin"));
29
29
  const join_1 = require("../joinBuilders/join");
30
30
  const abstractRequestBuilder_1 = __importDefault(require("./abstractRequestBuilder"));
31
+ const joinProxies_1 = require("./joins/proxies/joinProxies");
32
+ const selectJoinBuilder_1 = __importDefault(require("./joins/selectJoinBuilder"));
31
33
  class SelectTRB extends abstractRequestBuilder_1.default {
32
34
  constructor(session, mappedServiceToDb, props, table, logger, partial) {
33
35
  super(table, session, mappedServiceToDb, logger);
@@ -83,6 +85,33 @@ class SelectTRB extends abstractRequestBuilder_1.default {
83
85
  this.props = props;
84
86
  this.__partial = partial;
85
87
  }
88
+ join(value, callback, joinType, partial) {
89
+ const valueAsProxy = new Proxy(value, new joinProxies_1.JoinedHandler(1));
90
+ if (partial) {
91
+ for (const key of Object.keys(partial)) {
92
+ // eslint-disable-next-line no-param-reassign
93
+ partial[key] = new Proxy(partial[key], new joinProxies_1.JoinedColumn(value, 1));
94
+ }
95
+ }
96
+ const obj = this.__partial ? this.__partial
97
+ : this._table.mapServiceToDb();
98
+ const obj1 = partial ? partial
99
+ : valueAsProxy.mapServiceToDb();
100
+ const joinExpression = callback(obj, obj1);
101
+ return new selectJoinBuilder_1.default([obj, obj1], value.tableName(), 1, joinExpression, valueAsProxy, joinType, this._table, this._filter, this.props, this.__orderBy, this.__order, this.__distinct, this.__partial, this._logger);
102
+ }
103
+ innerJoin(value, callback, partial) {
104
+ return this.join(value, callback, 'INNER JOIN', partial);
105
+ }
106
+ leftJoin(value, callback, partial) {
107
+ return this.join(value, callback, 'LEFT JOIN', partial);
108
+ }
109
+ rightJoin(value, callback, partial) {
110
+ return this.join(value, callback, 'RIGHT JOIN', partial);
111
+ }
112
+ fullJoin(value, callback, partial) {
113
+ return this.join(value, callback, 'FULL JOIN', partial);
114
+ }
86
115
  orderBy(callback, order) {
87
116
  this.__orderBy = callback(this._table);
88
117
  this.__order = order;
@@ -93,7 +122,10 @@ class SelectTRB extends abstractRequestBuilder_1.default {
93
122
  // this.__groupBy = callback(this.__table);
94
123
  // return this;
95
124
  // }
96
- innerJoin(table, from, to, partial) {
125
+ /**
126
+ * @deprecated Since version 0.11.0. Will be deleted in version 0.12.0. Use {@link innerJoinV2()} instead
127
+ */
128
+ innerJoinV1(table, from, to, partial) {
97
129
  const toTable = this._table.db.create(table);
98
130
  const fromColumn = from(this._table);
99
131
  const toColumn = to(toTable);
@@ -101,7 +133,10 @@ class SelectTRB extends abstractRequestBuilder_1.default {
101
133
  .columns(fromColumn, toColumn).joinStrategy(join_1.JoinStrategy.INNER_JOIN);
102
134
  return new selectWithJoin_1.default(this._table, this._session, this._filter, join, this.props, this.__orderBy, this.__order, this.__distinct, this.__partial, partial, this._logger);
103
135
  }
104
- leftJoin(table, from, to, partial) {
136
+ /**
137
+ * @deprecated Since version 0.11.0. Will be deleted in version 0.12.0. Use {@link leftJoin()} instead
138
+ */
139
+ leftJoinV1(table, from, to, partial) {
105
140
  const toTable = this._table.db.create(table);
106
141
  const fromColumn = from(this._table);
107
142
  const toColumn = to(toTable);
@@ -109,7 +144,10 @@ class SelectTRB extends abstractRequestBuilder_1.default {
109
144
  .columns(fromColumn, toColumn).joinStrategy(join_1.JoinStrategy.LEFT_JOIN);
110
145
  return new selectWithJoin_1.default(this._table, this._session, this._filter, join, this.props, this.__orderBy, this.__order, this.__distinct, this.__partial, partial, this._logger);
111
146
  }
112
- rightJoin(table, from, to, partial) {
147
+ /**
148
+ * @deprecated Since version 0.11.0. Will be deleted in version 0.12.0. Use {@link rightJoin()} instead
149
+ */
150
+ rightJoinV1(table, from, to, partial) {
113
151
  const toTable = this._table.db.create(table);
114
152
  const fromColumn = from(this._table);
115
153
  const toColumn = to(toTable);
@@ -117,7 +155,10 @@ class SelectTRB extends abstractRequestBuilder_1.default {
117
155
  .columns(fromColumn, toColumn).joinStrategy(join_1.JoinStrategy.RIGHT_JOIN);
118
156
  return new selectWithJoin_1.default(this._table, this._session, this._filter, join, this.props, this.__orderBy, this.__order, this.__distinct, this.__partial, partial, this._logger);
119
157
  }
120
- fullJoin(table, from, to, partial) {
158
+ /**
159
+ * @deprecated Since version 0.11.0. Will be deleted in version 0.12.0. Use {@link fullJoin()} instead
160
+ */
161
+ fullJoinV1(table, from, to, partial) {
121
162
  const toTable = this._table.db.create(table);
122
163
  const fromColumn = from(this._table);
123
164
  const toColumn = to(toTable);
@@ -2,6 +2,9 @@ export { default as DeleteTRB } from './highLvlBuilders/deleteRequestBuilder';
2
2
  export { default as InsertTRB } from './highLvlBuilders/insertRequestBuilder';
3
3
  export { default as SelectTRB } from './highLvlBuilders/selectRequestBuilder';
4
4
  export { default as UpdateTRB } from './highLvlBuilders/updateRequestBuilder';
5
+ export * from './highLvlBuilders/joins/selectJoinBuilder';
6
+ export { default as JoinBuilderResponses } from './highLvlBuilders/joins/joinBuilderResponse';
7
+ export * from './highLvlBuilders/joins/proxies/joinProxies';
5
8
  export { default as to } from './joinBuilders/static';
6
9
  export { default as JoinWith } from './joinBuilders/joinWith';
7
10
  export { default as Join } from './joinBuilders/join';