@decaf-ts/core 0.5.27 → 0.5.29
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/dist/core.cjs +390 -5
- package/dist/core.esm.cjs +388 -6
- package/lib/esm/identity/decorators.js +5 -4
- package/lib/esm/index.d.ts +1 -1
- package/lib/esm/index.js +1 -1
- package/lib/esm/query/MethodQueryBuilder.d.ts +155 -0
- package/lib/esm/query/MethodQueryBuilder.js +272 -0
- package/lib/esm/query/decorators.d.ts +2 -0
- package/lib/esm/query/decorators.js +45 -0
- package/lib/esm/query/index.d.ts +4 -0
- package/lib/esm/query/index.js +5 -1
- package/lib/esm/query/types.d.ts +96 -0
- package/lib/esm/query/types.js +23 -0
- package/lib/esm/query/utils.d.ts +40 -0
- package/lib/esm/query/utils.js +51 -0
- package/lib/identity/decorators.cjs +5 -4
- package/lib/index.cjs +1 -1
- package/lib/index.d.ts +1 -1
- package/lib/query/MethodQueryBuilder.cjs +276 -0
- package/lib/query/MethodQueryBuilder.d.ts +155 -0
- package/lib/query/decorators.cjs +48 -0
- package/lib/query/decorators.d.ts +2 -0
- package/lib/query/index.cjs +5 -1
- package/lib/query/index.d.ts +4 -0
- package/lib/query/types.cjs +26 -0
- package/lib/query/types.d.ts +96 -0
- package/lib/query/utils.cjs +54 -0
- package/lib/query/utils.d.ts +40 -0
- package/package.json +1 -1
@@ -0,0 +1,96 @@
|
|
1
|
+
import { Condition } from "./Condition";
|
2
|
+
import { OrderBySelector } from "./selectors";
|
3
|
+
/**
|
4
|
+
* @description
|
5
|
+
* Options for configuring query building behavior.
|
6
|
+
*
|
7
|
+
* @summary
|
8
|
+
* The `QueryOptions` type defines flags that determine whether certain clauses
|
9
|
+
* (limit, offset, order by) are permitted, as well as whether violations
|
10
|
+
* should throw an error during query construction.
|
11
|
+
*
|
12
|
+
* @memberOf module:query
|
13
|
+
*/
|
14
|
+
export type QueryOptions = {
|
15
|
+
allowLimit?: boolean;
|
16
|
+
allowOffset?: boolean;
|
17
|
+
allowOrderBy?: boolean;
|
18
|
+
throws?: boolean;
|
19
|
+
};
|
20
|
+
/**
|
21
|
+
* @description
|
22
|
+
* Structured query object representing parsed query clauses.
|
23
|
+
*
|
24
|
+
* @summary
|
25
|
+
* The `QueryAssist` interface defines the standard structure returned
|
26
|
+
* by query builders. It includes actions such as find, optional clauses
|
27
|
+
* like select, groupBy, and orderBy, and pagination controls (limit, offset).
|
28
|
+
*
|
29
|
+
* @template T The entity or record type that conditions may apply to.
|
30
|
+
*
|
31
|
+
* @interface QueryAssist
|
32
|
+
* @memberOf module:query
|
33
|
+
*/
|
34
|
+
export interface QueryAssist {
|
35
|
+
action: "find";
|
36
|
+
select: undefined | string[];
|
37
|
+
where: Condition<any>;
|
38
|
+
groupBy?: string[];
|
39
|
+
orderBy?: OrderBySelector<any>[];
|
40
|
+
limit: number | undefined;
|
41
|
+
offset: number | undefined;
|
42
|
+
}
|
43
|
+
/**
|
44
|
+
* @description
|
45
|
+
* Enumeration of supported query clauses for building method-based queries.
|
46
|
+
*
|
47
|
+
* @summary
|
48
|
+
* The `QueryClause` enum defines string literals that represent
|
49
|
+
* different segments of a query (e.g., `findBy`, `Select`, `And`, `Or`).
|
50
|
+
*
|
51
|
+
* @enum QueryClause
|
52
|
+
* @memberOf module:query
|
53
|
+
*/
|
54
|
+
export declare enum QueryClause {
|
55
|
+
FIND_BY = "findBy",
|
56
|
+
SELECT = "Select",
|
57
|
+
AND = "And",
|
58
|
+
OR = "Or",
|
59
|
+
GROUP_BY = "GroupBy",
|
60
|
+
ORDER_BY = "OrderBy",
|
61
|
+
THEN = "Then",
|
62
|
+
THEN_BY = "ThenBy"
|
63
|
+
}
|
64
|
+
/**
|
65
|
+
* @description
|
66
|
+
* Function signature for parsing operators in query building.
|
67
|
+
*
|
68
|
+
* @summary
|
69
|
+
* The `OperatorParser` type represents a function that takes a field name
|
70
|
+
* and arguments, then produces a `Condition` object that can be used in a query.
|
71
|
+
*
|
72
|
+
* @template T The type of the condition result.
|
73
|
+
*
|
74
|
+
* @param field {string} - The name of the field being parsed.
|
75
|
+
* @param args {any[]} - Additional arguments for operator evaluation.
|
76
|
+
*
|
77
|
+
* @return {Condition<any>} A condition object representing the parsed operator.
|
78
|
+
*
|
79
|
+
* @memberOf module:query
|
80
|
+
*/
|
81
|
+
export type OperatorParser = (field: string, ...args: any) => Condition<any>;
|
82
|
+
/**
|
83
|
+
* @description
|
84
|
+
* Descriptor for fields and their associated operators in query parsing.
|
85
|
+
*
|
86
|
+
* @summary
|
87
|
+
* The `FilterDescriptor` interface defines the structure used when parsing
|
88
|
+
* method segments into filterable fields and associated operators.
|
89
|
+
*
|
90
|
+
* @interface FilterDescriptor
|
91
|
+
* @memberOf module:query
|
92
|
+
*/
|
93
|
+
export interface FilterDescriptor {
|
94
|
+
field: string;
|
95
|
+
operator?: string;
|
96
|
+
}
|
@@ -0,0 +1,23 @@
|
|
1
|
+
/**
|
2
|
+
* @description
|
3
|
+
* Enumeration of supported query clauses for building method-based queries.
|
4
|
+
*
|
5
|
+
* @summary
|
6
|
+
* The `QueryClause` enum defines string literals that represent
|
7
|
+
* different segments of a query (e.g., `findBy`, `Select`, `And`, `Or`).
|
8
|
+
*
|
9
|
+
* @enum QueryClause
|
10
|
+
* @memberOf module:query
|
11
|
+
*/
|
12
|
+
export var QueryClause;
|
13
|
+
(function (QueryClause) {
|
14
|
+
QueryClause["FIND_BY"] = "findBy";
|
15
|
+
QueryClause["SELECT"] = "Select";
|
16
|
+
QueryClause["AND"] = "And";
|
17
|
+
QueryClause["OR"] = "Or";
|
18
|
+
QueryClause["GROUP_BY"] = "GroupBy";
|
19
|
+
QueryClause["ORDER_BY"] = "OrderBy";
|
20
|
+
QueryClause["THEN"] = "Then";
|
21
|
+
QueryClause["THEN_BY"] = "ThenBy";
|
22
|
+
})(QueryClause || (QueryClause = {}));
|
23
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvcXVlcnkvdHlwZXMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBNkNBOzs7Ozs7Ozs7O0dBVUc7QUFDSCxNQUFNLENBQU4sSUFBWSxXQVNYO0FBVEQsV0FBWSxXQUFXO0lBQ3JCLGlDQUFrQixDQUFBO0lBQ2xCLGdDQUFpQixDQUFBO0lBQ2pCLDBCQUFXLENBQUE7SUFDWCx3QkFBUyxDQUFBO0lBQ1QsbUNBQW9CLENBQUE7SUFDcEIsbUNBQW9CLENBQUE7SUFDcEIsNEJBQWEsQ0FBQTtJQUNiLGlDQUFrQixDQUFBO0FBQ3BCLENBQUMsRUFUVyxXQUFXLEtBQVgsV0FBVyxRQVN0QiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbmRpdGlvbiB9IGZyb20gXCIuL0NvbmRpdGlvblwiO1xuaW1wb3J0IHsgT3JkZXJCeVNlbGVjdG9yIH0gZnJvbSBcIi4vc2VsZWN0b3JzXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uXG4gKiBPcHRpb25zIGZvciBjb25maWd1cmluZyBxdWVyeSBidWlsZGluZyBiZWhhdmlvci5cbiAqXG4gKiBAc3VtbWFyeVxuICogVGhlIGBRdWVyeU9wdGlvbnNgIHR5cGUgZGVmaW5lcyBmbGFncyB0aGF0IGRldGVybWluZSB3aGV0aGVyIGNlcnRhaW4gY2xhdXNlc1xuICogKGxpbWl0LCBvZmZzZXQsIG9yZGVyIGJ5KSBhcmUgcGVybWl0dGVkLCBhcyB3ZWxsIGFzIHdoZXRoZXIgdmlvbGF0aW9uc1xuICogc2hvdWxkIHRocm93IGFuIGVycm9yIGR1cmluZyBxdWVyeSBjb25zdHJ1Y3Rpb24uXG4gKlxuICogQG1lbWJlck9mIG1vZHVsZTpxdWVyeVxuICovXG5leHBvcnQgdHlwZSBRdWVyeU9wdGlvbnMgPSB7XG4gIGFsbG93TGltaXQ/OiBib29sZWFuO1xuICBhbGxvd09mZnNldD86IGJvb2xlYW47XG4gIGFsbG93T3JkZXJCeT86IGJvb2xlYW47XG4gIHRocm93cz86IGJvb2xlYW47XG59O1xuXG4vKipcbiAqIEBkZXNjcmlwdGlvblxuICogU3RydWN0dXJlZCBxdWVyeSBvYmplY3QgcmVwcmVzZW50aW5nIHBhcnNlZCBxdWVyeSBjbGF1c2VzLlxuICpcbiAqIEBzdW1tYXJ5XG4gKiBUaGUgYFF1ZXJ5QXNzaXN0YCBpbnRlcmZhY2UgZGVmaW5lcyB0aGUgc3RhbmRhcmQgc3RydWN0dXJlIHJldHVybmVkXG4gKiBieSBxdWVyeSBidWlsZGVycy4gSXQgaW5jbHVkZXMgYWN0aW9ucyBzdWNoIGFzIGZpbmQsIG9wdGlvbmFsIGNsYXVzZXNcbiAqIGxpa2Ugc2VsZWN0LCBncm91cEJ5LCBhbmQgb3JkZXJCeSwgYW5kIHBhZ2luYXRpb24gY29udHJvbHMgKGxpbWl0LCBvZmZzZXQpLlxuICpcbiAqIEB0ZW1wbGF0ZSBUIFRoZSBlbnRpdHkgb3IgcmVjb3JkIHR5cGUgdGhhdCBjb25kaXRpb25zIG1heSBhcHBseSB0by5cbiAqXG4gKiBAaW50ZXJmYWNlIFF1ZXJ5QXNzaXN0XG4gKiBAbWVtYmVyT2YgbW9kdWxlOnF1ZXJ5XG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUXVlcnlBc3Npc3Qge1xuICBhY3Rpb246IFwiZmluZFwiO1xuICBzZWxlY3Q6IHVuZGVmaW5lZCB8IHN0cmluZ1tdO1xuICB3aGVyZTogQ29uZGl0aW9uPGFueT47XG4gIGdyb3VwQnk/OiBzdHJpbmdbXTtcbiAgb3JkZXJCeT86IE9yZGVyQnlTZWxlY3Rvcjxhbnk+W107XG4gIGxpbWl0OiBudW1iZXIgfCB1bmRlZmluZWQ7XG4gIG9mZnNldDogbnVtYmVyIHwgdW5kZWZpbmVkO1xufVxuXG4vKipcbiAqIEBkZXNjcmlwdGlvblxuICogRW51bWVyYXRpb24gb2Ygc3VwcG9ydGVkIHF1ZXJ5IGNsYXVzZXMgZm9yIGJ1aWxkaW5nIG1ldGhvZC1iYXNlZCBxdWVyaWVzLlxuICpcbiAqIEBzdW1tYXJ5XG4gKiBUaGUgYFF1ZXJ5Q2xhdXNlYCBlbnVtIGRlZmluZXMgc3RyaW5nIGxpdGVyYWxzIHRoYXQgcmVwcmVzZW50XG4gKiBkaWZmZXJlbnQgc2VnbWVudHMgb2YgYSBxdWVyeSAoZS5nLiwgYGZpbmRCeWAsIGBTZWxlY3RgLCBgQW5kYCwgYE9yYCkuXG4gKlxuICogQGVudW0gUXVlcnlDbGF1c2VcbiAqIEBtZW1iZXJPZiBtb2R1bGU6cXVlcnlcbiAqL1xuZXhwb3J0IGVudW0gUXVlcnlDbGF1c2Uge1xuICBGSU5EX0JZID0gXCJmaW5kQnlcIixcbiAgU0VMRUNUID0gXCJTZWxlY3RcIixcbiAgQU5EID0gXCJBbmRcIixcbiAgT1IgPSBcIk9yXCIsXG4gIEdST1VQX0JZID0gXCJHcm91cEJ5XCIsXG4gIE9SREVSX0JZID0gXCJPcmRlckJ5XCIsXG4gIFRIRU4gPSBcIlRoZW5cIixcbiAgVEhFTl9CWSA9IFwiVGhlbkJ5XCIsXG59XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uXG4gKiBGdW5jdGlvbiBzaWduYXR1cmUgZm9yIHBhcnNpbmcgb3BlcmF0b3JzIGluIHF1ZXJ5IGJ1aWxkaW5nLlxuICpcbiAqIEBzdW1tYXJ5XG4gKiBUaGUgYE9wZXJhdG9yUGFyc2VyYCB0eXBlIHJlcHJlc2VudHMgYSBmdW5jdGlvbiB0aGF0IHRha2VzIGEgZmllbGQgbmFtZVxuICogYW5kIGFyZ3VtZW50cywgdGhlbiBwcm9kdWNlcyBhIGBDb25kaXRpb25gIG9iamVjdCB0aGF0IGNhbiBiZSB1c2VkIGluIGEgcXVlcnkuXG4gKlxuICogQHRlbXBsYXRlIFQgVGhlIHR5cGUgb2YgdGhlIGNvbmRpdGlvbiByZXN1bHQuXG4gKlxuICogQHBhcmFtIGZpZWxkIHtzdHJpbmd9IC0gVGhlIG5hbWUgb2YgdGhlIGZpZWxkIGJlaW5nIHBhcnNlZC5cbiAqIEBwYXJhbSBhcmdzIHthbnlbXX0gLSBBZGRpdGlvbmFsIGFyZ3VtZW50cyBmb3Igb3BlcmF0b3IgZXZhbHVhdGlvbi5cbiAqXG4gKiBAcmV0dXJuIHtDb25kaXRpb248YW55Pn0gQSBjb25kaXRpb24gb2JqZWN0IHJlcHJlc2VudGluZyB0aGUgcGFyc2VkIG9wZXJhdG9yLlxuICpcbiAqIEBtZW1iZXJPZiBtb2R1bGU6cXVlcnlcbiAqL1xuZXhwb3J0IHR5cGUgT3BlcmF0b3JQYXJzZXIgPSAoZmllbGQ6IHN0cmluZywgLi4uYXJnczogYW55KSA9PiBDb25kaXRpb248YW55PjtcblxuLyoqXG4gKiBAZGVzY3JpcHRpb25cbiAqIERlc2NyaXB0b3IgZm9yIGZpZWxkcyBhbmQgdGhlaXIgYXNzb2NpYXRlZCBvcGVyYXRvcnMgaW4gcXVlcnkgcGFyc2luZy5cbiAqXG4gKiBAc3VtbWFyeVxuICogVGhlIGBGaWx0ZXJEZXNjcmlwdG9yYCBpbnRlcmZhY2UgZGVmaW5lcyB0aGUgc3RydWN0dXJlIHVzZWQgd2hlbiBwYXJzaW5nXG4gKiBtZXRob2Qgc2VnbWVudHMgaW50byBmaWx0ZXJhYmxlIGZpZWxkcyBhbmQgYXNzb2NpYXRlZCBvcGVyYXRvcnMuXG4gKlxuICogQGludGVyZmFjZSBGaWx0ZXJEZXNjcmlwdG9yXG4gKiBAbWVtYmVyT2YgbW9kdWxlOnF1ZXJ5XG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgRmlsdGVyRGVzY3JpcHRvciB7XG4gIGZpZWxkOiBzdHJpbmc7XG4gIG9wZXJhdG9yPzogc3RyaW5nO1xufVxuIl19
|
@@ -0,0 +1,40 @@
|
|
1
|
+
import { OperatorParser } from "./types";
|
2
|
+
/**
|
3
|
+
* @description
|
4
|
+
* Map of supported operators to their corresponding parser functions.
|
5
|
+
*
|
6
|
+
* @summary
|
7
|
+
* The `OperatorsMap` defines a collection of operator names as keys
|
8
|
+
* (such as `Equals`, `LessThan`, `Between`, etc.), each mapped to a
|
9
|
+
* function that constructs a `Condition` object for that operator.
|
10
|
+
* These functions translate query clauses into concrete condition
|
11
|
+
* builders, enabling dynamic query construction from method names.
|
12
|
+
*
|
13
|
+
* @template T The type of the field values used in conditions.
|
14
|
+
*
|
15
|
+
* @param f {string} - The field name the condition applies to.
|
16
|
+
* @param v1 {any} - The value to compare the field against or the lower bound value for range-based operators.
|
17
|
+
* @param v2 {any} - The upper bound value for range-based operators.
|
18
|
+
*
|
19
|
+
* @return {Condition<any>} A condition object representing the operator applied to the field.
|
20
|
+
*
|
21
|
+
* @function OperatorsMap
|
22
|
+
*
|
23
|
+
* @mermaid
|
24
|
+
* sequenceDiagram
|
25
|
+
* participant Client as Caller
|
26
|
+
* participant Map as OperatorsMap
|
27
|
+
* participant Parser as OperatorParser
|
28
|
+
* participant Cond as Condition
|
29
|
+
*
|
30
|
+
* Client->>Map: Request operator parser ("Between", field, v1, v2)
|
31
|
+
* Map->>Parser: Call corresponding operator function
|
32
|
+
* Parser->>Cond: Condition.attribute(field)
|
33
|
+
* Cond-->>Parser: Condition instance
|
34
|
+
* Parser->>Cond: Apply gte(v1)
|
35
|
+
* Parser->>Cond: Apply and(lte(v2))
|
36
|
+
* Parser-->>Client: Return built Condition
|
37
|
+
*
|
38
|
+
* @memberOf module:query
|
39
|
+
*/
|
40
|
+
export declare const OperatorsMap: Record<string, OperatorParser>;
|
@@ -0,0 +1,51 @@
|
|
1
|
+
import { Condition } from "./Condition.js";
|
2
|
+
/**
|
3
|
+
* @description
|
4
|
+
* Map of supported operators to their corresponding parser functions.
|
5
|
+
*
|
6
|
+
* @summary
|
7
|
+
* The `OperatorsMap` defines a collection of operator names as keys
|
8
|
+
* (such as `Equals`, `LessThan`, `Between`, etc.), each mapped to a
|
9
|
+
* function that constructs a `Condition` object for that operator.
|
10
|
+
* These functions translate query clauses into concrete condition
|
11
|
+
* builders, enabling dynamic query construction from method names.
|
12
|
+
*
|
13
|
+
* @template T The type of the field values used in conditions.
|
14
|
+
*
|
15
|
+
* @param f {string} - The field name the condition applies to.
|
16
|
+
* @param v1 {any} - The value to compare the field against or the lower bound value for range-based operators.
|
17
|
+
* @param v2 {any} - The upper bound value for range-based operators.
|
18
|
+
*
|
19
|
+
* @return {Condition<any>} A condition object representing the operator applied to the field.
|
20
|
+
*
|
21
|
+
* @function OperatorsMap
|
22
|
+
*
|
23
|
+
* @mermaid
|
24
|
+
* sequenceDiagram
|
25
|
+
* participant Client as Caller
|
26
|
+
* participant Map as OperatorsMap
|
27
|
+
* participant Parser as OperatorParser
|
28
|
+
* participant Cond as Condition
|
29
|
+
*
|
30
|
+
* Client->>Map: Request operator parser ("Between", field, v1, v2)
|
31
|
+
* Map->>Parser: Call corresponding operator function
|
32
|
+
* Parser->>Cond: Condition.attribute(field)
|
33
|
+
* Cond-->>Parser: Condition instance
|
34
|
+
* Parser->>Cond: Apply gte(v1)
|
35
|
+
* Parser->>Cond: Apply and(lte(v2))
|
36
|
+
* Parser-->>Client: Return built Condition
|
37
|
+
*
|
38
|
+
* @memberOf module:query
|
39
|
+
*/
|
40
|
+
export const OperatorsMap = {
|
41
|
+
Equals: (f, v) => Condition.attribute(f).eq(v),
|
42
|
+
Diff: (f, v) => Condition.attribute(f).dif(v),
|
43
|
+
LessThan: (f, v) => Condition.attribute(f).lt(v),
|
44
|
+
LessThanEqual: (f, v) => Condition.attribute(f).lte(v),
|
45
|
+
GreaterThan: (f, v) => Condition.attribute(f).gt(v),
|
46
|
+
GreaterThanEqual: (f, v) => Condition.attribute(f).gte(v),
|
47
|
+
Between: (f, v1, v2) => Condition.attribute(f).gte(v1).and(Condition.attribute(f).lte(v2)),
|
48
|
+
In: (f, v) => Condition.attribute(f).in(v),
|
49
|
+
Matches: (f, v) => Condition.attribute(f).regexp(v),
|
50
|
+
};
|
51
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvcXVlcnkvdXRpbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSx1QkFBb0I7QUFHeEM7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FxQ0c7QUFDSCxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQW1DO0lBQzFELE1BQU0sRUFBRSxDQUFDLENBQUMsRUFBRSxDQUFDLEVBQUUsRUFBRSxDQUFDLFNBQVMsQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQztJQUM5QyxJQUFJLEVBQUUsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxTQUFTLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUM7SUFDN0MsUUFBUSxFQUFFLENBQUMsQ0FBQyxFQUFFLENBQUMsRUFBRSxFQUFFLENBQUMsU0FBUyxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDO0lBQ2hELGFBQWEsRUFBRSxDQUFDLENBQUMsRUFBRSxDQUFDLEVBQUUsRUFBRSxDQUFDLFNBQVMsQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBQztJQUN0RCxXQUFXLEVBQUUsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxTQUFTLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7SUFDbkQsZ0JBQWdCLEVBQUUsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxTQUFTLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUM7SUFDekQsT0FBTyxFQUFFLENBQUMsQ0FBQyxFQUFFLEVBQUUsRUFBRSxFQUFFLEVBQUUsRUFBRSxDQUNyQixTQUFTLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsQ0FBQyxHQUFHLENBQUMsU0FBUyxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLENBQUM7SUFDcEUsRUFBRSxFQUFFLENBQUMsQ0FBQyxFQUFFLENBQUMsRUFBRSxFQUFFLENBQUMsU0FBUyxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDO0lBQzFDLE9BQU8sRUFBRSxDQUFDLENBQUMsRUFBRSxDQUFDLEVBQUUsRUFBRSxDQUFDLFNBQVMsQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQztDQUNwRCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29uZGl0aW9uIH0gZnJvbSBcIi4vQ29uZGl0aW9uXCI7XG5pbXBvcnQgeyBPcGVyYXRvclBhcnNlciB9IGZyb20gXCIuL3R5cGVzXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uXG4gKiBNYXAgb2Ygc3VwcG9ydGVkIG9wZXJhdG9ycyB0byB0aGVpciBjb3JyZXNwb25kaW5nIHBhcnNlciBmdW5jdGlvbnMuXG4gKlxuICogQHN1bW1hcnlcbiAqIFRoZSBgT3BlcmF0b3JzTWFwYCBkZWZpbmVzIGEgY29sbGVjdGlvbiBvZiBvcGVyYXRvciBuYW1lcyBhcyBrZXlzXG4gKiAoc3VjaCBhcyBgRXF1YWxzYCwgYExlc3NUaGFuYCwgYEJldHdlZW5gLCBldGMuKSwgZWFjaCBtYXBwZWQgdG8gYVxuICogZnVuY3Rpb24gdGhhdCBjb25zdHJ1Y3RzIGEgYENvbmRpdGlvbmAgb2JqZWN0IGZvciB0aGF0IG9wZXJhdG9yLlxuICogVGhlc2UgZnVuY3Rpb25zIHRyYW5zbGF0ZSBxdWVyeSBjbGF1c2VzIGludG8gY29uY3JldGUgY29uZGl0aW9uXG4gKiBidWlsZGVycywgZW5hYmxpbmcgZHluYW1pYyBxdWVyeSBjb25zdHJ1Y3Rpb24gZnJvbSBtZXRob2QgbmFtZXMuXG4gKlxuICogQHRlbXBsYXRlIFQgVGhlIHR5cGUgb2YgdGhlIGZpZWxkIHZhbHVlcyB1c2VkIGluIGNvbmRpdGlvbnMuXG4gKlxuICogQHBhcmFtIGYge3N0cmluZ30gLSBUaGUgZmllbGQgbmFtZSB0aGUgY29uZGl0aW9uIGFwcGxpZXMgdG8uXG4gKiBAcGFyYW0gdjEge2FueX0gLSBUaGUgdmFsdWUgdG8gY29tcGFyZSB0aGUgZmllbGQgYWdhaW5zdCBvciB0aGUgbG93ZXIgYm91bmQgdmFsdWUgZm9yIHJhbmdlLWJhc2VkIG9wZXJhdG9ycy5cbiAqIEBwYXJhbSB2MiB7YW55fSAtIFRoZSB1cHBlciBib3VuZCB2YWx1ZSBmb3IgcmFuZ2UtYmFzZWQgb3BlcmF0b3JzLlxuICpcbiAqIEByZXR1cm4ge0NvbmRpdGlvbjxhbnk+fSBBIGNvbmRpdGlvbiBvYmplY3QgcmVwcmVzZW50aW5nIHRoZSBvcGVyYXRvciBhcHBsaWVkIHRvIHRoZSBmaWVsZC5cbiAqXG4gKiBAZnVuY3Rpb24gT3BlcmF0b3JzTWFwXG4gKlxuICogQG1lcm1haWRcbiAqIHNlcXVlbmNlRGlhZ3JhbVxuICogICBwYXJ0aWNpcGFudCBDbGllbnQgYXMgQ2FsbGVyXG4gKiAgIHBhcnRpY2lwYW50IE1hcCBhcyBPcGVyYXRvcnNNYXBcbiAqICAgcGFydGljaXBhbnQgUGFyc2VyIGFzIE9wZXJhdG9yUGFyc2VyXG4gKiAgIHBhcnRpY2lwYW50IENvbmQgYXMgQ29uZGl0aW9uXG4gKlxuICogICBDbGllbnQtPj5NYXA6IFJlcXVlc3Qgb3BlcmF0b3IgcGFyc2VyIChcIkJldHdlZW5cIiwgZmllbGQsIHYxLCB2MilcbiAqICAgTWFwLT4+UGFyc2VyOiBDYWxsIGNvcnJlc3BvbmRpbmcgb3BlcmF0b3IgZnVuY3Rpb25cbiAqICAgUGFyc2VyLT4+Q29uZDogQ29uZGl0aW9uLmF0dHJpYnV0ZShmaWVsZClcbiAqICAgQ29uZC0tPj5QYXJzZXI6IENvbmRpdGlvbiBpbnN0YW5jZVxuICogICBQYXJzZXItPj5Db25kOiBBcHBseSBndGUodjEpXG4gKiAgIFBhcnNlci0+PkNvbmQ6IEFwcGx5IGFuZChsdGUodjIpKVxuICogICBQYXJzZXItLT4+Q2xpZW50OiBSZXR1cm4gYnVpbHQgQ29uZGl0aW9uXG4gKlxuICogQG1lbWJlck9mIG1vZHVsZTpxdWVyeVxuICovXG5leHBvcnQgY29uc3QgT3BlcmF0b3JzTWFwOiBSZWNvcmQ8c3RyaW5nLCBPcGVyYXRvclBhcnNlcj4gPSB7XG4gIEVxdWFsczogKGYsIHYpID0+IENvbmRpdGlvbi5hdHRyaWJ1dGUoZikuZXEodiksXG4gIERpZmY6IChmLCB2KSA9PiBDb25kaXRpb24uYXR0cmlidXRlKGYpLmRpZih2KSxcbiAgTGVzc1RoYW46IChmLCB2KSA9PiBDb25kaXRpb24uYXR0cmlidXRlKGYpLmx0KHYpLFxuICBMZXNzVGhhbkVxdWFsOiAoZiwgdikgPT4gQ29uZGl0aW9uLmF0dHJpYnV0ZShmKS5sdGUodiksXG4gIEdyZWF0ZXJUaGFuOiAoZiwgdikgPT4gQ29uZGl0aW9uLmF0dHJpYnV0ZShmKS5ndCh2KSxcbiAgR3JlYXRlclRoYW5FcXVhbDogKGYsIHYpID0+IENvbmRpdGlvbi5hdHRyaWJ1dGUoZikuZ3RlKHYpLFxuICBCZXR3ZWVuOiAoZiwgdjEsIHYyKSA9PlxuICAgIENvbmRpdGlvbi5hdHRyaWJ1dGUoZikuZ3RlKHYxKS5hbmQoQ29uZGl0aW9uLmF0dHJpYnV0ZShmKS5sdGUodjIpKSxcbiAgSW46IChmLCB2KSA9PiBDb25kaXRpb24uYXR0cmlidXRlKGYpLmluKHYpLFxuICBNYXRjaGVzOiAoZiwgdikgPT4gQ29uZGl0aW9uLmF0dHJpYnV0ZShmKS5yZWdleHAodiksXG59O1xuIl19
|
@@ -10,6 +10,7 @@ const decorators_1 = require("./../model/decorators.cjs");
|
|
10
10
|
const utils_1 = require("./utils.cjs");
|
11
11
|
const repository_1 = require("./../repository/index.cjs");
|
12
12
|
const reflection_1 = require("@decaf-ts/reflection");
|
13
|
+
const defaultPkPriority = 60; // Default priority for primary key to run latter than other properties
|
13
14
|
/**
|
14
15
|
* @description Callback function for primary key creation
|
15
16
|
* @summary Handles the creation of primary key values for models using sequences
|
@@ -97,16 +98,16 @@ function pk(opts = SequenceOptions_1.DefaultSequenceOptions) {
|
|
97
98
|
: opts.generated || SequenceOptions_1.DefaultSequenceOptions.generated,
|
98
99
|
});
|
99
100
|
const key = Repository_1.Repository.key(db_decorators_1.DBKeys.ID);
|
100
|
-
function pkDec(options) {
|
101
|
+
function pkDec(options, groupsort) {
|
101
102
|
return function pkDec(obj, attr) {
|
102
|
-
return (0, reflection_1.apply)((0, decorators_1.index)([repository_1.OrderDirection.ASC, repository_1.OrderDirection.DSC]), (0, decorator_validation_1.required)(), (0, db_decorators_1.readonly)(), (0, decorator_validation_1.propMetadata)(key, options), (0, db_decorators_1.onCreate)(pkOnCreate, options), (0, decorator_validation_1.propMetadata)(db_decorators_1.DBKeys.ID, attr))(obj, attr);
|
103
|
+
return (0, reflection_1.apply)((0, decorators_1.index)([repository_1.OrderDirection.ASC, repository_1.OrderDirection.DSC]), (0, decorator_validation_1.required)(), (0, db_decorators_1.readonly)(), (0, decorator_validation_1.propMetadata)(key, options), (0, db_decorators_1.onCreate)(pkOnCreate, options, groupsort), (0, decorator_validation_1.propMetadata)(db_decorators_1.DBKeys.ID, attr))(obj, attr);
|
103
104
|
};
|
104
105
|
}
|
105
106
|
return decorator_validation_1.Decoration.for(key)
|
106
107
|
.define({
|
107
108
|
decorator: pkDec,
|
108
|
-
args: [opts],
|
109
|
+
args: [opts, { priority: defaultPkPriority }],
|
109
110
|
})
|
110
111
|
.apply();
|
111
112
|
}
|
112
|
-
//# sourceMappingURL=data:application/json;base64,
|
113
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decorators.js","sourceRoot":"","sources":["../../src/identity/decorators.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Decoration,\n  Model,\n  propMetadata,\n  required,\n} from \"@decaf-ts/decorator-validation\";\nimport {\n  DefaultSequenceOptions,\n  SequenceOptions,\n} from \"../interfaces/SequenceOptions\";\nimport {\n  DBKeys,\n  GroupSort,\n  InternalError,\n  onCreate,\n  readonly,\n  RepositoryFlags,\n} from \"@decaf-ts/db-decorators\";\nimport { Repo, Repository } from \"../repository/Repository\";\nimport { index } from \"../model/decorators\";\nimport { sequenceNameForModel } from \"./utils\";\nimport { Sequence } from \"../persistence/Sequence\";\nimport { Context } from \"@decaf-ts/db-decorators\";\nimport { OrderDirection } from \"../repository\";\nimport { apply } from \"@decaf-ts/reflection\";\n\nconst defaultPkPriority = 60; // Default priority for primary key to run latter than other properties\n\n/**\n * @description Callback function for primary key creation\n * @summary Handles the creation of primary key values for models using sequences\n * @template M - Type that extends Model\n * @template R - Type that extends Repo<M, F, C>\n * @template V - Type that extends SequenceOptions\n * @template F - Type that extends RepositoryFlags\n * @template C - Type that extends Context<F>\n * @param {Context<F>} context - The execution context\n * @param {V} data - The sequence options\n * @param key - The property key to set as primary key\n * @param {M} model - The model instance\n * @return {Promise<void>} A promise that resolves when the primary key is set\n * @function pkOnCreate\n * @category Property Decorators\n * @mermaid\n * sequenceDiagram\n *   participant Model\n *   participant pkOnCreate\n *   participant Adapter\n *   participant Sequence\n *\n *   Model->>pkOnCreate: Call with model instance\n *   Note over pkOnCreate: Check if key already exists\n *   alt Key exists or no type specified\n *     pkOnCreate-->>Model: Return early\n *   else Key needs to be created\n *     pkOnCreate->>pkOnCreate: Generate sequence name if not provided\n *     pkOnCreate->>Adapter: Request Sequence(data)\n *     Adapter->>Sequence: Create sequence\n *     Sequence-->>pkOnCreate: Return sequence\n *     pkOnCreate->>Sequence: Call next()\n *     Sequence-->>pkOnCreate: Return next value\n *     pkOnCreate->>Model: Set primary key value\n *   end\n */\nexport async function pkOnCreate<\n  M extends Model,\n  R extends Repo<M, F, C>,\n  V extends SequenceOptions,\n  F extends RepositoryFlags,\n  C extends Context<F>,\n>(\n  this: R,\n  context: Context<F>,\n  data: V,\n  key: keyof M,\n  model: M\n): Promise<void> {\n  if (!data.type || !data.generated || model[key]) {\n    return;\n  }\n\n  const setPrimaryKeyValue = function <M extends Model>(\n    target: M,\n    propertyKey: string,\n    value: string | number | bigint\n  ) {\n    Object.defineProperty(target, propertyKey, {\n      enumerable: true,\n      writable: false,\n      configurable: true,\n      value: value,\n    });\n  };\n\n  if (!data.name) data.name = sequenceNameForModel(model, \"pk\");\n  let sequence: Sequence;\n  try {\n    sequence = await this.adapter.Sequence(data);\n  } catch (e: any) {\n    throw new InternalError(\n      `Failed to instantiate Sequence ${data.name}: ${e}`\n    );\n  }\n\n  const next = await sequence.next();\n  setPrimaryKeyValue(model, key as string, next);\n}\n\n/**\n * @description Primary Key Decorator\n * @summary Marks a property as the model's primary key with automatic sequence generation\n * This decorator combines multiple behaviors: it marks the property as unique, required,\n * and ensures the index is created properly according to the provided sequence options.\n * @param {Omit<SequenceOptions, \"cycle\" | \"startWith\" | \"incrementBy\">} opts - Options for the sequence generation\n * @return {PropertyDecorator} A property decorator that can be applied to model properties\n * @function pk\n * @category Property Decorators\n * @example\n * ```typescript\n * class User extends BaseModel {\n *   @pk()\n *   id!: string;\n *\n *   @required()\n *   username!: string;\n * }\n * ```\n */\nexport function pk(\n  opts: Omit<\n    SequenceOptions,\n    \"cycle\" | \"startWith\" | \"incrementBy\"\n  > = DefaultSequenceOptions\n) {\n  opts = Object.assign({}, DefaultSequenceOptions, opts, {\n    generated:\n      opts.type && typeof opts.generated === \"undefined\"\n        ? true\n        : opts.generated || DefaultSequenceOptions.generated,\n  }) as SequenceOptions;\n\n  const key = Repository.key(DBKeys.ID);\n  function pkDec(options: SequenceOptions, groupsort?: GroupSort) {\n    return function pkDec(obj: any, attr: any) {\n      return apply(\n        index([OrderDirection.ASC, OrderDirection.DSC]),\n        required(),\n        readonly(),\n        propMetadata(key, options),\n        onCreate(pkOnCreate, options, groupsort),\n        propMetadata(DBKeys.ID, attr)\n      )(obj, attr);\n    };\n  }\n  return Decoration.for(key)\n    .define({\n      decorator: pkDec,\n      args: [opts, { priority: defaultPkPriority }],\n    })\n    .apply();\n}\n"]}
|
package/lib/index.cjs
CHANGED
@@ -42,5 +42,5 @@ __exportStar(require("./persistence/index.cjs"), exports);
|
|
42
42
|
* @const VERSION
|
43
43
|
* @memberOf module:core
|
44
44
|
*/
|
45
|
-
exports.VERSION = "0.5.
|
45
|
+
exports.VERSION = "0.5.29";
|
46
46
|
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7Ozs7R0FNRzs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFFSCw4REFBK0Q7QUFDL0QsMkVBQThEO0FBRTlELGtGQUFrRjtBQUNsRixtQ0FBVyxDQUFDLFdBQVcsQ0FBQyxJQUFJLGlDQUFtQixFQUFFLENBQUMsQ0FBQztBQUVuRCw0QkFBNEI7QUFDNUIseURBQTZCO0FBRTdCLHVEQUEyQjtBQUMzQix5REFBNkI7QUFDN0Isb0RBQXdCO0FBQ3hCLG9EQUF3QjtBQUN4QixvREFBd0I7QUFDeEIsdURBQWlDO0FBQ2pDLHlCQUF5QjtBQUN6QiwwREFBOEI7QUFFOUI7Ozs7O0dBS0c7QUFDVSxRQUFBLE9BQU8sR0FBRyxhQUFhLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBtb2R1bGUgY29yZVxuICogQGRlc2NyaXB0aW9uIENvcmUgbW9kdWxlIGZvciB0aGUgRGVjYWYgVHlwZVNjcmlwdCBmcmFtZXdvcmtcbiAqIEBzdW1tYXJ5IFRoaXMgbW9kdWxlIHByb3ZpZGVzIHRoZSBmb3VuZGF0aW9uYWwgY29tcG9uZW50cyBvZiB0aGUgRGVjYWYgZnJhbWV3b3JrLCBpbmNsdWRpbmcgaWRlbnRpdHkgbWFuYWdlbWVudCxcbiAqIG1vZGVsIGRlZmluaXRpb25zLCByZXBvc2l0b3J5IHBhdHRlcm5zLCBwZXJzaXN0ZW5jZSBsYXllciwgcXVlcnkgYnVpbGRpbmcsIGFuZCB1dGlsaXR5IGZ1bmN0aW9ucy5cbiAqIEl0IGV4cG9ydHMgZnVuY3Rpb25hbGl0eSBmcm9tIHZhcmlvdXMgc3VibW9kdWxlcyBhbmQgc2V0cyB1cCB0aGUgaW5qZWN0YWJsZSByZWdpc3RyeSBmb3IgcmVwb3NpdG9yeSBkZWNvcmF0b3JzLlxuICovXG5cbmltcG9ydCB7IEluamVjdGFibGVzUmVnaXN0cnkgfSBmcm9tIFwiLi9yZXBvc2l0b3J5L2luamVjdGFibGVzXCI7XG5pbXBvcnQgeyBJbmplY3RhYmxlcyB9IGZyb20gXCJAZGVjYWYtdHMvaW5qZWN0YWJsZS1kZWNvcmF0b3JzXCI7XG5cbi8vIG92ZXJyaWRlcyB0aGUgcHJldmlvdXMgSW5qZWN0YWJsZXMgcmVnaXN0cnkgdG8gZW5hYmxlIHRoZSBAcmVwb3NpdG9yeSBkZWNvcmF0b3JcbkluamVjdGFibGVzLnNldFJlZ2lzdHJ5KG5ldyBJbmplY3RhYmxlc1JlZ2lzdHJ5KCkpO1xuXG4vLyBpbXBvcnRlZCBmaXJzdCBvbiBwdXJwb3NlXG5leHBvcnQgKiBmcm9tIFwiLi9yZXBvc2l0b3J5XCI7XG5cbmV4cG9ydCAqIGZyb20gXCIuL2lkZW50aXR5XCI7XG5leHBvcnQgKiBmcm9tIFwiLi9pbnRlcmZhY2VzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9tb2RlbFwiO1xuZXhwb3J0ICogZnJvbSBcIi4vcXVlcnlcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3V0aWxzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9yYW0vUmFtQWRhcHRlclwiO1xuLy9sZWZ0IHRvIGxhc3Qgb24gcHVycG9zZVxuZXhwb3J0ICogZnJvbSBcIi4vcGVyc2lzdGVuY2VcIjtcblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gU3RvcmVzIHRoZSBjdXJyZW50IHBhY2thZ2UgdmVyc2lvblxuICogQHN1bW1hcnkgQSBjb25zdGFudCByZXByZXNlbnRpbmcgdGhlIHZlcnNpb24gb2YgdGhlIGNvcmUgcGFja2FnZVxuICogQGNvbnN0IFZFUlNJT05cbiAqIEBtZW1iZXJPZiBtb2R1bGU6Y29yZVxuICovXG5leHBvcnQgY29uc3QgVkVSU0lPTiA9IFwiIyNWRVJTSU9OIyNcIjtcbiJdfQ==
|
package/lib/index.d.ts
CHANGED
@@ -0,0 +1,276 @@
|
|
1
|
+
"use strict";
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
3
|
+
exports.MethodQueryBuilder = void 0;
|
4
|
+
const types_1 = require("./types.cjs");
|
5
|
+
const utils_1 = require("./utils.cjs");
|
6
|
+
const lowerFirst = (str) => str.charAt(0).toLowerCase() + str.slice(1);
|
7
|
+
/**
|
8
|
+
* @description
|
9
|
+
* Utility class to build query objects from repository method names.
|
10
|
+
*
|
11
|
+
* @summary
|
12
|
+
* The `MethodQueryBuilder` class parses method names that follow a specific naming convention
|
13
|
+
* (e.g., `findByNameAndAgeOrderByCountryAsc`) and converts them into structured query objects
|
14
|
+
* (`QueryAssist`). It extracts clauses such as `select`, `where`, `groupBy`, `orderBy`, `limit`,
|
15
|
+
* and `offset`, ensuring that developers can declare repository queries using expressive method names.
|
16
|
+
*
|
17
|
+
* @param methodName {string} - The repository method name to parse and convert into a query.
|
18
|
+
* @param values {any[]} - The values corresponding to method parameters used for query conditions.
|
19
|
+
*
|
20
|
+
* @return {QueryAssist} A structured query object describing the parsed action, select, where,
|
21
|
+
* groupBy, orderBy, limit, and offset clauses.
|
22
|
+
*
|
23
|
+
* @class
|
24
|
+
*
|
25
|
+
* @example
|
26
|
+
* ```ts
|
27
|
+
* const query = MethodQueryBuilder.build(
|
28
|
+
* "findByNameAndAgeOrderByCountryAsc",
|
29
|
+
* "John",
|
30
|
+
* 25,
|
31
|
+
* [["country", "ASC"]]
|
32
|
+
* );
|
33
|
+
*
|
34
|
+
* console.log(query);
|
35
|
+
* // {
|
36
|
+
* // action: "find",
|
37
|
+
* // select: undefined,
|
38
|
+
* // where: { ... },
|
39
|
+
* // groupBy: undefined,
|
40
|
+
* // orderBy: [["country", "ASC"]],
|
41
|
+
* // limit: undefined,
|
42
|
+
* // offset: undefined
|
43
|
+
* // }
|
44
|
+
* ```
|
45
|
+
*
|
46
|
+
* @mermaid
|
47
|
+
* sequenceDiagram
|
48
|
+
* participant Repo as Repository Method
|
49
|
+
* participant MQB as MethodQueryBuilder
|
50
|
+
* participant Query as QueryAssist
|
51
|
+
*
|
52
|
+
* Repo->>MQB: build(methodName, ...values)
|
53
|
+
* MQB->>MQB: extractCore(methodName)
|
54
|
+
* MQB->>MQB: extractSelect(methodName)
|
55
|
+
* MQB->>MQB: extractGroupBy(methodName)
|
56
|
+
* MQB->>MQB: buildWhere(core, values)
|
57
|
+
* MQB->>MQB: extractOrderLimitOffset(core, values)
|
58
|
+
* MQB->>Query: return structured QueryAssist object
|
59
|
+
*/
|
60
|
+
class MethodQueryBuilder {
|
61
|
+
/**
|
62
|
+
* @description
|
63
|
+
* Builds a `QueryAssist` object by parsing a repository method name and values.
|
64
|
+
*
|
65
|
+
* @summary
|
66
|
+
* The method validates the method name, extracts clauses (core, select, groupBy, where,
|
67
|
+
* orderBy, limit, and offset), and assembles them into a structured query object
|
68
|
+
* that can be executed against a data source.
|
69
|
+
*
|
70
|
+
* @param methodName {string} - The repository method name that encodes query information.
|
71
|
+
* @param values {any[]} - The values corresponding to conditions and extra clauses.
|
72
|
+
*
|
73
|
+
* @return {QueryAssist} A structured query object representing the parsed query.
|
74
|
+
*/
|
75
|
+
static build(methodName, ...values) {
|
76
|
+
if (!methodName.startsWith(types_1.QueryClause.FIND_BY)) {
|
77
|
+
throw new Error(`Unsupported method ${methodName}`);
|
78
|
+
}
|
79
|
+
const core = this.extractCore(methodName);
|
80
|
+
const select = this.extractSelect(methodName);
|
81
|
+
const groupBy = this.extractGroupBy(methodName);
|
82
|
+
// const orderBy = this.extractOrderBy(methodName);
|
83
|
+
const where = this.buildWhere(core, values);
|
84
|
+
const { orderBy, limit, offset } = this.extractOrderLimitOffset(core, values);
|
85
|
+
return {
|
86
|
+
action: "find",
|
87
|
+
select: select,
|
88
|
+
where,
|
89
|
+
groupBy,
|
90
|
+
orderBy,
|
91
|
+
limit,
|
92
|
+
offset,
|
93
|
+
};
|
94
|
+
}
|
95
|
+
/**
|
96
|
+
* @description
|
97
|
+
* Extracts the core part of the method name after `findBy` and before any special clauses.
|
98
|
+
*
|
99
|
+
* @summary
|
100
|
+
* Removes prefixes and detects delimiters (`Then`, `OrderBy`, `GroupBy`, `Limit`, `Offset`)
|
101
|
+
* to isolate the main conditional part of the query.
|
102
|
+
*
|
103
|
+
* @param methodName {string} - The method name to parse.
|
104
|
+
*
|
105
|
+
* @return {string} The extracted core string used for building conditions.
|
106
|
+
*/
|
107
|
+
static extractCore(methodName) {
|
108
|
+
const afterFindBy = methodName.substring(types_1.QueryClause.FIND_BY.length);
|
109
|
+
const regex = /(Then[A-Z]|OrderBy|GroupBy|Limit|Offset)/;
|
110
|
+
const match = afterFindBy.match(regex);
|
111
|
+
return match ? afterFindBy.substring(0, match.index) : afterFindBy;
|
112
|
+
}
|
113
|
+
/**
|
114
|
+
* @description
|
115
|
+
* Extracts the select clause from a method name.
|
116
|
+
*
|
117
|
+
* @summary
|
118
|
+
* Detects the `Select` keyword in the method name, isolates the fields following it,
|
119
|
+
* and returns them as an array of lowercase-first strings.
|
120
|
+
*
|
121
|
+
* @param methodName {string} - The method name to parse.
|
122
|
+
*
|
123
|
+
* @return {string[] | undefined} An array of selected fields or `undefined` if no select clause exists.
|
124
|
+
*/
|
125
|
+
static extractSelect(methodName) {
|
126
|
+
const selectIndex = methodName.indexOf(types_1.QueryClause.SELECT);
|
127
|
+
if (selectIndex === -1)
|
128
|
+
return undefined;
|
129
|
+
const afterSelect = methodName.substring(selectIndex + types_1.QueryClause.SELECT.length);
|
130
|
+
// Search for next Then, GroupBy, OrderBy...
|
131
|
+
const match = afterSelect.match(/(Then[A-Z]|OrderBy|GroupBy|Limit|Offset)/);
|
132
|
+
const selectPart = match
|
133
|
+
? afterSelect.substring(0, match.index)
|
134
|
+
: afterSelect;
|
135
|
+
return selectPart.split(types_1.QueryClause.AND).map(lowerFirst).filter(Boolean);
|
136
|
+
}
|
137
|
+
/**
|
138
|
+
* @description
|
139
|
+
* Extracts the group by clause from a method name.
|
140
|
+
*
|
141
|
+
* @summary
|
142
|
+
* Detects the `GroupBy` keyword in the method name, isolates the fields following it,
|
143
|
+
* and returns them as an array of lowercase-first strings.
|
144
|
+
*
|
145
|
+
* @param methodName {string} - The method name to parse.
|
146
|
+
*
|
147
|
+
* @return {string[] | undefined} An array of group by fields or `undefined` if no group by clause exists.
|
148
|
+
*/
|
149
|
+
static extractGroupBy(methodName) {
|
150
|
+
const groupByIndex = methodName.indexOf(types_1.QueryClause.GROUP_BY);
|
151
|
+
if (groupByIndex === -1)
|
152
|
+
return undefined;
|
153
|
+
const after = methodName.substring(groupByIndex + types_1.QueryClause.GROUP_BY.length);
|
154
|
+
const groupByPart = after.split(types_1.QueryClause.ORDER_BY)[0];
|
155
|
+
return groupByPart
|
156
|
+
.split(types_1.QueryClause.THEN_BY)
|
157
|
+
.map(lowerFirst)
|
158
|
+
.filter(Boolean);
|
159
|
+
}
|
160
|
+
// private static extractOrderBy(
|
161
|
+
// methodName: string
|
162
|
+
// ): OrderBySelector<any>[] | undefined {
|
163
|
+
// const orderByIndex = methodName.indexOf(QueryClause.ORDER_BY);
|
164
|
+
// if (orderByIndex === -1) return undefined;
|
165
|
+
//
|
166
|
+
// const after = methodName.substring(
|
167
|
+
// orderByIndex + QueryClause.ORDER_BY.length
|
168
|
+
// );
|
169
|
+
// const orderParts = after.split("ThenBy");
|
170
|
+
//
|
171
|
+
// return orderParts.map((part) => {
|
172
|
+
// const match = part.match(/(.*?)(Asc|Desc|Dsc)$/);
|
173
|
+
// if (!match) throw new Error(`Invalid OrderBy part: ${part}`);
|
174
|
+
// const [, field, dir] = match;
|
175
|
+
// return [
|
176
|
+
// lowerFirst(field),
|
177
|
+
// dir.toLowerCase() === "dsc"
|
178
|
+
// ? OrderDirection.DSC
|
179
|
+
// : (dir.toLowerCase() as OrderDirection),
|
180
|
+
// ];
|
181
|
+
// });
|
182
|
+
// }
|
183
|
+
/**
|
184
|
+
* @description
|
185
|
+
* Builds the `where` condition object based on the parsed core string and parameter values.
|
186
|
+
*
|
187
|
+
* @summary
|
188
|
+
* Splits the core string by logical operators (`And`, `Or`), parses each token into a field
|
189
|
+
* and operator, and combines them into a `Condition` object using the provided values.
|
190
|
+
*
|
191
|
+
* @param core {string} - The extracted core string from the method name.
|
192
|
+
* @param values {any[]} - The values corresponding to the conditions.
|
193
|
+
*
|
194
|
+
* @return {Condition<any>} A structured condition object representing the query's where clause.
|
195
|
+
*/
|
196
|
+
static buildWhere(core, values) {
|
197
|
+
const parts = core.split(/OrderBy|GroupBy/)[0] || "";
|
198
|
+
const conditions = parts.split(/And|Or/);
|
199
|
+
const operators = core.match(/And|Or/g) || [];
|
200
|
+
let where;
|
201
|
+
conditions.forEach((token, idx) => {
|
202
|
+
const { field, operator } = this.parseFieldAndOperator(token);
|
203
|
+
const parser = operator ? utils_1.OperatorsMap[operator] : utils_1.OperatorsMap.Equals;
|
204
|
+
if (!parser)
|
205
|
+
throw new Error(`Unsupported operator ${operator}`);
|
206
|
+
const conditionValue = values[idx];
|
207
|
+
if (typeof conditionValue === "undefined") {
|
208
|
+
throw new Error(`Invalid value for field ${field}`);
|
209
|
+
}
|
210
|
+
const condition = parser(field, conditionValue);
|
211
|
+
where =
|
212
|
+
idx === 0
|
213
|
+
? condition
|
214
|
+
: operators[idx - 1] === types_1.QueryClause.AND
|
215
|
+
? where.and(condition)
|
216
|
+
: where.or(condition);
|
217
|
+
});
|
218
|
+
if (!where)
|
219
|
+
throw new Error("No conditions found in method name");
|
220
|
+
return where;
|
221
|
+
}
|
222
|
+
/**
|
223
|
+
* @description
|
224
|
+
* Parses a field name and operator from a string token.
|
225
|
+
*
|
226
|
+
* @summary
|
227
|
+
* Identifies the operator suffix (if present) and returns a descriptor containing the field
|
228
|
+
* name in lowercase-first format along with the operator.
|
229
|
+
*
|
230
|
+
* @param str {string} - The token string to parse.
|
231
|
+
*
|
232
|
+
* @return {FilterDescriptor} An object containing the field name and operator.
|
233
|
+
*/
|
234
|
+
static parseFieldAndOperator(str) {
|
235
|
+
for (const operator of Object.keys(utils_1.OperatorsMap)) {
|
236
|
+
if (str.endsWith(operator)) {
|
237
|
+
const field = str.slice(0, -operator.length);
|
238
|
+
return { field: lowerFirst(field), operator };
|
239
|
+
}
|
240
|
+
}
|
241
|
+
return { field: lowerFirst(str) };
|
242
|
+
}
|
243
|
+
/**
|
244
|
+
* @description
|
245
|
+
* Extracts `orderBy`, `limit`, and `offset` clauses from method arguments.
|
246
|
+
*
|
247
|
+
* @summary
|
248
|
+
* Determines the number of condition arguments, then checks the remaining arguments
|
249
|
+
* to resolve sorting, limiting, and pagination.
|
250
|
+
*
|
251
|
+
* @param core {string} - The extracted core string from the method name.
|
252
|
+
* @param values {any[]} - The values corresponding to method arguments, including conditions and extras.
|
253
|
+
*
|
254
|
+
* @return {{
|
255
|
+
* orderBy?: OrderBySelector<any>[];
|
256
|
+
* limit?: number;
|
257
|
+
* offset?: number;
|
258
|
+
* }} An object containing orderBy, limit, and offset values if present.
|
259
|
+
*/
|
260
|
+
static extractOrderLimitOffset(core, values) {
|
261
|
+
const conditionCount = core.split(/And|Or/).length;
|
262
|
+
const extraArgs = values.slice(conditionCount);
|
263
|
+
let orderBy;
|
264
|
+
let limit;
|
265
|
+
let offset;
|
266
|
+
if (extraArgs.length >= 1 && Array.isArray(extraArgs[0]))
|
267
|
+
orderBy = extraArgs[0];
|
268
|
+
if (extraArgs.length >= 2 && typeof extraArgs[1] === "number")
|
269
|
+
limit = extraArgs[1];
|
270
|
+
if (extraArgs.length >= 3 && typeof extraArgs[2] === "number")
|
271
|
+
offset = extraArgs[2];
|
272
|
+
return { orderBy, limit, offset };
|
273
|
+
}
|
274
|
+
}
|
275
|
+
exports.MethodQueryBuilder = MethodQueryBuilder;
|
276
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"MethodQueryBuilder.js","sourceRoot":"","sources":["../../src/query/MethodQueryBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import { Condition, OrderBySelector } from \"../query\";\nimport { FilterDescriptor, QueryAssist, QueryClause } from \"./types\";\nimport { OperatorsMap } from \"./utils\";\n\nconst lowerFirst = (str: string): string =>\n  str.charAt(0).toLowerCase() + str.slice(1);\n\n/**\n * @description\n * Utility class to build query objects from repository method names.\n *\n * @summary\n * The `MethodQueryBuilder` class parses method names that follow a specific naming convention\n * (e.g., `findByNameAndAgeOrderByCountryAsc`) and converts them into structured query objects\n * (`QueryAssist`). It extracts clauses such as `select`, `where`, `groupBy`, `orderBy`, `limit`,\n * and `offset`, ensuring that developers can declare repository queries using expressive method names.\n *\n * @param methodName {string} - The repository method name to parse and convert into a query.\n * @param values {any[]} - The values corresponding to method parameters used for query conditions.\n *\n * @return {QueryAssist} A structured query object describing the parsed action, select, where,\n * groupBy, orderBy, limit, and offset clauses.\n *\n * @class\n *\n * @example\n * ```ts\n * const query = MethodQueryBuilder.build(\n *   \"findByNameAndAgeOrderByCountryAsc\",\n *   \"John\",\n *   25,\n *   [[\"country\", \"ASC\"]]\n * );\n *\n * console.log(query);\n * // {\n * //   action: \"find\",\n * //   select: undefined,\n * //   where: { ... },\n * //   groupBy: undefined,\n * //   orderBy: [[\"country\", \"ASC\"]],\n * //   limit: undefined,\n * //   offset: undefined\n * // }\n * ```\n *\n * @mermaid\n * sequenceDiagram\n *   participant Repo as Repository Method\n *   participant MQB as MethodQueryBuilder\n *   participant Query as QueryAssist\n *\n *   Repo->>MQB: build(methodName, ...values)\n *   MQB->>MQB: extractCore(methodName)\n *   MQB->>MQB: extractSelect(methodName)\n *   MQB->>MQB: extractGroupBy(methodName)\n *   MQB->>MQB: buildWhere(core, values)\n *   MQB->>MQB: extractOrderLimitOffset(core, values)\n *   MQB->>Query: return structured QueryAssist object\n */\nexport class MethodQueryBuilder {\n  /**\n   * @description\n   * Builds a `QueryAssist` object by parsing a repository method name and values.\n   *\n   * @summary\n   * The method validates the method name, extracts clauses (core, select, groupBy, where,\n   * orderBy, limit, and offset), and assembles them into a structured query object\n   * that can be executed against a data source.\n   *\n   * @param methodName {string} - The repository method name that encodes query information.\n   * @param values {any[]} - The values corresponding to conditions and extra clauses.\n   *\n   * @return {QueryAssist} A structured query object representing the parsed query.\n   */\n  static build(methodName: string, ...values: any[]): QueryAssist {\n    if (!methodName.startsWith(QueryClause.FIND_BY)) {\n      throw new Error(`Unsupported method ${methodName}`);\n    }\n\n    const core = this.extractCore(methodName);\n    const select = this.extractSelect(methodName);\n    const groupBy = this.extractGroupBy(methodName);\n    // const orderBy = this.extractOrderBy(methodName);\n    const where = this.buildWhere(core, values);\n    const { orderBy, limit, offset } = this.extractOrderLimitOffset(\n      core,\n      values\n    );\n\n    return {\n      action: \"find\",\n      select: select,\n      where,\n      groupBy,\n      orderBy,\n      limit,\n      offset,\n    };\n  }\n\n  /**\n   * @description\n   * Extracts the core part of the method name after `findBy` and before any special clauses.\n   *\n   * @summary\n   * Removes prefixes and detects delimiters (`Then`, `OrderBy`, `GroupBy`, `Limit`, `Offset`)\n   * to isolate the main conditional part of the query.\n   *\n   * @param methodName {string} - The method name to parse.\n   *\n   * @return {string} The extracted core string used for building conditions.\n   */\n  private static extractCore(methodName: string): string {\n    const afterFindBy = methodName.substring(QueryClause.FIND_BY.length);\n    const regex = /(Then[A-Z]|OrderBy|GroupBy|Limit|Offset)/;\n    const match = afterFindBy.match(regex);\n    return match ? afterFindBy.substring(0, match.index) : afterFindBy;\n  }\n\n  /**\n   * @description\n   * Extracts the select clause from a method name.\n   *\n   * @summary\n   * Detects the `Select` keyword in the method name, isolates the fields following it,\n   * and returns them as an array of lowercase-first strings.\n   *\n   * @param methodName {string} - The method name to parse.\n   *\n   * @return {string[] | undefined} An array of selected fields or `undefined` if no select clause exists.\n   */\n  private static extractSelect(methodName: string): string[] | undefined {\n    const selectIndex = methodName.indexOf(QueryClause.SELECT);\n    if (selectIndex === -1) return undefined;\n\n    const afterSelect = methodName.substring(\n      selectIndex + QueryClause.SELECT.length\n    );\n\n    // Search for next Then, GroupBy, OrderBy...\n    const match = afterSelect.match(/(Then[A-Z]|OrderBy|GroupBy|Limit|Offset)/);\n\n    const selectPart = match\n      ? afterSelect.substring(0, match.index)\n      : afterSelect;\n\n    return selectPart.split(QueryClause.AND).map(lowerFirst).filter(Boolean);\n  }\n\n  /**\n   * @description\n   * Extracts the group by clause from a method name.\n   *\n   * @summary\n   * Detects the `GroupBy` keyword in the method name, isolates the fields following it,\n   * and returns them as an array of lowercase-first strings.\n   *\n   * @param methodName {string} - The method name to parse.\n   *\n   * @return {string[] | undefined} An array of group by fields or `undefined` if no group by clause exists.\n   */\n  private static extractGroupBy(methodName: string): string[] | undefined {\n    const groupByIndex = methodName.indexOf(QueryClause.GROUP_BY);\n    if (groupByIndex === -1) return undefined;\n\n    const after = methodName.substring(\n      groupByIndex + QueryClause.GROUP_BY.length\n    );\n    const groupByPart = after.split(QueryClause.ORDER_BY)[0];\n    return groupByPart\n      .split(QueryClause.THEN_BY)\n      .map(lowerFirst)\n      .filter(Boolean);\n  }\n\n  // private static extractOrderBy(\n  //   methodName: string\n  // ): OrderBySelector<any>[] | undefined {\n  //   const orderByIndex = methodName.indexOf(QueryClause.ORDER_BY);\n  //   if (orderByIndex === -1) return undefined;\n  //\n  //   const after = methodName.substring(\n  //     orderByIndex + QueryClause.ORDER_BY.length\n  //   );\n  //   const orderParts = after.split(\"ThenBy\");\n  //\n  //   return orderParts.map((part) => {\n  //     const match = part.match(/(.*?)(Asc|Desc|Dsc)$/);\n  //     if (!match) throw new Error(`Invalid OrderBy part: ${part}`);\n  //     const [, field, dir] = match;\n  //     return [\n  //       lowerFirst(field),\n  //       dir.toLowerCase() === \"dsc\"\n  //         ? OrderDirection.DSC\n  //         : (dir.toLowerCase() as OrderDirection),\n  //     ];\n  //   });\n  // }\n\n  /**\n   * @description\n   * Builds the `where` condition object based on the parsed core string and parameter values.\n   *\n   * @summary\n   * Splits the core string by logical operators (`And`, `Or`), parses each token into a field\n   * and operator, and combines them into a `Condition` object using the provided values.\n   *\n   * @param core {string} - The extracted core string from the method name.\n   * @param values {any[]} - The values corresponding to the conditions.\n   *\n   * @return {Condition<any>} A structured condition object representing the query's where clause.\n   */\n  private static buildWhere(core: string, values: any[]): Condition<any> {\n    const parts = core.split(/OrderBy|GroupBy/)[0] || \"\";\n    const conditions = parts.split(/And|Or/);\n\n    const operators = core.match(/And|Or/g) || [];\n\n    let where: Condition<any> | undefined;\n\n    conditions.forEach((token, idx) => {\n      const { field, operator } = this.parseFieldAndOperator(token);\n      const parser = operator ? OperatorsMap[operator] : OperatorsMap.Equals;\n      if (!parser) throw new Error(`Unsupported operator ${operator}`);\n\n      const conditionValue = values[idx];\n      if (typeof conditionValue === \"undefined\") {\n        throw new Error(`Invalid value for field ${field}`);\n      }\n\n      const condition = parser(field, conditionValue);\n      where =\n        idx === 0\n          ? condition\n          : operators[idx - 1] === QueryClause.AND\n            ? where!.and(condition)\n            : where!.or(condition);\n    });\n\n    if (!where) throw new Error(\"No conditions found in method name\");\n    return where;\n  }\n\n  /**\n   * @description\n   * Parses a field name and operator from a string token.\n   *\n   * @summary\n   * Identifies the operator suffix (if present) and returns a descriptor containing the field\n   * name in lowercase-first format along with the operator.\n   *\n   * @param str {string} - The token string to parse.\n   *\n   * @return {FilterDescriptor} An object containing the field name and operator.\n   */\n  private static parseFieldAndOperator(str: string): FilterDescriptor {\n    for (const operator of Object.keys(OperatorsMap)) {\n      if (str.endsWith(operator)) {\n        const field = str.slice(0, -operator.length);\n        return { field: lowerFirst(field), operator };\n      }\n    }\n    return { field: lowerFirst(str) };\n  }\n\n  /**\n   * @description\n   * Extracts `orderBy`, `limit`, and `offset` clauses from method arguments.\n   *\n   * @summary\n   * Determines the number of condition arguments, then checks the remaining arguments\n   * to resolve sorting, limiting, and pagination.\n   *\n   * @param core {string} - The extracted core string from the method name.\n   * @param values {any[]} - The values corresponding to method arguments, including conditions and extras.\n   *\n   * @return {{\n   *   orderBy?: OrderBySelector<any>[];\n   *   limit?: number;\n   *   offset?: number;\n   * }} An object containing orderBy, limit, and offset values if present.\n   */\n  private static extractOrderLimitOffset(\n    core: string,\n    values: any[]\n  ): { orderBy?: OrderBySelector<any>[]; limit?: number; offset?: number } {\n    const conditionCount = core.split(/And|Or/).length;\n    const extraArgs = values.slice(conditionCount);\n\n    let orderBy: OrderBySelector<any>[] | undefined;\n    let limit: number | undefined;\n    let offset: number | undefined;\n\n    if (extraArgs.length >= 1 && Array.isArray(extraArgs[0]))\n      orderBy = extraArgs[0] as OrderBySelector<any>[];\n\n    if (extraArgs.length >= 2 && typeof extraArgs[1] === \"number\")\n      limit = extraArgs[1];\n\n    if (extraArgs.length >= 3 && typeof extraArgs[2] === \"number\")\n      offset = extraArgs[2];\n\n    return { orderBy, limit, offset };\n  }\n}\n"]}
|