@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.
- package/CHANGELOG.md +30 -0
- package/README.md +37 -20
- package/dist/base-entity.d.ts +1 -1
- package/dist/base-entity.d.ts.map +1 -1
- package/dist/base-entity.js +21 -15
- package/dist/base-entity.js.map +1 -1
- package/dist/constants.js +4 -1
- package/dist/constants.js.map +1 -1
- package/dist/criteria.d.ts.map +1 -1
- package/dist/criteria.js +7 -3
- package/dist/criteria.js.map +1 -1
- package/dist/deep-proxy.d.ts +3 -1
- package/dist/deep-proxy.d.ts.map +1 -1
- package/dist/deep-proxy.js +110 -33
- package/dist/deep-proxy.js.map +1 -1
- package/dist/domain-event-bus.js +7 -2
- package/dist/domain-event-bus.js.map +1 -1
- package/dist/domain-event.js +7 -3
- package/dist/domain-event.js.map +1 -1
- package/dist/entity.js +8 -3
- package/dist/entity.js.map +1 -1
- package/dist/id.d.ts +3 -3
- package/dist/id.d.ts.map +1 -1
- package/dist/id.js +10 -6
- package/dist/id.js.map +1 -1
- package/dist/index.d.ts +0 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +39 -17
- package/dist/index.js.map +1 -1
- package/dist/mapper.js +5 -1
- package/dist/mapper.js.map +1 -1
- package/dist/paginated-result.js +7 -3
- package/dist/paginated-result.js.map +1 -1
- package/dist/repository/base-repository.js +11 -4
- package/dist/repository/base-repository.js.map +1 -1
- package/dist/repository/in-memory-repository.js +8 -4
- package/dist/repository/in-memory-repository.js.map +1 -1
- package/dist/repository/index.d.ts +1 -37
- package/dist/repository/index.d.ts.map +1 -1
- package/dist/repository/index.js +26 -40
- package/dist/repository/index.js.map +1 -1
- package/dist/repository/unit-of-work.js +9 -3
- package/dist/repository/unit-of-work.js.map +1 -1
- package/dist/types/criteria.d.ts +2 -2
- package/dist/types/criteria.d.ts.map +1 -1
- package/dist/types/criteria.js +4 -1
- package/dist/types/criteria.js.map +1 -1
- package/dist/types/domain.js +2 -1
- package/dist/types/history-tracker.js +2 -1
- package/dist/types/index.js +22 -6
- package/dist/types/index.js.map +1 -1
- package/dist/types/standard-schema.js +2 -1
- package/dist/types/unit-of-work.js +2 -1
- package/dist/types/utils.js +2 -1
- package/dist/validation-error.js +9 -3
- package/dist/validation-error.js.map +1 -1
- package/dist/value-object.js +9 -5
- package/dist/value-object.js.map +1 -1
- package/package.json +1 -1
- package/src/base-entity.ts +3 -2
- package/src/criteria.ts +2 -2
- package/src/deep-proxy.ts +435 -339
- package/src/id.ts +4 -4
- package/src/index.ts +1 -3
- package/src/repository/index.ts +1 -38
- package/src/types/criteria.ts +2 -2
- package/tests/entity-validation.test.ts +1 -1
- package/tests/history-tracker.spec.ts +57 -17
- package/tests/id.test.ts +341 -341
- package/tests/repository.test.ts +4 -2
- package/tests/to-json.test.ts +103 -91
- package/tests/value-objects.test.ts +52 -52
- package/tsconfig.json +2 -2
- package/dist/filtering.d.ts +0 -107
- package/dist/filtering.d.ts.map +0 -1
- package/dist/filtering.js +0 -202
- package/dist/filtering.js.map +0 -1
- package/dist/ordering.d.ts +0 -93
- package/dist/ordering.d.ts.map +0 -1
- package/dist/ordering.js +0 -154
- package/dist/ordering.js.map +0 -1
- package/dist/pagination.d.ts +0 -218
- package/dist/pagination.d.ts.map +0 -1
- package/dist/pagination.js +0 -281
- package/dist/pagination.js.map +0 -1
- package/dist/repository/mapper.d.ts +0 -56
- package/dist/repository/mapper.d.ts.map +0 -1
- package/dist/repository/mapper.js +0 -15
- package/dist/repository/mapper.js.map +0 -1
- package/dist/repository/types.d.ts +0 -87
- package/dist/repository/types.d.ts.map +0 -1
- package/dist/repository/types.js +0 -6
- package/dist/repository/types.js.map +0 -1
- package/dist/repository.d.ts +0 -2
- package/dist/repository.d.ts.map +0 -1
- package/dist/repository.js +0 -21
- package/dist/repository.js.map +0 -1
- package/dist/specification.d.ts +0 -102
- package/dist/specification.d.ts.map +0 -1
- package/dist/specification.js +0 -187
- package/dist/specification.js.map +0 -1
- package/dist/types/repository.d.ts +0 -43
- package/dist/types/repository.d.ts.map +0 -1
- package/dist/types/repository.js +0 -2
- package/dist/types/repository.js.map +0 -1
- package/dist/types.d.ts +0 -88
- package/dist/types.d.ts.map +0 -1
- package/dist/types.js +0 -12
- package/dist/types.js.map +0 -1
package/tests/repository.test.ts
CHANGED
|
@@ -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) =>
|
|
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");
|
package/tests/to-json.test.ts
CHANGED
|
@@ -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": "
|
|
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": "
|
|
23
|
+
"moduleResolution": "node",
|
|
24
24
|
"esModuleInterop": true,
|
|
25
25
|
"skipLibCheck": true,
|
|
26
26
|
"forceConsistentCasingInFileNames": true,
|
package/dist/filtering.d.ts
DELETED
|
@@ -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
|
package/dist/filtering.d.ts.map
DELETED
|
@@ -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
|
package/dist/filtering.js.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"filtering.js","sourceRoot":"","sources":["../src/filtering.ts"],"names":[],"mappings":"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"}
|