@woltz/rich-domain 0.2.2 → 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (109) hide show
  1. package/CHANGELOG.md +30 -0
  2. package/README.md +37 -20
  3. package/dist/base-entity.d.ts +1 -1
  4. package/dist/base-entity.d.ts.map +1 -1
  5. package/dist/base-entity.js +21 -15
  6. package/dist/base-entity.js.map +1 -1
  7. package/dist/constants.js +4 -1
  8. package/dist/constants.js.map +1 -1
  9. package/dist/criteria.d.ts.map +1 -1
  10. package/dist/criteria.js +7 -3
  11. package/dist/criteria.js.map +1 -1
  12. package/dist/deep-proxy.d.ts +3 -1
  13. package/dist/deep-proxy.d.ts.map +1 -1
  14. package/dist/deep-proxy.js +110 -33
  15. package/dist/deep-proxy.js.map +1 -1
  16. package/dist/domain-event-bus.js +7 -2
  17. package/dist/domain-event-bus.js.map +1 -1
  18. package/dist/domain-event.js +7 -3
  19. package/dist/domain-event.js.map +1 -1
  20. package/dist/entity.js +8 -3
  21. package/dist/entity.js.map +1 -1
  22. package/dist/id.d.ts +3 -3
  23. package/dist/id.d.ts.map +1 -1
  24. package/dist/id.js +10 -6
  25. package/dist/id.js.map +1 -1
  26. package/dist/index.d.ts +0 -1
  27. package/dist/index.d.ts.map +1 -1
  28. package/dist/index.js +39 -17
  29. package/dist/index.js.map +1 -1
  30. package/dist/mapper.js +5 -1
  31. package/dist/mapper.js.map +1 -1
  32. package/dist/paginated-result.js +7 -3
  33. package/dist/paginated-result.js.map +1 -1
  34. package/dist/repository/base-repository.js +11 -4
  35. package/dist/repository/base-repository.js.map +1 -1
  36. package/dist/repository/in-memory-repository.js +8 -4
  37. package/dist/repository/in-memory-repository.js.map +1 -1
  38. package/dist/repository/index.d.ts +1 -37
  39. package/dist/repository/index.d.ts.map +1 -1
  40. package/dist/repository/index.js +26 -40
  41. package/dist/repository/index.js.map +1 -1
  42. package/dist/repository/unit-of-work.js +9 -3
  43. package/dist/repository/unit-of-work.js.map +1 -1
  44. package/dist/types/criteria.d.ts +2 -2
  45. package/dist/types/criteria.d.ts.map +1 -1
  46. package/dist/types/criteria.js +4 -1
  47. package/dist/types/criteria.js.map +1 -1
  48. package/dist/types/domain.js +2 -1
  49. package/dist/types/history-tracker.js +2 -1
  50. package/dist/types/index.js +22 -6
  51. package/dist/types/index.js.map +1 -1
  52. package/dist/types/standard-schema.js +2 -1
  53. package/dist/types/unit-of-work.js +2 -1
  54. package/dist/types/utils.js +2 -1
  55. package/dist/validation-error.js +9 -3
  56. package/dist/validation-error.js.map +1 -1
  57. package/dist/value-object.js +9 -5
  58. package/dist/value-object.js.map +1 -1
  59. package/package.json +1 -1
  60. package/src/base-entity.ts +3 -2
  61. package/src/criteria.ts +2 -2
  62. package/src/deep-proxy.ts +435 -339
  63. package/src/id.ts +4 -4
  64. package/src/index.ts +1 -3
  65. package/src/repository/index.ts +1 -38
  66. package/src/types/criteria.ts +2 -2
  67. package/tests/entity-validation.test.ts +1 -1
  68. package/tests/history-tracker.spec.ts +57 -17
  69. package/tests/id.test.ts +341 -341
  70. package/tests/repository.test.ts +4 -2
  71. package/tests/to-json.test.ts +103 -91
  72. package/tests/value-objects.test.ts +52 -52
  73. package/tsconfig.json +2 -2
  74. package/dist/filtering.d.ts +0 -107
  75. package/dist/filtering.d.ts.map +0 -1
  76. package/dist/filtering.js +0 -202
  77. package/dist/filtering.js.map +0 -1
  78. package/dist/ordering.d.ts +0 -93
  79. package/dist/ordering.d.ts.map +0 -1
  80. package/dist/ordering.js +0 -154
  81. package/dist/ordering.js.map +0 -1
  82. package/dist/pagination.d.ts +0 -218
  83. package/dist/pagination.d.ts.map +0 -1
  84. package/dist/pagination.js +0 -281
  85. package/dist/pagination.js.map +0 -1
  86. package/dist/repository/mapper.d.ts +0 -56
  87. package/dist/repository/mapper.d.ts.map +0 -1
  88. package/dist/repository/mapper.js +0 -15
  89. package/dist/repository/mapper.js.map +0 -1
  90. package/dist/repository/types.d.ts +0 -87
  91. package/dist/repository/types.d.ts.map +0 -1
  92. package/dist/repository/types.js +0 -6
  93. package/dist/repository/types.js.map +0 -1
  94. package/dist/repository.d.ts +0 -2
  95. package/dist/repository.d.ts.map +0 -1
  96. package/dist/repository.js +0 -21
  97. package/dist/repository.js.map +0 -1
  98. package/dist/specification.d.ts +0 -102
  99. package/dist/specification.d.ts.map +0 -1
  100. package/dist/specification.js +0 -187
  101. package/dist/specification.js.map +0 -1
  102. package/dist/types/repository.d.ts +0 -43
  103. package/dist/types/repository.d.ts.map +0 -1
  104. package/dist/types/repository.js +0 -2
  105. package/dist/types/repository.js.map +0 -1
  106. package/dist/types.d.ts +0 -88
  107. package/dist/types.d.ts.map +0 -1
  108. package/dist/types.js +0 -12
  109. package/dist/types.js.map +0 -1
@@ -482,7 +482,7 @@ describe("Repository", () => {
482
482
  });
483
483
 
484
484
  it("should save new user (insert)", async () => {
485
- expect(user.isNew).toBe(true);
485
+ expect(user.isNew()).toBe(true);
486
486
  await repository.create(user);
487
487
 
488
488
  const found = await repository.findById(user.id.value);
@@ -644,7 +644,9 @@ describe("Repository", () => {
644
644
  }),
645
645
  ];
646
646
 
647
- const persistenceList = domainList.map((d) => toPersistenceMapper.build(d));
647
+ const persistenceList = domainList.map((d) =>
648
+ toPersistenceMapper.build(d)
649
+ );
648
650
 
649
651
  expect(persistenceList).toHaveLength(2);
650
652
  expect(persistenceList[0].id).toBe("1");
@@ -1,91 +1,103 @@
1
- import { Id } from "../src";
2
- import { Post, User, Address, Comment } from "./utils";
3
-
4
- describe("toJson Functionality", () => {
5
- it("should convert simple entity to JSON", () => {
6
- const post = new Post({
7
- id: new Id("1"),
8
- title: "First Post",
9
- content: "Hello World",
10
- likes: 5,
11
- });
12
-
13
- const json = post.toJson();
14
- expect(json).toEqual({
15
- id: "1",
16
- title: "First Post",
17
- content: "Hello World",
18
- likes: 5,
19
- });
20
- });
21
-
22
- it("should convert nested entities to JSON", () => {
23
- const user = new User({
24
- id: new Id("1"),
25
- name: "John Doe",
26
- email: "john@example.com",
27
- posts: [
28
- new Post({
29
- id: new Id("1"),
30
- title: "Post 1",
31
- content: "Content 1",
32
- likes: 0,
33
- }),
34
- new Post({
35
- id: new Id("2"),
36
- title: "Post 2",
37
- content: "Content 2",
38
- likes: 5,
39
- }),
40
- ],
41
- address: new Address({
42
- street: "Main St",
43
- city: "NYC",
44
- zipCode: "10001",
45
- }),
46
- comments: [
47
- new Comment({
48
- id: new Id("1"),
49
- text: "Great post!",
50
- author: "Alice",
51
- }),
52
- ],
53
- });
54
-
55
- const json = user.toJson();
56
-
57
- expect(json.id).toBe("1");
58
- expect(json.name).toBe("John Doe");
59
- expect(json.posts).toHaveLength(2);
60
- expect(json.posts[0].title).toBe("Post 1");
61
- expect(json.address.city).toBe("NYC");
62
- expect(json.comments[0].text).toBe("Great post!");
63
- });
64
-
65
- it("should handle deeply nested structures", () => {
66
- const user = new User({
67
- id: new Id("1"),
68
- name: "John Doe",
69
- email: "john@example.com",
70
- posts: [
71
- new Post({
72
- id: new Id("1"),
73
- title: "Post 1",
74
- content: "Content 1",
75
- likes: 0,
76
- }),
77
- ],
78
- address: new Address({
79
- street: "Main St",
80
- city: "NYC",
81
- zipCode: "10001",
82
- }),
83
- comments: [],
84
- });
85
-
86
- const json = user.toJson();
87
- expect(typeof json).toBe("object");
88
- expect(Array.isArray(json.posts)).toBe(true);
89
- expect(json.posts[0].id).toBe("1");
90
- });
91
- });
1
+ import { Entity, Id } from "../src";
2
+ import { Post, User, Address, Comment } from "./utils";
3
+
4
+ describe("toJson Functionality", () => {
5
+ it("should convert simple entity to JSON", () => {
6
+ const post = new Post({
7
+ id: new Id("1"),
8
+ title: "First Post",
9
+ content: "Hello World",
10
+ likes: 5,
11
+ });
12
+
13
+ const json = post.toJson();
14
+ expect(json).toEqual({
15
+ id: "1",
16
+ title: "First Post",
17
+ content: "Hello World",
18
+ likes: 5,
19
+ });
20
+ });
21
+
22
+ it("should convert nested entities to JSON", () => {
23
+ const user = new User({
24
+ id: new Id("1"),
25
+ name: "John Doe",
26
+ email: "john@example.com",
27
+ posts: [
28
+ new Post({
29
+ id: new Id("1"),
30
+ title: "Post 1",
31
+ content: "Content 1",
32
+ likes: 0,
33
+ }),
34
+ new Post({
35
+ id: new Id("2"),
36
+ title: "Post 2",
37
+ content: "Content 2",
38
+ likes: 5,
39
+ }),
40
+ ],
41
+ address: new Address({
42
+ street: "Main St",
43
+ city: "NYC",
44
+ zipCode: "10001",
45
+ }),
46
+ comments: [
47
+ new Comment({
48
+ id: new Id("1"),
49
+ text: "Great post!",
50
+ author: "Alice",
51
+ }),
52
+ ],
53
+ });
54
+
55
+ const json = user.toJson();
56
+
57
+ expect(json.id).toBe("1");
58
+ expect(json.name).toBe("John Doe");
59
+ expect(json.posts).toHaveLength(2);
60
+ expect(json.posts[0].title).toBe("Post 1");
61
+ expect(json.address.city).toBe("NYC");
62
+ expect(json.comments[0].text).toBe("Great post!");
63
+ });
64
+
65
+ it("should handle deeply nested structures", () => {
66
+ const user = new User({
67
+ id: new Id("1"),
68
+ name: "John Doe",
69
+ email: "john@example.com",
70
+ posts: [
71
+ new Post({
72
+ id: new Id("1"),
73
+ title: "Post 1",
74
+ content: "Content 1",
75
+ likes: 0,
76
+ }),
77
+ ],
78
+ address: new Address({
79
+ street: "Main St",
80
+ city: "NYC",
81
+ zipCode: "10001",
82
+ }),
83
+ comments: [],
84
+ });
85
+
86
+ const json = user.toJson();
87
+ expect(typeof json).toBe("object");
88
+ expect(Array.isArray(json.posts)).toBe(true);
89
+ expect(json.posts[0].id).toBe("1");
90
+ });
91
+
92
+ it("should serialize date correctly", () => {
93
+ class Test extends Entity<{ id: Id; createdAt: Date }> {}
94
+
95
+ const test = new Test({
96
+ id: new Id("1"),
97
+ createdAt: new Date(),
98
+ });
99
+
100
+ const json = test.toJson();
101
+ expect(json.createdAt).toBe(test.props.createdAt.toISOString());
102
+ });
103
+ });
@@ -1,52 +1,52 @@
1
- import { Address } from "./utils";
2
-
3
- describe("Value Object", () => {
4
- it("should create immutable value object", () => {
5
- const address = new Address({
6
- street: "Main St",
7
- city: "NYC",
8
- zipCode: "10001",
9
- });
10
-
11
- expect(address.street).toBe("Main St");
12
- expect(address.city).toBe("NYC");
13
- });
14
-
15
- it("should compare value objects by value", () => {
16
- const address1 = new Address({
17
- street: "Main St",
18
- city: "NYC",
19
- zipCode: "10001",
20
- });
21
-
22
- const address2 = new Address({
23
- street: "Main St",
24
- city: "NYC",
25
- zipCode: "10001",
26
- });
27
-
28
- const address3 = new Address({
29
- street: "Broadway",
30
- city: "NYC",
31
- zipCode: "10001",
32
- });
33
-
34
- expect(address1.equals(address2)).toBe(true);
35
- expect(address1.equals(address3)).toBe(false);
36
- });
37
-
38
- it("should convert value object to JSON", () => {
39
- const address = new Address({
40
- street: "Main St",
41
- city: "NYC",
42
- zipCode: "10001",
43
- });
44
-
45
- const json = address.toJson();
46
- expect(json).toEqual({
47
- street: "Main St",
48
- city: "NYC",
49
- zipCode: "10001",
50
- });
51
- });
52
- });
1
+ import { Address } from "./utils";
2
+
3
+ describe("Value Object", () => {
4
+ it("should create immutable value object", () => {
5
+ const address = new Address({
6
+ street: "Main St",
7
+ city: "NYC",
8
+ zipCode: "10001",
9
+ });
10
+
11
+ expect(address.street).toBe("Main St");
12
+ expect(address.city).toBe("NYC");
13
+ });
14
+
15
+ it("should compare value objects by value", () => {
16
+ const address1 = new Address({
17
+ street: "Main St",
18
+ city: "NYC",
19
+ zipCode: "10001",
20
+ });
21
+
22
+ const address2 = new Address({
23
+ street: "Main St",
24
+ city: "NYC",
25
+ zipCode: "10001",
26
+ });
27
+
28
+ const address3 = new Address({
29
+ street: "Broadway",
30
+ city: "NYC",
31
+ zipCode: "10001",
32
+ });
33
+
34
+ expect(address1.equals(address2)).toBe(true);
35
+ expect(address1.equals(address3)).toBe(false);
36
+ });
37
+
38
+ it("should convert value object to JSON", () => {
39
+ const address = new Address({
40
+ street: "Main St",
41
+ city: "NYC",
42
+ zipCode: "10001",
43
+ });
44
+
45
+ const json = address.toJson();
46
+ expect(json).toEqual({
47
+ street: "Main St",
48
+ city: "NYC",
49
+ zipCode: "10001",
50
+ });
51
+ });
52
+ });
package/tsconfig.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "compilerOptions": {
3
3
  "target": "ES2020",
4
- "module": "esnext",
4
+ "module": "commonjs",
5
5
  "lib": ["ES2020"],
6
6
  "declaration": true,
7
7
  "declarationMap": true,
@@ -20,7 +20,7 @@
20
20
  "noUnusedParameters": false,
21
21
  "noImplicitReturns": true,
22
22
  "noFallthroughCasesInSwitch": true,
23
- "moduleResolution": "bundler",
23
+ "moduleResolution": "node",
24
24
  "esModuleInterop": true,
25
25
  "skipLibCheck": true,
26
26
  "forceConsistentCasingInFileNames": true,
@@ -1,107 +0,0 @@
1
- /**
2
- * Supported filter operators
3
- */
4
- export type FilterOperator = "equals" | "notEquals" | "greaterThan" | "greaterThanOrEquals" | "lessThan" | "lessThanOrEquals" | "in" | "notIn" | "contains" | "startsWith" | "endsWith" | "isNull" | "isNotNull";
5
- /**
6
- * Configuration for a single filter
7
- */
8
- export interface FilterConfig<T> {
9
- field: keyof T | string;
10
- operator: FilterOperator;
11
- value?: any;
12
- }
13
- /**
14
- * Represents a single filter condition
15
- * Can be used for in-memory filtering or converted to database-specific queries
16
- *
17
- * @example
18
- * ```typescript
19
- * const filter = new Filter('age', 'greaterThan', 18);
20
- * const adults = users.filter(filter.predicateFn());
21
- * ```
22
- */
23
- export declare class Filter<T> {
24
- private readonly _field;
25
- private readonly _operator;
26
- private readonly _value?;
27
- constructor(field: keyof T | string, operator: FilterOperator, value?: any);
28
- /**
29
- * Gets the field to filter
30
- */
31
- get field(): keyof T | string;
32
- /**
33
- * Gets the filter operator
34
- */
35
- get operator(): FilterOperator;
36
- /**
37
- * Gets the filter value
38
- */
39
- get value(): any;
40
- /**
41
- * Returns configuration object for serialization
42
- */
43
- toConfig(): FilterConfig<T>;
44
- /**
45
- * Checks if the operator requires a value
46
- * @private
47
- */
48
- private requiresValue;
49
- /**
50
- * Creates a predicate function for in-memory filtering
51
- * Handles nested properties using dot notation (e.g., "address.city")
52
- *
53
- * @example
54
- * ```typescript
55
- * const filter = new Filter<User>('age', 'greaterThan', 18);
56
- * const adults = users.filter(filter.predicateFn());
57
- * ```
58
- */
59
- predicateFn(): (item: T) => boolean;
60
- /**
61
- * Gets nested property value using dot notation
62
- * @private
63
- */
64
- private getNestedValue;
65
- }
66
- /**
67
- * Logical operator for combining filters
68
- */
69
- export type LogicalOperator = "and" | "or";
70
- /**
71
- * Represents a composite filter that combines multiple filters with AND/OR logic
72
- *
73
- * @example
74
- * ```typescript
75
- * const composite = new CompositeFilter<User>('and', [
76
- * new Filter('age', 'greaterThan', 18),
77
- * new Filter('status', 'equals', 'active')
78
- * ]);
79
- * const filtered = users.filter(composite.predicateFn());
80
- * ```
81
- */
82
- export declare class CompositeFilter<T> {
83
- private readonly _operator;
84
- private readonly _filters;
85
- constructor(operator: LogicalOperator, filters: (Filter<T> | CompositeFilter<T>)[]);
86
- /**
87
- * Gets the logical operator
88
- */
89
- get operator(): LogicalOperator;
90
- /**
91
- * Gets all filters
92
- */
93
- get filters(): readonly (Filter<T> | CompositeFilter<T>)[];
94
- /**
95
- * Creates a predicate function that applies the logical operator
96
- */
97
- predicateFn(): (item: T) => boolean;
98
- /**
99
- * Adds another filter with AND logic
100
- */
101
- and(filter: Filter<T> | CompositeFilter<T>): CompositeFilter<T>;
102
- /**
103
- * Adds another filter with OR logic
104
- */
105
- or(filter: Filter<T> | CompositeFilter<T>): CompositeFilter<T>;
106
- }
107
- //# sourceMappingURL=filtering.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"filtering.d.ts","sourceRoot":"","sources":["../src/filtering.ts"],"names":[],"mappings":"AAMA;;GAEG;AACH,MAAM,MAAM,cAAc,GACtB,QAAQ,GACR,WAAW,GACX,aAAa,GACb,qBAAqB,GACrB,UAAU,GACV,kBAAkB,GAClB,IAAI,GACJ,OAAO,GACP,UAAU,GACV,YAAY,GACZ,UAAU,GACV,QAAQ,GACR,WAAW,CAAC;AAEhB;;GAEG;AACH,MAAM,WAAW,YAAY,CAAC,CAAC;IAC7B,KAAK,EAAE,MAAM,CAAC,GAAG,MAAM,CAAC;IACxB,QAAQ,EAAE,cAAc,CAAC;IACzB,KAAK,CAAC,EAAE,GAAG,CAAC;CACb;AAMD;;;;;;;;;GASG;AACH,qBAAa,MAAM,CAAC,CAAC;IACnB,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAmB;IAC1C,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAiB;IAC3C,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAM;gBAElB,KAAK,EAAE,MAAM,CAAC,GAAG,MAAM,EAAE,QAAQ,EAAE,cAAc,EAAE,KAAK,CAAC,EAAE,GAAG;IAW1E;;OAEG;IACH,IAAI,KAAK,IAAI,MAAM,CAAC,GAAG,MAAM,CAE5B;IAED;;OAEG;IACH,IAAI,QAAQ,IAAI,cAAc,CAE7B;IAED;;OAEG;IACH,IAAI,KAAK,IAAI,GAAG,CAEf;IAED;;OAEG;IACH,QAAQ,IAAI,YAAY,CAAC,CAAC,CAAC;IAQ3B;;;OAGG;IACH,OAAO,CAAC,aAAa;IAIrB;;;;;;;;;OASG;IACH,WAAW,IAAI,CAAC,IAAI,EAAE,CAAC,KAAK,OAAO;IA+DnC;;;OAGG;IACH,OAAO,CAAC,cAAc;CAGvB;AAMD;;GAEG;AACH,MAAM,MAAM,eAAe,GAAG,KAAK,GAAG,IAAI,CAAC;AAE3C;;;;;;;;;;;GAWG;AACH,qBAAa,eAAe,CAAC,CAAC;IAC5B,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAkB;IAC5C,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAqC;gBAElD,QAAQ,EAAE,eAAe,EAAE,OAAO,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,GAAG,eAAe,CAAC,CAAC,CAAC,CAAC,EAAE;IAQlF;;OAEG;IACH,IAAI,QAAQ,IAAI,eAAe,CAE9B;IAED;;OAEG;IACH,IAAI,OAAO,IAAI,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC,GAAG,eAAe,CAAC,CAAC,CAAC,CAAC,EAAE,CAEzD;IAED;;OAEG;IACH,WAAW,IAAI,CAAC,IAAI,EAAE,CAAC,KAAK,OAAO;IAWnC;;OAEG;IACH,GAAG,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,CAAC,GAAG,eAAe,CAAC,CAAC,CAAC,GAAG,eAAe,CAAC,CAAC,CAAC;IAU/D;;OAEG;IACH,EAAE,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,CAAC,GAAG,eAAe,CAAC,CAAC,CAAC,GAAG,eAAe,CAAC,CAAC,CAAC;CAS/D"}
package/dist/filtering.js DELETED
@@ -1,202 +0,0 @@
1
- "use strict";
2
- // ============================================================================
3
- // Filtering System
4
- // ============================================================================
5
- // Provides filtering/where clause capabilities for queries and in-memory collections
6
- // Supports common comparison operators and can be extended for custom operators
7
- Object.defineProperty(exports, "__esModule", { value: true });
8
- exports.CompositeFilter = exports.Filter = void 0;
9
- // ============================================================================
10
- // Filter Class
11
- // ============================================================================
12
- /**
13
- * Represents a single filter condition
14
- * Can be used for in-memory filtering or converted to database-specific queries
15
- *
16
- * @example
17
- * ```typescript
18
- * const filter = new Filter('age', 'greaterThan', 18);
19
- * const adults = users.filter(filter.predicateFn());
20
- * ```
21
- */
22
- class Filter {
23
- constructor(field, operator, value) {
24
- this._field = field;
25
- this._operator = operator;
26
- this._value = value;
27
- // Validate that value is provided when required
28
- if (this.requiresValue() && value === undefined) {
29
- throw new Error(`Filter operator '${operator}' requires a value`);
30
- }
31
- }
32
- /**
33
- * Gets the field to filter
34
- */
35
- get field() {
36
- return this._field;
37
- }
38
- /**
39
- * Gets the filter operator
40
- */
41
- get operator() {
42
- return this._operator;
43
- }
44
- /**
45
- * Gets the filter value
46
- */
47
- get value() {
48
- return this._value;
49
- }
50
- /**
51
- * Returns configuration object for serialization
52
- */
53
- toConfig() {
54
- return {
55
- field: this._field,
56
- operator: this._operator,
57
- value: this._value,
58
- };
59
- }
60
- /**
61
- * Checks if the operator requires a value
62
- * @private
63
- */
64
- requiresValue() {
65
- return this._operator !== "isNull" && this._operator !== "isNotNull";
66
- }
67
- /**
68
- * Creates a predicate function for in-memory filtering
69
- * Handles nested properties using dot notation (e.g., "address.city")
70
- *
71
- * @example
72
- * ```typescript
73
- * const filter = new Filter<User>('age', 'greaterThan', 18);
74
- * const adults = users.filter(filter.predicateFn());
75
- * ```
76
- */
77
- predicateFn() {
78
- return (item) => {
79
- const fieldValue = this.getNestedValue(item, this._field);
80
- switch (this._operator) {
81
- case "equals":
82
- return fieldValue === this._value;
83
- case "notEquals":
84
- return fieldValue !== this._value;
85
- case "greaterThan":
86
- return fieldValue > this._value;
87
- case "greaterThanOrEquals":
88
- return fieldValue >= this._value;
89
- case "lessThan":
90
- return fieldValue < this._value;
91
- case "lessThanOrEquals":
92
- return fieldValue <= this._value;
93
- case "in":
94
- return Array.isArray(this._value) && this._value.includes(fieldValue);
95
- case "notIn":
96
- return Array.isArray(this._value) && !this._value.includes(fieldValue);
97
- case "contains":
98
- return (typeof fieldValue === "string" &&
99
- typeof this._value === "string" &&
100
- fieldValue.includes(this._value));
101
- case "startsWith":
102
- return (typeof fieldValue === "string" &&
103
- typeof this._value === "string" &&
104
- fieldValue.startsWith(this._value));
105
- case "endsWith":
106
- return (typeof fieldValue === "string" &&
107
- typeof this._value === "string" &&
108
- fieldValue.endsWith(this._value));
109
- case "isNull":
110
- return fieldValue === null || fieldValue === undefined;
111
- case "isNotNull":
112
- return fieldValue !== null && fieldValue !== undefined;
113
- default:
114
- // This should never happen with proper typing
115
- throw new Error(`Unknown filter operator: ${this._operator}`);
116
- }
117
- };
118
- }
119
- /**
120
- * Gets nested property value using dot notation
121
- * @private
122
- */
123
- getNestedValue(obj, path) {
124
- return path.split(".").reduce((current, prop) => current?.[prop], obj);
125
- }
126
- }
127
- exports.Filter = Filter;
128
- /**
129
- * Represents a composite filter that combines multiple filters with AND/OR logic
130
- *
131
- * @example
132
- * ```typescript
133
- * const composite = new CompositeFilter<User>('and', [
134
- * new Filter('age', 'greaterThan', 18),
135
- * new Filter('status', 'equals', 'active')
136
- * ]);
137
- * const filtered = users.filter(composite.predicateFn());
138
- * ```
139
- */
140
- class CompositeFilter {
141
- constructor(operator, filters) {
142
- if (filters.length === 0) {
143
- throw new Error("At least one filter is required");
144
- }
145
- this._operator = operator;
146
- this._filters = filters;
147
- }
148
- /**
149
- * Gets the logical operator
150
- */
151
- get operator() {
152
- return this._operator;
153
- }
154
- /**
155
- * Gets all filters
156
- */
157
- get filters() {
158
- return this._filters;
159
- }
160
- /**
161
- * Creates a predicate function that applies the logical operator
162
- */
163
- predicateFn() {
164
- return (item) => {
165
- if (this._operator === "and") {
166
- return this._filters.every((filter) => filter.predicateFn()(item));
167
- }
168
- else {
169
- // OR
170
- return this._filters.some((filter) => filter.predicateFn()(item));
171
- }
172
- };
173
- }
174
- /**
175
- * Adds another filter with AND logic
176
- */
177
- and(filter) {
178
- if (this._operator === "and") {
179
- // Flatten if already AND
180
- return new CompositeFilter("and", [...this._filters, filter]);
181
- }
182
- else {
183
- // Wrap in new AND
184
- return new CompositeFilter("and", [this, filter]);
185
- }
186
- }
187
- /**
188
- * Adds another filter with OR logic
189
- */
190
- or(filter) {
191
- if (this._operator === "or") {
192
- // Flatten if already OR
193
- return new CompositeFilter("or", [...this._filters, filter]);
194
- }
195
- else {
196
- // Wrap in new OR
197
- return new CompositeFilter("or", [this, filter]);
198
- }
199
- }
200
- }
201
- exports.CompositeFilter = CompositeFilter;
202
- //# sourceMappingURL=filtering.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"filtering.js","sourceRoot":"","sources":["../src/filtering.ts"],"names":[],"mappings":";AAAA,+EAA+E;AAC/E,mBAAmB;AACnB,+EAA+E;AAC/E,qFAAqF;AACrF,gFAAgF;;;AA6BhF,+EAA+E;AAC/E,eAAe;AACf,+EAA+E;AAE/E;;;;;;;;;GASG;AACH,MAAa,MAAM;IAKjB,YAAY,KAAuB,EAAE,QAAwB,EAAE,KAAW;QACxE,IAAI,CAAC,MAAM,GAAG,KAAK,CAAC;QACpB,IAAI,CAAC,SAAS,GAAG,QAAQ,CAAC;QAC1B,IAAI,CAAC,MAAM,GAAG,KAAK,CAAC;QAEpB,gDAAgD;QAChD,IAAI,IAAI,CAAC,aAAa,EAAE,IAAI,KAAK,KAAK,SAAS,EAAE,CAAC;YAChD,MAAM,IAAI,KAAK,CAAC,oBAAoB,QAAQ,oBAAoB,CAAC,CAAC;QACpE,CAAC;IACH,CAAC;IAED;;OAEG;IACH,IAAI,KAAK;QACP,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;IAED;;OAEG;IACH,IAAI,QAAQ;QACV,OAAO,IAAI,CAAC,SAAS,CAAC;IACxB,CAAC;IAED;;OAEG;IACH,IAAI,KAAK;QACP,OAAO,IAAI,CAAC,MAAM,CAAC;IACrB,CAAC;IAED;;OAEG;IACH,QAAQ;QACN,OAAO;YACL,KAAK,EAAE,IAAI,CAAC,MAAM;YAClB,QAAQ,EAAE,IAAI,CAAC,SAAS;YACxB,KAAK,EAAE,IAAI,CAAC,MAAM;SACnB,CAAC;IACJ,CAAC;IAED;;;OAGG;IACK,aAAa;QACnB,OAAO,IAAI,CAAC,SAAS,KAAK,QAAQ,IAAI,IAAI,CAAC,SAAS,KAAK,WAAW,CAAC;IACvE,CAAC;IAED;;;;;;;;;OASG;IACH,WAAW;QACT,OAAO,CAAC,IAAO,EAAW,EAAE;YAC1B,MAAM,UAAU,GAAG,IAAI,CAAC,cAAc,CAAC,IAAI,EAAE,IAAI,CAAC,MAAgB,CAAC,CAAC;YAEpE,QAAQ,IAAI,CAAC,SAAS,EAAE,CAAC;gBACvB,KAAK,QAAQ;oBACX,OAAO,UAAU,KAAK,IAAI,CAAC,MAAM,CAAC;gBAEpC,KAAK,WAAW;oBACd,OAAO,UAAU,KAAK,IAAI,CAAC,MAAM,CAAC;gBAEpC,KAAK,aAAa;oBAChB,OAAO,UAAU,GAAG,IAAI,CAAC,MAAM,CAAC;gBAElC,KAAK,qBAAqB;oBACxB,OAAO,UAAU,IAAI,IAAI,CAAC,MAAM,CAAC;gBAEnC,KAAK,UAAU;oBACb,OAAO,UAAU,GAAG,IAAI,CAAC,MAAM,CAAC;gBAElC,KAAK,kBAAkB;oBACrB,OAAO,UAAU,IAAI,IAAI,CAAC,MAAM,CAAC;gBAEnC,KAAK,IAAI;oBACP,OAAO,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAExE,KAAK,OAAO;oBACV,OAAO,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;gBAEzE,KAAK,UAAU;oBACb,OAAO,CACL,OAAO,UAAU,KAAK,QAAQ;wBAC9B,OAAO,IAAI,CAAC,MAAM,KAAK,QAAQ;wBAC/B,UAAU,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,CACjC,CAAC;gBAEJ,KAAK,YAAY;oBACf,OAAO,CACL,OAAO,UAAU,KAAK,QAAQ;wBAC9B,OAAO,IAAI,CAAC,MAAM,KAAK,QAAQ;wBAC/B,UAAU,CAAC,UAAU,CAAC,IAAI,CAAC,MAAM,CAAC,CACnC,CAAC;gBAEJ,KAAK,UAAU;oBACb,OAAO,CACL,OAAO,UAAU,KAAK,QAAQ;wBAC9B,OAAO,IAAI,CAAC,MAAM,KAAK,QAAQ;wBAC/B,UAAU,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,CACjC,CAAC;gBAEJ,KAAK,QAAQ;oBACX,OAAO,UAAU,KAAK,IAAI,IAAI,UAAU,KAAK,SAAS,CAAC;gBAEzD,KAAK,WAAW;oBACd,OAAO,UAAU,KAAK,IAAI,IAAI,UAAU,KAAK,SAAS,CAAC;gBAEzD;oBACE,8CAA8C;oBAC9C,MAAM,IAAI,KAAK,CAAC,4BAA4B,IAAI,CAAC,SAAS,EAAE,CAAC,CAAC;YAClE,CAAC;QACH,CAAC,CAAC;IACJ,CAAC;IAED;;;OAGG;IACK,cAAc,CAAC,GAAQ,EAAE,IAAY;QAC3C,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,CAAC,OAAO,EAAE,IAAI,EAAE,EAAE,CAAC,OAAO,EAAE,CAAC,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC;IACzE,CAAC;CACF;AAxID,wBAwIC;AAWD;;;;;;;;;;;GAWG;AACH,MAAa,eAAe;IAI1B,YAAY,QAAyB,EAAE,OAA2C;QAChF,IAAI,OAAO,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACzB,MAAM,IAAI,KAAK,CAAC,iCAAiC,CAAC,CAAC;QACrD,CAAC;QACD,IAAI,CAAC,SAAS,GAAG,QAAQ,CAAC;QAC1B,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC;IAC1B,CAAC;IAED;;OAEG;IACH,IAAI,QAAQ;QACV,OAAO,IAAI,CAAC,SAAS,CAAC;IACxB,CAAC;IAED;;OAEG;IACH,IAAI,OAAO;QACT,OAAO,IAAI,CAAC,QAAQ,CAAC;IACvB,CAAC;IAED;;OAEG;IACH,WAAW;QACT,OAAO,CAAC,IAAO,EAAW,EAAE;YAC1B,IAAI,IAAI,CAAC,SAAS,KAAK,KAAK,EAAE,CAAC;gBAC7B,OAAO,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,WAAW,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC;YACrE,CAAC;iBAAM,CAAC;gBACN,KAAK;gBACL,OAAO,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,WAAW,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC;YACpE,CAAC;QACH,CAAC,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,GAAG,CAAC,MAAsC;QACxC,IAAI,IAAI,CAAC,SAAS,KAAK,KAAK,EAAE,CAAC;YAC7B,yBAAyB;YACzB,OAAO,IAAI,eAAe,CAAI,KAAK,EAAE,CAAC,GAAG,IAAI,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC,CAAC;QACnE,CAAC;aAAM,CAAC;YACN,kBAAkB;YAClB,OAAO,IAAI,eAAe,CAAI,KAAK,EAAE,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC,CAAC;QACvD,CAAC;IACH,CAAC;IAED;;OAEG;IACH,EAAE,CAAC,MAAsC;QACvC,IAAI,IAAI,CAAC,SAAS,KAAK,IAAI,EAAE,CAAC;YAC5B,wBAAwB;YACxB,OAAO,IAAI,eAAe,CAAI,IAAI,EAAE,CAAC,GAAG,IAAI,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC,CAAC;QAClE,CAAC;aAAM,CAAC;YACN,iBAAiB;YACjB,OAAO,IAAI,eAAe,CAAI,IAAI,EAAE,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC,CAAC;QACtD,CAAC;IACH,CAAC;CACF;AAjED,0CAiEC"}