@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.
- package/LICENSE.md +21 -0
- package/README.md +93 -0
- package/dist/for-typeorm.cjs +2553 -0
- package/dist/for-typeorm.esm.cjs +2538 -0
- package/lib/TypeORMAdapter.cjs +1129 -0
- package/lib/TypeORMAdapter.d.ts +221 -0
- package/lib/TypeORMDispatch.cjs +134 -0
- package/lib/TypeORMDispatch.d.ts +87 -0
- package/lib/TypeORMEventSubscriber.cjs +96 -0
- package/lib/TypeORMEventSubscriber.d.ts +56 -0
- package/lib/TypeORMRepository.cjs +209 -0
- package/lib/TypeORMRepository.d.ts +125 -0
- package/lib/constants.cjs +43 -0
- package/lib/constants.d.ts +39 -0
- package/lib/errors.cjs +28 -0
- package/lib/errors.d.ts +21 -0
- package/lib/esm/TypeORMAdapter.d.ts +221 -0
- package/lib/esm/TypeORMAdapter.js +1124 -0
- package/lib/esm/TypeORMDispatch.d.ts +87 -0
- package/lib/esm/TypeORMDispatch.js +130 -0
- package/lib/esm/TypeORMEventSubscriber.d.ts +56 -0
- package/lib/esm/TypeORMEventSubscriber.js +93 -0
- package/lib/esm/TypeORMRepository.d.ts +125 -0
- package/lib/esm/TypeORMRepository.js +206 -0
- package/lib/esm/constants.d.ts +39 -0
- package/lib/esm/constants.js +40 -0
- package/lib/esm/errors.d.ts +21 -0
- package/lib/esm/errors.js +24 -0
- package/lib/esm/index.d.ts +22 -0
- package/lib/esm/index.js +25 -0
- package/lib/esm/indexes/generator.d.ts +50 -0
- package/lib/esm/indexes/generator.js +95 -0
- package/lib/esm/indexes/index.d.ts +1 -0
- package/lib/esm/indexes/index.js +2 -0
- package/lib/esm/overrides/Column.d.ts +74 -0
- package/lib/esm/overrides/Column.js +70 -0
- package/lib/esm/overrides/CreateDateColumn.d.ts +2 -0
- package/lib/esm/overrides/CreateDateColumn.js +9 -0
- package/lib/esm/overrides/Entity.d.ts +11 -0
- package/lib/esm/overrides/Entity.js +28 -0
- package/lib/esm/overrides/PrimaryColumn.d.ts +20 -0
- package/lib/esm/overrides/PrimaryColumn.js +53 -0
- package/lib/esm/overrides/PrimaryGeneratedColumn.d.ts +24 -0
- package/lib/esm/overrides/PrimaryGeneratedColumn.js +51 -0
- package/lib/esm/overrides/UpdateDateColumn.d.ts +2 -0
- package/lib/esm/overrides/UpdateDateColumn.js +9 -0
- package/lib/esm/overrides/utils.d.ts +2 -0
- package/lib/esm/overrides/utils.js +29 -0
- package/lib/esm/query/Paginator.d.ts +86 -0
- package/lib/esm/query/Paginator.js +124 -0
- package/lib/esm/query/Statement.d.ts +131 -0
- package/lib/esm/query/Statement.js +242 -0
- package/lib/esm/query/constants.d.ts +52 -0
- package/lib/esm/query/constants.js +74 -0
- package/lib/esm/query/index.d.ts +4 -0
- package/lib/esm/query/index.js +5 -0
- package/lib/esm/query/translate.d.ts +34 -0
- package/lib/esm/query/translate.js +42 -0
- package/lib/esm/raw/postgres.d.ts +36 -0
- package/lib/esm/raw/postgres.js +2 -0
- package/lib/esm/sequences/Sequence.d.ts +67 -0
- package/lib/esm/sequences/Sequence.js +117 -0
- package/lib/esm/sequences/index.d.ts +1 -0
- package/lib/esm/sequences/index.js +2 -0
- package/lib/esm/types.d.ts +67 -0
- package/lib/esm/types.js +28 -0
- package/lib/esm/utils.d.ts +16 -0
- package/lib/esm/utils.js +29 -0
- package/lib/index.cjs +42 -0
- package/lib/index.d.ts +22 -0
- package/lib/indexes/generator.cjs +98 -0
- package/lib/indexes/generator.d.ts +50 -0
- package/lib/indexes/index.cjs +18 -0
- package/lib/indexes/index.d.ts +1 -0
- package/lib/overrides/Column.cjs +73 -0
- package/lib/overrides/Column.d.ts +74 -0
- package/lib/overrides/CreateDateColumn.cjs +12 -0
- package/lib/overrides/CreateDateColumn.d.ts +2 -0
- package/lib/overrides/Entity.cjs +31 -0
- package/lib/overrides/Entity.d.ts +11 -0
- package/lib/overrides/PrimaryColumn.cjs +56 -0
- package/lib/overrides/PrimaryColumn.d.ts +20 -0
- package/lib/overrides/PrimaryGeneratedColumn.cjs +54 -0
- package/lib/overrides/PrimaryGeneratedColumn.d.ts +24 -0
- package/lib/overrides/UpdateDateColumn.cjs +12 -0
- package/lib/overrides/UpdateDateColumn.d.ts +2 -0
- package/lib/overrides/utils.cjs +32 -0
- package/lib/overrides/utils.d.ts +2 -0
- package/lib/query/Paginator.cjs +128 -0
- package/lib/query/Paginator.d.ts +86 -0
- package/lib/query/Statement.cjs +246 -0
- package/lib/query/Statement.d.ts +131 -0
- package/lib/query/constants.cjs +77 -0
- package/lib/query/constants.d.ts +52 -0
- package/lib/query/index.cjs +21 -0
- package/lib/query/index.d.ts +4 -0
- package/lib/query/translate.cjs +45 -0
- package/lib/query/translate.d.ts +34 -0
- package/lib/raw/postgres.cjs +3 -0
- package/lib/raw/postgres.d.ts +36 -0
- package/lib/sequences/Sequence.cjs +121 -0
- package/lib/sequences/Sequence.d.ts +67 -0
- package/lib/sequences/index.cjs +18 -0
- package/lib/sequences/index.d.ts +1 -0
- package/lib/types.cjs +31 -0
- package/lib/types.d.ts +67 -0
- package/lib/utils.cjs +32 -0
- package/lib/utils.d.ts +16 -0
- 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":"AAAA,OAAO,EAEL,aAAa,EACb,QAAQ,EACR,cAAc,EAEd,UAAU,EACV,SAAS,GACV,MAAM,gBAAgB,CAAC;AACxB,OAAO,EAAS,SAAS,EAAE,MAAM,gCAAgC,CAAC;AAClE,OAAO,EAAE,kBAAkB,EAAE,uBAAoB;AACjD,OAAO,EAAE,iBAAiB,EAAE,uBAAoB;AAChD,OAAO,EAAE,gBAAgB,EAAE,uBAAoB;AAC/C,OAAO,EAAE,cAAc,EAAE,aAAa,EAAE,MAAM,yBAAyB,CAAC;AAOxE;;;;;;;;;;;;;;;;GAgBG;AACH,MAAM,OAAO,gBAAqC,SAAQ,SAIzD;IAGC,YAAY,OAAuB;QACjC,KAAK,CAAC,OAAO,CAAC,CAAC;IACjB,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;OA2CG;IACO,KAAK;QACb,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACrC,MAAM,SAAS,GAAG,UAAU,CAAC,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QACtD,MAAM,CAAC,GAAG,IAAI,IAAI,CAAC,YAAY,EAAE,CAAC;QAElC,MAAM,CAAC,GAA2C;YAChD,KAAK,EAAE,IAAI,CAAC,OAAO,CAAC,UAAU;iBAC3B,aAAa,CACZ,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC,MAAwC,CAAC,CACtE;iBACA,kBAAkB,CAAC,SAAS,CAA0B;SAC1D,CAAC;QAEF,IAAI,IAAI,CAAC,cAAc;YACrB,CAAC,CAAC,KAAK,GAAG,CAAC,CAAC,KAAK,CAAC,MAAM,CACtB,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,SAAS,IAAI,CAAW,EAAE,CAAC,CAC9D,CAAC;;YACC,CAAC,CAAC,KAAK,GAAG,CAAC,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;QAChC,EAAE;QACF,uDAAuD;QACvD,2EAA2E;QAC3E,cAAc;QACd,KAAK;QAEL,IAAI,IAAI,CAAC,cAAc;YACrB,CAAC,CAAC,KAAK,GAAG,IAAI,CAAC,cAAc,CAC3B,IAAI,CAAC,cAAc,EACnB,SAAS,EACT,CAAC,CAAC,KAAgC,CACnC,CAAC,KAAyC,CAAC;QAE9C,IAAI,WAAqC,CAAC;QAC1C,IAAI,CAAC,IAAI,CAAC,eAAe;YACvB,WAAW,GAAG;gBACZ,GAAG,SAAS,IAAI,cAAc,CAAC,CAAC,CAAC,CAAC,EAAY,EAAE;gBAChD,cAAc,CAAC,GAAG,CAAC,WAAW,EAAW;aAC1C,CAAC;;YAEF,WAAW,GAAG;gBACZ,GAAG,SAAS,IAAI,IAAI,CAAC,eAAe,CAAC,CAAC,CAAW,EAAE;gBACnD,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,WAAW,EAAoB;aACxD,CAAC;QAEJ,CAAC,CAAC,KAAK,GAAI,CAAC,CAAC,KAAiC,CAAC,OAAO,CAAC,GAAG,WAAW,CAAC,CAAC;QACvE,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;YACvB,CAAC,CAAC,KAAK,GAAI,CAAC,CAAC,KAAiC,CAAC,KAAK,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;QAC3E,CAAC;aAAM,CAAC;YACN,GAAG,CAAC,KAAK,CACP,qDAAqD,iBAAiB,EAAE,CACzE,CAAC;YACF,CAAC,CAAC,KAAK,GAAI,CAAC,CAAC,KAAiC,CAAC,KAAK,CAAC,iBAAiB,CAAC,CAAC;QAC1E,CAAC;QAED,aAAa;QACb,IAAI,IAAI,CAAC,cAAc;YACrB,CAAC,CAAC,KAAK,GAAI,CAAC,CAAC,KAAiC,CAAC,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;QAE3E,OAAO,CAAQ,CAAC;IAClB,CAAC;IAED;;;;;;;OAOG;IACH,KAAK,CAAC,QAAQ,CAAI,IAAY;QAC5B,IAAI,CAAC;YACH,MAAM,KAAK,GAAiB,IAAI,CAAC,KAAK,EAAE,CAAC;YACzC,MAAM,gBAAgB,GAAuB,EAAE,CAAC;YAChD,MAAM,CAAC,GAAG,KAAK,CAAC,KAAyC,CAAC;YAC1D,IAAI,IAAI,CAAC,cAAc;gBACrB,gBAAgB,CAAC,KAAK,GAAG,IAAI,CAAC,2BAA2B,CACvD,IAAI,CAAC,cAAc,EACnB,UAAU,CAAC,KAAK,CAAC,IAAI,CAAC,YAAY,CAAC,CACpC,CAAC;YAEJ,IAAI,IAAI,CAAC,eAAe;gBACtB,gBAAgB,CAAC,KAAK,GAAG;oBACvB,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,EAAE,IAAI,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE;iBACvD,CAAC;YAEX,OAAO,IAAI,gBAAgB,CACzB,IAAI,CAAC,OAAc,EACnB,gBAAuB,EACvB,IAAI,EACJ,IAAI,CAAC,YAAY,CAClB,CAAC;QACJ,CAAC;QAAC,OAAO,CAAM,EAAE,CAAC;YAChB,MAAM,IAAI,aAAa,CAAC,CAAC,CAAC,CAAC;QAC7B,CAAC;IACH,CAAC;IAED;;;;;;;OAOG;IACK,aAAa,CAAC,CAAM,EAAE,MAAe;QAC3C,IAAI,OAAO,CAAC,CAAC,MAAM,CAAC,KAAK,WAAW,EAAE,CAAC;YACrC,OAAO,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,EAAE,IAAI,CAAC,YAAY,EAAE,MAAM,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC;QACtE,CAAC;QACD,OAAO,CAAC,CAAC;IACX,CAAC;IAED;;;;;;OAMG;IACM,KAAK,CAAC,GAAG,CAAI,QAAyB;QAC7C,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QACnC,GAAG,CAAC,KAAK,CACP,wBAAyB,QAAQ,CAAC,KAA0C,CAAC,MAAM,EAAE,EAAE,CACxF,CAAC;QACF,OAAO,CAAC,MACN,QAAQ,CAAC,KACV,CAAC,OAAO,EAAE,CAAM,CAAC;IACpB,CAAC;IAES,2BAA2B,CACnC,SAAuB,EACvB,SAAiB,EACjB,OAAO,GAAG,CAAC,EACX,aAAwC;QAExC,MAAM,IAAI,aAAa,CAAC,iBAAiB,CAAC,CAAC;IAC7C,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;OA8BG;IACO,cAAc,CACtB,SAAuB,EACvB,SAAiB,EACjB,EAA2B,EAC3B,OAAO,GAAG,CAAC,EACX,aAAwC;QAExC,MAAM,EAAE,KAAK,EAAE,QAAQ,EAAE,UAAU,EAAE,GAAG,SAIvC,CAAC;QAEF,SAAS,KAAK;YACZ,MAAM,WAAW,GAAG,kBAAkB,CAAC,QAAQ,CAAC,CAAC;YACjD,MAAM,OAAO,GAAG,GAAG,KAAK,GAAG,OAAO,EAAE,CAAC;YACrC,MAAM,QAAQ,GAAG,GAAG,SAAS,IAAI,KAAK,IAAI,WAAW,KAAK,OAAO,EAAE,CAAC;YACpE,MAAM,MAAM,GAAG;gBACb,CAAC,OAAO,CAAC,EAAE,UAAU;aACtB,CAAC;YACF,QAAQ,aAAa,EAAE,CAAC;gBACtB,KAAK,aAAa,CAAC,GAAG;oBACpB,OAAO;wBACL,KAAK,EAAE,EAAE,CAAC,QAAQ,CAAC,QAAQ,EAAE,MAAM,CAAQ;qBAC5C,CAAC;gBACJ,KAAK,aAAa,CAAC,EAAE;oBACnB,OAAO;wBACL,KAAK,EAAE,EAAE,CAAC,OAAO,CAAC,QAAQ,EAAE,MAAM,CAAQ;qBAC3C,CAAC;gBACJ,KAAK,QAAQ,CAAC,GAAG;oBACf,MAAM,IAAI,KAAK,CAAC,8BAA8B,CAAC,CAAC;gBAClD;oBACE,OAAO;wBACL,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,QAAQ,EAAE,MAAM,CAAQ;qBACzC,CAAC;YACN,CAAC;QACH,CAAC;QAED,IACE,CAAC,aAAa,CAAC,GAAG,EAAE,aAAa,CAAC,EAAE,EAAE,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAO,CACzD,QAAyB,CAC1B,KAAK,CAAC,CAAC,EACR,CAAC;YACD,OAAO,KAAK,EAAE,CAAC;QACjB,CAAC;QACD,mBAAmB;aACd,IAAI,QAAQ,KAAK,QAAQ,CAAC,GAAG,EAAE,CAAC;YACnC,MAAM,IAAI,KAAK,CAAC,8BAA8B,CAAC,CAAC;QAClD,CAAC;QACD,uBAAuB;aAClB,CAAC;YACJ,EAAE,GAAG,IAAI,CAAC,cAAc,CAAC,KAAqB,EAAE,SAAS,EAAE,EAAE,EAAE,EAAE,OAAO,CAAC;iBACtE,KAAyC,CAAC;YAC7C,OAAO,IAAI,CAAC,cAAc,CACxB,UAAU,EACV,SAAS,EACT,EAAE,EACF,EAAE,OAAO,EACT,QAAQ,CACT,CAAC;QACJ,CAAC;IACH,CAAC;CACF","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>;
|