@decaf-ts/for-typeorm 0.0.6

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 (109) hide show
  1. package/LICENSE.md +21 -0
  2. package/README.md +93 -0
  3. package/dist/for-typeorm.cjs +2553 -0
  4. package/dist/for-typeorm.esm.cjs +2538 -0
  5. package/lib/TypeORMAdapter.cjs +1129 -0
  6. package/lib/TypeORMAdapter.d.ts +221 -0
  7. package/lib/TypeORMDispatch.cjs +134 -0
  8. package/lib/TypeORMDispatch.d.ts +87 -0
  9. package/lib/TypeORMEventSubscriber.cjs +96 -0
  10. package/lib/TypeORMEventSubscriber.d.ts +56 -0
  11. package/lib/TypeORMRepository.cjs +209 -0
  12. package/lib/TypeORMRepository.d.ts +125 -0
  13. package/lib/constants.cjs +43 -0
  14. package/lib/constants.d.ts +39 -0
  15. package/lib/errors.cjs +28 -0
  16. package/lib/errors.d.ts +21 -0
  17. package/lib/esm/TypeORMAdapter.d.ts +221 -0
  18. package/lib/esm/TypeORMAdapter.js +1124 -0
  19. package/lib/esm/TypeORMDispatch.d.ts +87 -0
  20. package/lib/esm/TypeORMDispatch.js +130 -0
  21. package/lib/esm/TypeORMEventSubscriber.d.ts +56 -0
  22. package/lib/esm/TypeORMEventSubscriber.js +93 -0
  23. package/lib/esm/TypeORMRepository.d.ts +125 -0
  24. package/lib/esm/TypeORMRepository.js +206 -0
  25. package/lib/esm/constants.d.ts +39 -0
  26. package/lib/esm/constants.js +40 -0
  27. package/lib/esm/errors.d.ts +21 -0
  28. package/lib/esm/errors.js +24 -0
  29. package/lib/esm/index.d.ts +22 -0
  30. package/lib/esm/index.js +25 -0
  31. package/lib/esm/indexes/generator.d.ts +50 -0
  32. package/lib/esm/indexes/generator.js +95 -0
  33. package/lib/esm/indexes/index.d.ts +1 -0
  34. package/lib/esm/indexes/index.js +2 -0
  35. package/lib/esm/overrides/Column.d.ts +74 -0
  36. package/lib/esm/overrides/Column.js +70 -0
  37. package/lib/esm/overrides/CreateDateColumn.d.ts +2 -0
  38. package/lib/esm/overrides/CreateDateColumn.js +9 -0
  39. package/lib/esm/overrides/Entity.d.ts +11 -0
  40. package/lib/esm/overrides/Entity.js +28 -0
  41. package/lib/esm/overrides/PrimaryColumn.d.ts +20 -0
  42. package/lib/esm/overrides/PrimaryColumn.js +53 -0
  43. package/lib/esm/overrides/PrimaryGeneratedColumn.d.ts +24 -0
  44. package/lib/esm/overrides/PrimaryGeneratedColumn.js +51 -0
  45. package/lib/esm/overrides/UpdateDateColumn.d.ts +2 -0
  46. package/lib/esm/overrides/UpdateDateColumn.js +9 -0
  47. package/lib/esm/overrides/utils.d.ts +2 -0
  48. package/lib/esm/overrides/utils.js +29 -0
  49. package/lib/esm/query/Paginator.d.ts +86 -0
  50. package/lib/esm/query/Paginator.js +124 -0
  51. package/lib/esm/query/Statement.d.ts +131 -0
  52. package/lib/esm/query/Statement.js +242 -0
  53. package/lib/esm/query/constants.d.ts +52 -0
  54. package/lib/esm/query/constants.js +74 -0
  55. package/lib/esm/query/index.d.ts +4 -0
  56. package/lib/esm/query/index.js +5 -0
  57. package/lib/esm/query/translate.d.ts +34 -0
  58. package/lib/esm/query/translate.js +42 -0
  59. package/lib/esm/raw/postgres.d.ts +36 -0
  60. package/lib/esm/raw/postgres.js +2 -0
  61. package/lib/esm/sequences/Sequence.d.ts +67 -0
  62. package/lib/esm/sequences/Sequence.js +117 -0
  63. package/lib/esm/sequences/index.d.ts +1 -0
  64. package/lib/esm/sequences/index.js +2 -0
  65. package/lib/esm/types.d.ts +67 -0
  66. package/lib/esm/types.js +28 -0
  67. package/lib/esm/utils.d.ts +16 -0
  68. package/lib/esm/utils.js +29 -0
  69. package/lib/index.cjs +42 -0
  70. package/lib/index.d.ts +22 -0
  71. package/lib/indexes/generator.cjs +98 -0
  72. package/lib/indexes/generator.d.ts +50 -0
  73. package/lib/indexes/index.cjs +18 -0
  74. package/lib/indexes/index.d.ts +1 -0
  75. package/lib/overrides/Column.cjs +73 -0
  76. package/lib/overrides/Column.d.ts +74 -0
  77. package/lib/overrides/CreateDateColumn.cjs +12 -0
  78. package/lib/overrides/CreateDateColumn.d.ts +2 -0
  79. package/lib/overrides/Entity.cjs +31 -0
  80. package/lib/overrides/Entity.d.ts +11 -0
  81. package/lib/overrides/PrimaryColumn.cjs +56 -0
  82. package/lib/overrides/PrimaryColumn.d.ts +20 -0
  83. package/lib/overrides/PrimaryGeneratedColumn.cjs +54 -0
  84. package/lib/overrides/PrimaryGeneratedColumn.d.ts +24 -0
  85. package/lib/overrides/UpdateDateColumn.cjs +12 -0
  86. package/lib/overrides/UpdateDateColumn.d.ts +2 -0
  87. package/lib/overrides/utils.cjs +32 -0
  88. package/lib/overrides/utils.d.ts +2 -0
  89. package/lib/query/Paginator.cjs +128 -0
  90. package/lib/query/Paginator.d.ts +86 -0
  91. package/lib/query/Statement.cjs +246 -0
  92. package/lib/query/Statement.d.ts +131 -0
  93. package/lib/query/constants.cjs +77 -0
  94. package/lib/query/constants.d.ts +52 -0
  95. package/lib/query/index.cjs +21 -0
  96. package/lib/query/index.d.ts +4 -0
  97. package/lib/query/translate.cjs +45 -0
  98. package/lib/query/translate.d.ts +34 -0
  99. package/lib/raw/postgres.cjs +3 -0
  100. package/lib/raw/postgres.d.ts +36 -0
  101. package/lib/sequences/Sequence.cjs +121 -0
  102. package/lib/sequences/Sequence.d.ts +67 -0
  103. package/lib/sequences/index.cjs +18 -0
  104. package/lib/sequences/index.d.ts +1 -0
  105. package/lib/types.cjs +31 -0
  106. package/lib/types.d.ts +67 -0
  107. package/lib/utils.cjs +32 -0
  108. package/lib/utils.d.ts +16 -0
  109. package/package.json +128 -0
@@ -0,0 +1,124 @@
1
+ import { Paginator, PagingError } from "@decaf-ts/core";
2
+ import { ModelKeys } from "@decaf-ts/decorator-validation";
3
+ import { findPrimaryKey } from "@decaf-ts/db-decorators";
4
+ /**
5
+ * @description Paginator for TypeORM query results.
6
+ * @summary Implements pagination for TypeORM-built queries using take/skip for efficient navigation through result sets.
7
+ * @template M The model type that extends Model.
8
+ * @template R The result type.
9
+ * @param {TypeORMAdapter} adapter The TypeORM adapter.
10
+ * @param {TypeORMQuery} query The query container to paginate.
11
+ * @param {number} size The page size.
12
+ * @param {Constructor<M>} clazz The model constructor.
13
+ * @class TypeORMPaginator
14
+ * @example
15
+ * // Example of using TypeORMPaginator
16
+ * const paginator = new TypeORMPaginator(adapter, { query: qb }, 10, User);
17
+ * const page1 = await paginator.page(1);
18
+ * const page2 = await paginator.page(2);
19
+ */
20
+ export class TypeORMPaginator extends Paginator {
21
+ /**
22
+ * @description Gets the total number of pages
23
+ * @summary Returns the total number of pages based on the record count and page size
24
+ * @return {number} The total number of pages
25
+ */
26
+ get total() {
27
+ return this._totalPages;
28
+ }
29
+ /**
30
+ * @description Gets the total record count
31
+ * @summary Returns the total number of records matching the query
32
+ * @return {number} The total record count
33
+ */
34
+ get count() {
35
+ return this._recordCount;
36
+ }
37
+ get repo() {
38
+ if (!this.__repo) {
39
+ this.__repo = this.adapter.dataSource.getRepository(this.clazz[ModelKeys.ANCHOR]);
40
+ }
41
+ return this.__repo;
42
+ }
43
+ /**
44
+ * @description Creates a new TypeORMPaginator instance.
45
+ * @summary Initializes a paginator for TypeORM query results.
46
+ * @param {TypeORMAdapter} adapter The TypeORM adapter.
47
+ * @param {TypeORMQuery} query The TypeORM query container to paginate.
48
+ * @param {number} size The page size.
49
+ * @param {Constructor<M>} clazz The model constructor.
50
+ */
51
+ constructor(adapter, query, size, clazz) {
52
+ super(adapter, query, size, clazz);
53
+ }
54
+ /**
55
+ * @description Prepares a query for pagination
56
+ * @summary Modifies the raw query to include pagination parameters
57
+ * @param {TypeORMQuery} rawStatement - The original PostgreSQL query
58
+ * @return {TypeORMQuery} The prepared query with pagination parameters
59
+ */
60
+ prepare(rawStatement) {
61
+ const query = { ...rawStatement };
62
+ return query;
63
+ }
64
+ /**
65
+ * @description Retrieves a specific page of results.
66
+ * @summary Executes the query with pagination and processes the results.
67
+ * @param {number} [page=1] The page number to retrieve.
68
+ * @return {Promise<R[]>} A promise that resolves to an array of results.
69
+ * @throws {PagingError} If trying to access an invalid page or if no class is defined.
70
+ * @mermaid
71
+ * sequenceDiagram
72
+ * participant Client
73
+ * participant Paginator as TypeORMPaginator
74
+ * participant Adapter
75
+ * participant DB as Database
76
+ *
77
+ * Client->>Paginator: page(pageNumber)
78
+ * Note over Paginator: Prepare options (skip/take)
79
+ *
80
+ * alt First time or need count
81
+ * Paginator->>Adapter: Get count
82
+ * Adapter->>DB: Execute COUNT
83
+ * DB-->>Adapter: count
84
+ * Adapter-->>Paginator: count
85
+ * Paginator->>Paginator: Calculate total pages
86
+ * end
87
+ *
88
+ * Paginator->>Adapter: Execute query
89
+ * Adapter->>DB: findAndCount(options)
90
+ * DB-->>Adapter: rows, count
91
+ * Adapter-->>Paginator: rows, count
92
+ *
93
+ * Paginator->>Paginator: Map rows to models
94
+ * Paginator-->>Client: results
95
+ */
96
+ async page(page = 1) {
97
+ const statement = { ...this.statement };
98
+ // Get total count if not already calculated
99
+ if (!this._recordCount || !this._totalPages) {
100
+ this._totalPages = this._recordCount = 0;
101
+ }
102
+ const opts = Object.assign(statement, {
103
+ skip: (this.current || 0) * this.size,
104
+ take: this.size,
105
+ });
106
+ // this.validatePage(page);
107
+ const result = await this.repo.findAndCount(opts);
108
+ this._recordCount = result[1];
109
+ this._totalPages = Math.ceil(this._recordCount / this.size);
110
+ if (!this.clazz)
111
+ throw new PagingError("No statement target defined");
112
+ const pkDef = findPrimaryKey(new this.clazz());
113
+ const rows = result[0] || [];
114
+ const results =
115
+ // statement.columns && statement.columns.length
116
+ // ? rows // has columns means it's not full model
117
+ rows.map((row) => {
118
+ return this.adapter.revert(row, this.clazz, pkDef.id, row[pkDef.id]);
119
+ });
120
+ this._currentPage = page;
121
+ return results;
122
+ }
123
+ }
124
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Paginator.js","sourceRoot":"","sources":["../../../src/query/Paginator.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAExD,OAAO,EAAsB,SAAS,EAAE,MAAM,gCAAgC,CAAC;AAG/E,OAAO,EAAE,cAAc,EAAE,MAAM,yBAAyB,CAAC;AAEzD;;;;;;;;;;;;;;;GAeG;AACH,MAAM,OAAO,gBAAqC,SAAQ,SAIzD;IACC;;;;OAIG;IACH,IAAa,KAAK;QAChB,OAAO,IAAI,CAAC,WAAW,CAAC;IAC1B,CAAC;IAED;;;;OAIG;IACH,IAAa,KAAK;QAChB,OAAO,IAAI,CAAC,YAAY,CAAC;IAC3B,CAAC;IAID,IAAc,IAAI;QAChB,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;YACjB,IAAI,CAAC,MAAM,GAAI,IAAI,CAAC,OAA0B,CAAC,UAAU,CAAC,aAAa,CACrE,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,MAAiC,CAAC,CACxD,CAAC;QACJ,CAAC;QACD,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;IAED;;;;;;;OAOG;IACH,YACE,OAAuB,EACvB,KAAmB,EACnB,IAAY,EACZ,KAAqB;QAErB,KAAK,CAAC,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,KAAK,CAAC,CAAC;IACrC,CAAC;IAED;;;;;OAKG;IACO,OAAO,CAAC,YAA0B;QAC1C,MAAM,KAAK,GAAiB,EAAE,GAAG,YAAY,EAAE,CAAC;QAChD,OAAO,KAAK,CAAC;IACf,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;OA+BG;IAEH,KAAK,CAAC,IAAI,CAAC,OAAe,CAAC;QACzB,MAAM,SAAS,GAAG,EAAE,GAAG,IAAI,CAAC,SAAS,EAAE,CAAC;QAExC,4CAA4C;QAC5C,IAAI,CAAC,IAAI,CAAC,YAAY,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC;YAC5C,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,YAAY,GAAG,CAAC,CAAC;QAC3C,CAAC;QAED,MAAM,IAAI,GAAuB,MAAM,CAAC,MAAM,CAAC,SAAS,EAAE;YACxD,IAAI,EAAE,CAAC,IAAI,CAAC,OAAO,IAAI,CAAC,CAAC,GAAG,IAAI,CAAC,IAAI;YACrC,IAAI,EAAE,IAAI,CAAC,IAAI;SAChB,CAAC,CAAC;QAEH,2BAA2B;QAE3B,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;QAElD,IAAI,CAAC,YAAY,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;QAC9B,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC;QAE5D,IAAI,CAAC,IAAI,CAAC,KAAK;YAAE,MAAM,IAAI,WAAW,CAAC,6BAA6B,CAAC,CAAC;QAEtE,MAAM,KAAK,GAAG,cAAc,CAAC,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;QAC/C,MAAM,IAAI,GAAG,MAAM,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;QAE7B,MAAM,OAAO;QACX,gDAAgD;QAChD,oDAAoD;QACpD,IAAI,CAAC,GAAG,CAAC,CAAC,GAAQ,EAAE,EAAE;YACpB,OAAO,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,GAAG,EAAE,IAAI,CAAC,KAAK,EAAE,KAAK,CAAC,EAAE,EAAE,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC;QACvE,CAAC,CAAC,CAAC;QAEL,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;QACzB,OAAO,OAAyB,CAAC;IACnC,CAAC;CACF","sourcesContent":["import { Paginator, PagingError } from \"@decaf-ts/core\";\nimport { TypeORMQuery } from \"../types\";\nimport { Constructor, Model, ModelKeys } from \"@decaf-ts/decorator-validation\";\nimport { TypeORMAdapter } from \"../TypeORMAdapter\";\nimport { FindManyOptions, Repository as Repo } from \"typeorm\";\nimport { findPrimaryKey } from \"@decaf-ts/db-decorators\";\n\n/**\n * @description Paginator for TypeORM query results.\n * @summary Implements pagination for TypeORM-built queries using take/skip for efficient navigation through result sets.\n * @template M The model type that extends Model.\n * @template R The result type.\n * @param {TypeORMAdapter} adapter The TypeORM adapter.\n * @param {TypeORMQuery} query The query container to paginate.\n * @param {number} size The page size.\n * @param {Constructor<M>} clazz The model constructor.\n * @class TypeORMPaginator\n * @example\n * // Example of using TypeORMPaginator\n * const paginator = new TypeORMPaginator(adapter, { query: qb }, 10, User);\n * const page1 = await paginator.page(1);\n * const page2 = await paginator.page(2);\n */\nexport class TypeORMPaginator<M extends Model, R> extends Paginator<\n  M,\n  R,\n  TypeORMQuery\n> {\n  /**\n   * @description Gets the total number of pages\n   * @summary Returns the total number of pages based on the record count and page size\n   * @return {number} The total number of pages\n   */\n  override get total(): number {\n    return this._totalPages;\n  }\n\n  /**\n   * @description Gets the total record count\n   * @summary Returns the total number of records matching the query\n   * @return {number} The total record count\n   */\n  override get count(): number {\n    return this._recordCount;\n  }\n\n  private __repo?: Repo<any>;\n\n  protected get repo() {\n    if (!this.__repo) {\n      this.__repo = (this.adapter as TypeORMAdapter).dataSource.getRepository(\n        this.clazz[ModelKeys.ANCHOR as keyof typeof this.clazz]\n      );\n    }\n    return this.__repo;\n  }\n\n  /**\n   * @description Creates a new TypeORMPaginator instance.\n   * @summary Initializes a paginator for TypeORM query results.\n   * @param {TypeORMAdapter} adapter The TypeORM adapter.\n   * @param {TypeORMQuery} query The TypeORM query container to paginate.\n   * @param {number} size The page size.\n   * @param {Constructor<M>} clazz The model constructor.\n   */\n  constructor(\n    adapter: TypeORMAdapter,\n    query: TypeORMQuery,\n    size: number,\n    clazz: Constructor<M>\n  ) {\n    super(adapter, query, size, clazz);\n  }\n\n  /**\n   * @description Prepares a query for pagination\n   * @summary Modifies the raw query to include pagination parameters\n   * @param {TypeORMQuery} rawStatement - The original PostgreSQL query\n   * @return {TypeORMQuery} The prepared query with pagination parameters\n   */\n  protected prepare(rawStatement: TypeORMQuery): TypeORMQuery {\n    const query: TypeORMQuery = { ...rawStatement };\n    return query;\n  }\n\n  /**\n   * @description Retrieves a specific page of results.\n   * @summary Executes the query with pagination and processes the results.\n   * @param {number} [page=1] The page number to retrieve.\n   * @return {Promise<R[]>} A promise that resolves to an array of results.\n   * @throws {PagingError} If trying to access an invalid page or if no class is defined.\n   * @mermaid\n   * sequenceDiagram\n   *   participant Client\n   *   participant Paginator as TypeORMPaginator\n   *   participant Adapter\n   *   participant DB as Database\n   *\n   *   Client->>Paginator: page(pageNumber)\n   *   Note over Paginator: Prepare options (skip/take)\n   *\n   *   alt First time or need count\n   *     Paginator->>Adapter: Get count\n   *     Adapter->>DB: Execute COUNT\n   *     DB-->>Adapter: count\n   *     Adapter-->>Paginator: count\n   *     Paginator->>Paginator: Calculate total pages\n   *   end\n   *\n   *   Paginator->>Adapter: Execute query\n   *   Adapter->>DB: findAndCount(options)\n   *   DB-->>Adapter: rows, count\n   *   Adapter-->>Paginator: rows, count\n   *\n   *   Paginator->>Paginator: Map rows to models\n   *   Paginator-->>Client: results\n   */\n\n  async page(page: number = 1): Promise<R[]> {\n    const statement = { ...this.statement };\n\n    // Get total count if not already calculated\n    if (!this._recordCount || !this._totalPages) {\n      this._totalPages = this._recordCount = 0;\n    }\n\n    const opts: FindManyOptions<M> = Object.assign(statement, {\n      skip: (this.current || 0) * this.size,\n      take: this.size,\n    });\n\n    // this.validatePage(page);\n\n    const result = await this.repo.findAndCount(opts);\n\n    this._recordCount = result[1];\n    this._totalPages = Math.ceil(this._recordCount / this.size);\n\n    if (!this.clazz) throw new PagingError(\"No statement target defined\");\n\n    const pkDef = findPrimaryKey(new this.clazz());\n    const rows = result[0] || [];\n\n    const results =\n      // statement.columns && statement.columns.length\n      //   ? rows // has columns means it's not full model\n      rows.map((row: any) => {\n        return this.adapter.revert(row, this.clazz, pkDef.id, row[pkDef.id]);\n      });\n\n    this._currentPage = page;\n    return results as unknown as R[];\n  }\n}\n"]}
@@ -0,0 +1,131 @@
1
+ import { Condition, GroupOperator, Operator, Paginator, Statement } from "@decaf-ts/core";
2
+ import { Model } from "@decaf-ts/decorator-validation";
3
+ import { TypeORMQuery } from "../types";
4
+ import { TypeORMAdapter } from "../TypeORMAdapter";
5
+ import { SelectQueryBuilder } from "typeorm";
6
+ import { FindOptionsWhere } from "typeorm/find-options/FindOptionsWhere";
7
+ /**
8
+ * @description Statement builder for TypeORM-backed queries.
9
+ * @summary Provides a fluent interface for building SQL queries via TypeORM's SelectQueryBuilder with type safety and Decaf.ts abstractions.
10
+ * @template M The model type that extends Model.
11
+ * @template R The result type returned from execution.
12
+ * @param {TypeORMAdapter} adapter The TypeORM adapter.
13
+ * @class TypeORMStatement
14
+ * @example
15
+ * // Example using TypeORMStatement
16
+ * const statement = new TypeORMStatement<User, User[]>(adapter);
17
+ * const users = await statement
18
+ * .from(User)
19
+ * .where(Condition.attribute<User>('age').gt(18))
20
+ * .orderBy('lastName', 'asc')
21
+ * .limit(10)
22
+ * .execute();
23
+ */
24
+ export declare class TypeORMStatement<M extends Model, R> extends Statement<TypeORMQuery<M>, M, R> {
25
+ protected adapter: TypeORMAdapter;
26
+ constructor(adapter: TypeORMAdapter);
27
+ /**
28
+ * @description Builds a TypeORM SelectQueryBuilder from the statement.
29
+ * @summary Converts the statement's conditions, selectors, and options into a TypeORM-backed query object.
30
+ * @return {TypeORMQuery} The built TypeORM query container.
31
+ * @throws {Error} If there are invalid query conditions.
32
+ * @mermaid
33
+ * sequenceDiagram
34
+ * participant Statement
35
+ * participant Repository
36
+ * participant parseCondition
37
+ *
38
+ * Statement->>Statement: build()
39
+ * Note over Statement: Initialize query
40
+ * Statement->>Repository: Get table name
41
+ * Repository-->>Statement: Return table name
42
+ * Statement->>Statement: Create base query
43
+ *
44
+ * alt Has selectSelector
45
+ * Statement->>Statement: Add columns to query
46
+ * end
47
+ *
48
+ * alt Has whereCondition
49
+ * Statement->>Statement: Create combined condition with table
50
+ * Statement->>parseCondition: Parse condition
51
+ * parseCondition-->>Statement: Return parsed conditions
52
+ * Statement->>Statement: Add conditions to query
53
+ * end
54
+ *
55
+ * alt Has orderBySelector
56
+ * Statement->>Statement: Add orderBy to query
57
+ * end
58
+ *
59
+ * alt Has limitSelector
60
+ * Statement->>Statement: Set limit
61
+ * else
62
+ * Statement->>Statement: Use default limit
63
+ * end
64
+ *
65
+ * alt Has offsetSelector
66
+ * Statement->>Statement: Set offset
67
+ * end
68
+ *
69
+ * Statement-->>Statement: Return query
70
+ */
71
+ protected build(): TypeORMQuery<M>;
72
+ /**
73
+ * @description Creates a paginator for the statement.
74
+ * @summary Builds the query and returns a TypeORMPaginator for paginated results.
75
+ * @template R The result type.
76
+ * @param {number} size The page size.
77
+ * @return {Promise<Paginator<M, R, TypeORMQuery>>} A promise that resolves to a paginator.
78
+ * @throws {InternalError} If there's an error building the query.
79
+ */
80
+ paginate<R>(size: number): Promise<Paginator<M, R, TypeORMQuery>>;
81
+ /**
82
+ * @description Processes a record.
83
+ * @summary Converts a raw result row to a model instance using the adapter.
84
+ * @param {any} r The raw record.
85
+ * @param {keyof M} pkAttr The primary key attribute of the model.
86
+ * @param {"Number" | "BigInt" | undefined} sequenceType The type of the sequence.
87
+ * @return {any} The processed record.
88
+ */
89
+ private processRecord;
90
+ /**
91
+ * @description Executes a raw TypeORM query builder.
92
+ * @summary Sends the built SelectQueryBuilder to the database via TypeORM and returns the results.
93
+ * @template R The result type.
94
+ * @param {TypeORMQuery} rawInput The query container to execute.
95
+ * @return {Promise<R>} A promise that resolves to the query results.
96
+ */
97
+ raw<R>(rawInput: TypeORMQuery<M>): Promise<R>;
98
+ protected parseConditionForPagination(condition: Condition<M>, tableName: string, counter?: number, conditionalOp?: GroupOperator | Operator): FindOptionsWhere<M>[] | FindOptionsWhere<M>;
99
+ /**
100
+ * @description Parses a condition into PostgreSQL conditions
101
+ * @summary Converts a Condition object into PostgreSQL condition structures
102
+ * @param {Condition<M>} condition - The condition to parse
103
+ * @param {string} [tableName] - the positional index of the arguments
104
+ * @return {TypeORMQuery} The PostgresSQL condition
105
+ * @mermaid
106
+ * sequenceDiagram
107
+ * participant Statement
108
+ * participant translateOperators
109
+ * participant parseCondition
110
+ *
111
+ * Statement->>Statement: parseCondition(condition)
112
+ *
113
+ * Note over Statement: Extract condition parts
114
+ *
115
+ * alt Simple comparison operator
116
+ * Statement->>translateOperators: translateOperators(operator)
117
+ * translateOperators-->>Statement: Return PostgreSQL operator
118
+ * Statement->>Statement: Create condition with column, operator, and value
119
+ * else NOT operator
120
+ * Statement->>Statement: parseCondition(attr1)
121
+ * Statement->>Statement: Add NOT to conditions
122
+ * else AND/OR operator
123
+ * Statement->>Statement: parseCondition(attr1)
124
+ * Statement->>Statement: parseCondition(comparison)
125
+ * Statement->>Statement: Combine conditions with AND/OR
126
+ * end
127
+ *
128
+ * Statement-->>Statement: Return conditions array
129
+ */
130
+ protected parseCondition(condition: Condition<M>, tableName: string, qb: SelectQueryBuilder<any>, counter?: number, conditionalOp?: GroupOperator | Operator): TypeORMQuery<M>;
131
+ }
@@ -0,0 +1,242 @@
1
+ import { GroupOperator, Operator, OrderDirection, Repository, Statement, } from "@decaf-ts/core";
2
+ import { ModelKeys } from "@decaf-ts/decorator-validation";
3
+ import { translateOperators } from "./translate.js";
4
+ import { TypeORMQueryLimit } from "./constants.js";
5
+ import { TypeORMPaginator } from "./Paginator.js";
6
+ import { findPrimaryKey, InternalError } from "@decaf-ts/db-decorators";
7
+ /**
8
+ * @description Statement builder for TypeORM-backed queries.
9
+ * @summary Provides a fluent interface for building SQL queries via TypeORM's SelectQueryBuilder with type safety and Decaf.ts abstractions.
10
+ * @template M The model type that extends Model.
11
+ * @template R The result type returned from execution.
12
+ * @param {TypeORMAdapter} adapter The TypeORM adapter.
13
+ * @class TypeORMStatement
14
+ * @example
15
+ * // Example using TypeORMStatement
16
+ * const statement = new TypeORMStatement<User, User[]>(adapter);
17
+ * const users = await statement
18
+ * .from(User)
19
+ * .where(Condition.attribute<User>('age').gt(18))
20
+ * .orderBy('lastName', 'asc')
21
+ * .limit(10)
22
+ * .execute();
23
+ */
24
+ export class TypeORMStatement extends Statement {
25
+ constructor(adapter) {
26
+ super(adapter);
27
+ }
28
+ /**
29
+ * @description Builds a TypeORM SelectQueryBuilder from the statement.
30
+ * @summary Converts the statement's conditions, selectors, and options into a TypeORM-backed query object.
31
+ * @return {TypeORMQuery} The built TypeORM query container.
32
+ * @throws {Error} If there are invalid query conditions.
33
+ * @mermaid
34
+ * sequenceDiagram
35
+ * participant Statement
36
+ * participant Repository
37
+ * participant parseCondition
38
+ *
39
+ * Statement->>Statement: build()
40
+ * Note over Statement: Initialize query
41
+ * Statement->>Repository: Get table name
42
+ * Repository-->>Statement: Return table name
43
+ * Statement->>Statement: Create base query
44
+ *
45
+ * alt Has selectSelector
46
+ * Statement->>Statement: Add columns to query
47
+ * end
48
+ *
49
+ * alt Has whereCondition
50
+ * Statement->>Statement: Create combined condition with table
51
+ * Statement->>parseCondition: Parse condition
52
+ * parseCondition-->>Statement: Return parsed conditions
53
+ * Statement->>Statement: Add conditions to query
54
+ * end
55
+ *
56
+ * alt Has orderBySelector
57
+ * Statement->>Statement: Add orderBy to query
58
+ * end
59
+ *
60
+ * alt Has limitSelector
61
+ * Statement->>Statement: Set limit
62
+ * else
63
+ * Statement->>Statement: Use default limit
64
+ * end
65
+ *
66
+ * alt Has offsetSelector
67
+ * Statement->>Statement: Set offset
68
+ * end
69
+ *
70
+ * Statement-->>Statement: Return query
71
+ */
72
+ build() {
73
+ const log = this.log.for(this.build);
74
+ const tableName = Repository.table(this.fromSelector);
75
+ const m = new this.fromSelector();
76
+ const q = {
77
+ query: this.adapter.dataSource
78
+ .getRepository(this.fromSelector[ModelKeys.ANCHOR])
79
+ .createQueryBuilder(tableName),
80
+ };
81
+ if (this.selectSelector)
82
+ q.query = q.query.select(this.selectSelector.map((s) => `${tableName}.${s}`));
83
+ else
84
+ q.query = q.query.select();
85
+ //
86
+ // q.query = (q.query as SelectQueryBuilder<any>).from(
87
+ // this.fromSelector[ModelKeys.ANCHOR as keyof typeof this.fromSelector],
88
+ // tableName
89
+ // );
90
+ if (this.whereCondition)
91
+ q.query = this.parseCondition(this.whereCondition, tableName, q.query).query;
92
+ let orderByArgs;
93
+ if (!this.orderBySelector)
94
+ orderByArgs = [
95
+ `${tableName}.${findPrimaryKey(m).id}`,
96
+ OrderDirection.ASC.toUpperCase(),
97
+ ];
98
+ else
99
+ orderByArgs = [
100
+ `${tableName}.${this.orderBySelector[0]}`,
101
+ this.orderBySelector[1].toUpperCase(),
102
+ ];
103
+ q.query = q.query.orderBy(...orderByArgs);
104
+ if (this.limitSelector) {
105
+ q.query = q.query.limit(this.limitSelector);
106
+ }
107
+ else {
108
+ log.debug(`No limit selector defined. Using default limit of ${TypeORMQueryLimit}`);
109
+ q.query = q.query.limit(TypeORMQueryLimit);
110
+ }
111
+ // Add offset
112
+ if (this.offsetSelector)
113
+ q.query = q.query.skip(this.offsetSelector);
114
+ return q;
115
+ }
116
+ /**
117
+ * @description Creates a paginator for the statement.
118
+ * @summary Builds the query and returns a TypeORMPaginator for paginated results.
119
+ * @template R The result type.
120
+ * @param {number} size The page size.
121
+ * @return {Promise<Paginator<M, R, TypeORMQuery>>} A promise that resolves to a paginator.
122
+ * @throws {InternalError} If there's an error building the query.
123
+ */
124
+ async paginate(size) {
125
+ try {
126
+ const query = this.build();
127
+ const transformedQuery = {};
128
+ const a = query.query;
129
+ if (this.whereCondition)
130
+ transformedQuery.where = this.parseConditionForPagination(this.whereCondition, Repository.table(this.fromSelector));
131
+ if (this.orderBySelector)
132
+ transformedQuery.order = {
133
+ [this.orderBySelector[0]]: this.orderBySelector[1].toString(),
134
+ };
135
+ return new TypeORMPaginator(this.adapter, transformedQuery, size, this.fromSelector);
136
+ }
137
+ catch (e) {
138
+ throw new InternalError(e);
139
+ }
140
+ }
141
+ /**
142
+ * @description Processes a record.
143
+ * @summary Converts a raw result row to a model instance using the adapter.
144
+ * @param {any} r The raw record.
145
+ * @param {keyof M} pkAttr The primary key attribute of the model.
146
+ * @param {"Number" | "BigInt" | undefined} sequenceType The type of the sequence.
147
+ * @return {any} The processed record.
148
+ */
149
+ processRecord(r, pkAttr) {
150
+ if (typeof r[pkAttr] !== "undefined") {
151
+ return this.adapter.revert(r, this.fromSelector, pkAttr, r[pkAttr]);
152
+ }
153
+ return r;
154
+ }
155
+ /**
156
+ * @description Executes a raw TypeORM query builder.
157
+ * @summary Sends the built SelectQueryBuilder to the database via TypeORM and returns the results.
158
+ * @template R The result type.
159
+ * @param {TypeORMQuery} rawInput The query container to execute.
160
+ * @return {Promise<R>} A promise that resolves to the query results.
161
+ */
162
+ async raw(rawInput) {
163
+ const log = this.log.for(this.raw);
164
+ log.debug(`Executing raw query: ${rawInput.query.getSql()}`);
165
+ return (await rawInput.query.getMany());
166
+ }
167
+ parseConditionForPagination(condition, tableName, counter = 0, conditionalOp) {
168
+ throw new InternalError("Not implemented");
169
+ }
170
+ /**
171
+ * @description Parses a condition into PostgreSQL conditions
172
+ * @summary Converts a Condition object into PostgreSQL condition structures
173
+ * @param {Condition<M>} condition - The condition to parse
174
+ * @param {string} [tableName] - the positional index of the arguments
175
+ * @return {TypeORMQuery} The PostgresSQL condition
176
+ * @mermaid
177
+ * sequenceDiagram
178
+ * participant Statement
179
+ * participant translateOperators
180
+ * participant parseCondition
181
+ *
182
+ * Statement->>Statement: parseCondition(condition)
183
+ *
184
+ * Note over Statement: Extract condition parts
185
+ *
186
+ * alt Simple comparison operator
187
+ * Statement->>translateOperators: translateOperators(operator)
188
+ * translateOperators-->>Statement: Return PostgreSQL operator
189
+ * Statement->>Statement: Create condition with column, operator, and value
190
+ * else NOT operator
191
+ * Statement->>Statement: parseCondition(attr1)
192
+ * Statement->>Statement: Add NOT to conditions
193
+ * else AND/OR operator
194
+ * Statement->>Statement: parseCondition(attr1)
195
+ * Statement->>Statement: parseCondition(comparison)
196
+ * Statement->>Statement: Combine conditions with AND/OR
197
+ * end
198
+ *
199
+ * Statement-->>Statement: Return conditions array
200
+ */
201
+ parseCondition(condition, tableName, qb, counter = 0, conditionalOp) {
202
+ const { attr1, operator, comparison } = condition;
203
+ function parse() {
204
+ const sqlOperator = translateOperators(operator);
205
+ const attrRef = `${attr1}${counter}`;
206
+ const queryStr = `${tableName}.${attr1} ${sqlOperator} :${attrRef}`;
207
+ const values = {
208
+ [attrRef]: comparison,
209
+ };
210
+ switch (conditionalOp) {
211
+ case GroupOperator.AND:
212
+ return {
213
+ query: qb.andWhere(queryStr, values),
214
+ };
215
+ case GroupOperator.OR:
216
+ return {
217
+ query: qb.orWhere(queryStr, values),
218
+ };
219
+ case Operator.NOT:
220
+ throw new Error("NOT operator not implemented");
221
+ default:
222
+ return {
223
+ query: qb.where(queryStr, values),
224
+ };
225
+ }
226
+ }
227
+ if ([GroupOperator.AND, GroupOperator.OR, Operator.NOT].indexOf(operator) === -1) {
228
+ return parse();
229
+ }
230
+ // For NOT operator
231
+ else if (operator === Operator.NOT) {
232
+ throw new Error("NOT operator not implemented");
233
+ }
234
+ // For AND/OR operators
235
+ else {
236
+ qb = this.parseCondition(attr1, tableName, qb, ++counter)
237
+ .query;
238
+ return this.parseCondition(comparison, tableName, qb, ++counter, operator);
239
+ }
240
+ }
241
+ }
242
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Statement.js","sourceRoot":"","sources":["../../../src/query/Statement.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Condition,\n  GroupOperator,\n  Operator,\n  OrderDirection,\n  Paginator,\n  Repository,\n  Statement,\n} from \"@decaf-ts/core\";\nimport { Model, ModelKeys } from \"@decaf-ts/decorator-validation\";\nimport { translateOperators } from \"./translate\";\nimport { TypeORMQueryLimit } from \"./constants\";\nimport { TypeORMPaginator } from \"./Paginator\";\nimport { findPrimaryKey, InternalError } from \"@decaf-ts/db-decorators\";\nimport { TypeORMQuery } from \"../types\";\nimport { TypeORMAdapter } from \"../TypeORMAdapter\";\nimport { FindManyOptions, SelectQueryBuilder } from \"typeorm\";\nimport { FindOptionsWhere } from \"typeorm/find-options/FindOptionsWhere\";\nimport { FindOptionsOrder } from \"typeorm/find-options/FindOptionsOrder\";\n\n/**\n * @description Statement builder for TypeORM-backed queries.\n * @summary Provides a fluent interface for building SQL queries via TypeORM's SelectQueryBuilder with type safety and Decaf.ts abstractions.\n * @template M The model type that extends Model.\n * @template R The result type returned from execution.\n * @param {TypeORMAdapter} adapter The TypeORM adapter.\n * @class TypeORMStatement\n * @example\n * // Example using TypeORMStatement\n * const statement = new TypeORMStatement<User, User[]>(adapter);\n * const users = await statement\n *   .from(User)\n *   .where(Condition.attribute<User>('age').gt(18))\n *   .orderBy('lastName', 'asc')\n *   .limit(10)\n *   .execute();\n */\nexport class TypeORMStatement<M extends Model, R> extends Statement<\n  TypeORMQuery<M>,\n  M,\n  R\n> {\n  protected override adapter!: TypeORMAdapter;\n\n  constructor(adapter: TypeORMAdapter) {\n    super(adapter);\n  }\n\n  /**\n   * @description Builds a TypeORM SelectQueryBuilder from the statement.\n   * @summary Converts the statement's conditions, selectors, and options into a TypeORM-backed query object.\n   * @return {TypeORMQuery} The built TypeORM query container.\n   * @throws {Error} If there are invalid query conditions.\n   * @mermaid\n   * sequenceDiagram\n   *   participant Statement\n   *   participant Repository\n   *   participant parseCondition\n   *\n   *   Statement->>Statement: build()\n   *   Note over Statement: Initialize query\n   *   Statement->>Repository: Get table name\n   *   Repository-->>Statement: Return table name\n   *   Statement->>Statement: Create base query\n   *\n   *   alt Has selectSelector\n   *     Statement->>Statement: Add columns to query\n   *   end\n   *\n   *   alt Has whereCondition\n   *     Statement->>Statement: Create combined condition with table\n   *     Statement->>parseCondition: Parse condition\n   *     parseCondition-->>Statement: Return parsed conditions\n   *     Statement->>Statement: Add conditions to query\n   *   end\n   *\n   *   alt Has orderBySelector\n   *     Statement->>Statement: Add orderBy to query\n   *   end\n   *\n   *   alt Has limitSelector\n   *     Statement->>Statement: Set limit\n   *   else\n   *     Statement->>Statement: Use default limit\n   *   end\n   *\n   *   alt Has offsetSelector\n   *     Statement->>Statement: Set offset\n   *   end\n   *\n   *   Statement-->>Statement: Return query\n   */\n  protected build(): TypeORMQuery<M> {\n    const log = this.log.for(this.build);\n    const tableName = Repository.table(this.fromSelector);\n    const m = new this.fromSelector();\n\n    const q: TypeORMQuery<M, SelectQueryBuilder<M>> = {\n      query: this.adapter.dataSource\n        .getRepository(\n          this.fromSelector[ModelKeys.ANCHOR as keyof typeof this.fromSelector]\n        )\n        .createQueryBuilder(tableName) as SelectQueryBuilder<M>,\n    };\n\n    if (this.selectSelector)\n      q.query = q.query.select(\n        this.selectSelector.map((s) => `${tableName}.${s as string}`)\n      );\n    else q.query = q.query.select();\n    //\n    // q.query = (q.query as SelectQueryBuilder<any>).from(\n    //   this.fromSelector[ModelKeys.ANCHOR as keyof typeof this.fromSelector],\n    //   tableName\n    // );\n\n    if (this.whereCondition)\n      q.query = this.parseCondition(\n        this.whereCondition,\n        tableName,\n        q.query as SelectQueryBuilder<any>\n      ).query as unknown as SelectQueryBuilder<M>;\n\n    let orderByArgs: [string, \"DESC\" | \"ASC\"];\n    if (!this.orderBySelector)\n      orderByArgs = [\n        `${tableName}.${findPrimaryKey(m).id as string}`,\n        OrderDirection.ASC.toUpperCase() as \"ASC\",\n      ];\n    else\n      orderByArgs = [\n        `${tableName}.${this.orderBySelector[0] as string}`,\n        this.orderBySelector[1].toUpperCase() as \"DESC\" | \"ASC\",\n      ];\n\n    q.query = (q.query as SelectQueryBuilder<any>).orderBy(...orderByArgs);\n    if (this.limitSelector) {\n      q.query = (q.query as SelectQueryBuilder<any>).limit(this.limitSelector);\n    } else {\n      log.debug(\n        `No limit selector defined. Using default limit of ${TypeORMQueryLimit}`\n      );\n      q.query = (q.query as SelectQueryBuilder<any>).limit(TypeORMQueryLimit);\n    }\n\n    // Add offset\n    if (this.offsetSelector)\n      q.query = (q.query as SelectQueryBuilder<any>).skip(this.offsetSelector);\n\n    return q as any;\n  }\n\n  /**\n   * @description Creates a paginator for the statement.\n   * @summary Builds the query and returns a TypeORMPaginator for paginated results.\n   * @template R The result type.\n   * @param {number} size The page size.\n   * @return {Promise<Paginator<M, R, TypeORMQuery>>} A promise that resolves to a paginator.\n   * @throws {InternalError} If there's an error building the query.\n   */\n  async paginate<R>(size: number): Promise<Paginator<M, R, TypeORMQuery>> {\n    try {\n      const query: TypeORMQuery = this.build();\n      const transformedQuery: FindManyOptions<M> = {};\n      const a = query.query as unknown as SelectQueryBuilder<M>;\n      if (this.whereCondition)\n        transformedQuery.where = this.parseConditionForPagination(\n          this.whereCondition,\n          Repository.table(this.fromSelector)\n        );\n\n      if (this.orderBySelector)\n        transformedQuery.order = {\n          [this.orderBySelector[0]]: this.orderBySelector[1].toString(),\n        } as any;\n\n      return new TypeORMPaginator(\n        this.adapter as any,\n        transformedQuery as any,\n        size,\n        this.fromSelector\n      );\n    } catch (e: any) {\n      throw new InternalError(e);\n    }\n  }\n\n  /**\n   * @description Processes a record.\n   * @summary Converts a raw result row to a model instance using the adapter.\n   * @param {any} r The raw record.\n   * @param {keyof M} pkAttr The primary key attribute of the model.\n   * @param {\"Number\" | \"BigInt\" | undefined} sequenceType The type of the sequence.\n   * @return {any} The processed record.\n   */\n  private processRecord(r: any, pkAttr: keyof M) {\n    if (typeof r[pkAttr] !== \"undefined\") {\n      return this.adapter.revert(r, this.fromSelector, pkAttr, r[pkAttr]);\n    }\n    return r;\n  }\n\n  /**\n   * @description Executes a raw TypeORM query builder.\n   * @summary Sends the built SelectQueryBuilder to the database via TypeORM and returns the results.\n   * @template R The result type.\n   * @param {TypeORMQuery} rawInput The query container to execute.\n   * @return {Promise<R>} A promise that resolves to the query results.\n   */\n  override async raw<R>(rawInput: TypeORMQuery<M>): Promise<R> {\n    const log = this.log.for(this.raw);\n    log.debug(\n      `Executing raw query: ${(rawInput.query as unknown as SelectQueryBuilder<M>).getSql()}`\n    );\n    return (await (\n      rawInput.query as unknown as SelectQueryBuilder<M>\n    ).getMany()) as R;\n  }\n\n  protected parseConditionForPagination(\n    condition: Condition<M>,\n    tableName: string,\n    counter = 0,\n    conditionalOp?: GroupOperator | Operator\n  ): FindOptionsWhere<M>[] | FindOptionsWhere<M> {\n    throw new InternalError(\"Not implemented\");\n  }\n\n  /**\n   * @description Parses a condition into PostgreSQL conditions\n   * @summary Converts a Condition object into PostgreSQL condition structures\n   * @param {Condition<M>} condition - The condition to parse\n   * @param {string} [tableName] - the positional index of the arguments\n   * @return {TypeORMQuery} The PostgresSQL condition\n   * @mermaid\n   * sequenceDiagram\n   *   participant Statement\n   *   participant translateOperators\n   *   participant parseCondition\n   *\n   *   Statement->>Statement: parseCondition(condition)\n   *\n   *   Note over Statement: Extract condition parts\n   *\n   *   alt Simple comparison operator\n   *     Statement->>translateOperators: translateOperators(operator)\n   *     translateOperators-->>Statement: Return PostgreSQL operator\n   *     Statement->>Statement: Create condition with column, operator, and value\n   *   else NOT operator\n   *     Statement->>Statement: parseCondition(attr1)\n   *     Statement->>Statement: Add NOT to conditions\n   *   else AND/OR operator\n   *     Statement->>Statement: parseCondition(attr1)\n   *     Statement->>Statement: parseCondition(comparison)\n   *     Statement->>Statement: Combine conditions with AND/OR\n   *   end\n   *\n   *   Statement-->>Statement: Return conditions array\n   */\n  protected parseCondition(\n    condition: Condition<M>,\n    tableName: string,\n    qb: SelectQueryBuilder<any>,\n    counter = 0,\n    conditionalOp?: GroupOperator | Operator\n  ): TypeORMQuery<M> {\n    const { attr1, operator, comparison } = condition as unknown as {\n      attr1: string | Condition<M>;\n      operator: Operator | GroupOperator;\n      comparison: any;\n    };\n\n    function parse(): TypeORMQuery<M> {\n      const sqlOperator = translateOperators(operator);\n      const attrRef = `${attr1}${counter}`;\n      const queryStr = `${tableName}.${attr1} ${sqlOperator} :${attrRef}`;\n      const values = {\n        [attrRef]: comparison,\n      };\n      switch (conditionalOp) {\n        case GroupOperator.AND:\n          return {\n            query: qb.andWhere(queryStr, values) as any,\n          };\n        case GroupOperator.OR:\n          return {\n            query: qb.orWhere(queryStr, values) as any,\n          };\n        case Operator.NOT:\n          throw new Error(\"NOT operator not implemented\");\n        default:\n          return {\n            query: qb.where(queryStr, values) as any,\n          };\n      }\n    }\n\n    if (\n      [GroupOperator.AND, GroupOperator.OR, Operator.NOT].indexOf(\n        operator as GroupOperator\n      ) === -1\n    ) {\n      return parse();\n    }\n    // For NOT operator\n    else if (operator === Operator.NOT) {\n      throw new Error(\"NOT operator not implemented\");\n    }\n    // For AND/OR operators\n    else {\n      qb = this.parseCondition(attr1 as Condition<M>, tableName, qb, ++counter)\n        .query as unknown as SelectQueryBuilder<M>;\n      return this.parseCondition(\n        comparison,\n        tableName,\n        qb,\n        ++counter,\n        operator\n      );\n    }\n  }\n}\n"]}
@@ -0,0 +1,52 @@
1
+ import { SQLOperator } from "../types";
2
+ /**
3
+ * @description Default query limit for TypeORM-backed queries.
4
+ * @summary Maximum number of records to return in a single page when paginating results.
5
+ * @const TypeORMQueryLimit
6
+ * @memberOf module:for-typeorm
7
+ */
8
+ export declare const TypeORMQueryLimit = 250;
9
+ /**
10
+ * @description Mapping of operator names to SQL operators.
11
+ * @summary Constants for comparison operators used when translating high-level filters into SQL via TypeORM.
12
+ * @typedef {Object} PostgreSQLOperatorType
13
+ * @property {string} EQUAL Equality operator (=)
14
+ * @property {string} DIFFERENT Inequality operator (<>)
15
+ * @property {string} BIGGER Greater than operator (>)
16
+ * @property {string} BIGGER_EQ Greater than or equal operator (>=)
17
+ * @property {string} SMALLER Less than operator (<)
18
+ * @property {string} SMALLER_EQ Less than or equal operator (<=)
19
+ * @property {string} NOT Negation operator (NOT)
20
+ * @property {string} IN In array operator (IN)
21
+ * @property {string} REGEXP Regular expression operator (~)
22
+ * @property {string} IREGEXP Case-insensitive regular expression operator (~*)
23
+ * @property {string} LIKE Pattern matching operator (LIKE)
24
+ * @property {string} ILIKE Case-insensitive pattern matching operator (ILIKE)
25
+ * @property {string} BETWEEN Range operator (BETWEEN)
26
+ * @property {string} IS_NULL NULL check operator (IS NULL)
27
+ * @property {string} IS_NOT_NULL NOT NULL check operator (IS NOT NULL)
28
+ * @const TypeORMOperator
29
+ * @type {PostgreSQLOperatorType}
30
+ * @memberOf module:for-typeorm
31
+ */
32
+ export declare const TypeORMOperator: Record<string, SQLOperator | string>;
33
+ /**
34
+ * @description Mapping of logical operator names to SQL operators.
35
+ * @summary Constants for logical operators used when building WHERE clause groups in TypeORM queries.
36
+ * @typedef {Object} PostgreSQLGroupOperatorType
37
+ * @property {string} AND Logical AND operator (AND)
38
+ * @property {string} OR Logical OR operator (OR)
39
+ * @const TypeORMGroupOperator
40
+ * @type {PostgreSQLGroupOperatorType}
41
+ * @memberOf module:for-typeorm
42
+ */
43
+ export declare const TypeORMGroupOperator: Record<string, string>;
44
+ /**
45
+ * @description Special constant values used in queries.
46
+ * @summary String constants representing special values used while composing SQL with TypeORM.
47
+ * @typedef {Object} PostgreSQLConstType
48
+ * @property {string} NULL String representation of null value.
49
+ * @const TypeORMConst
50
+ * @memberOf module:for-typeorm
51
+ */
52
+ export declare const TypeORMConst: Record<string, string>;