@woltz/rich-domain 1.2.2 → 1.3.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 (71) hide show
  1. package/dist/aggregate-changes.d.ts +56 -14
  2. package/dist/aggregate-changes.d.ts.map +1 -1
  3. package/dist/aggregate-changes.js +102 -22
  4. package/dist/aggregate-changes.js.map +1 -1
  5. package/dist/base-entity.d.ts +1 -1
  6. package/dist/base-entity.d.ts.map +1 -1
  7. package/dist/base-entity.js +21 -13
  8. package/dist/base-entity.js.map +1 -1
  9. package/dist/change-tracker.d.ts +1 -0
  10. package/dist/change-tracker.d.ts.map +1 -1
  11. package/dist/change-tracker.js +43 -28
  12. package/dist/change-tracker.js.map +1 -1
  13. package/dist/criteria.d.ts +7 -15
  14. package/dist/criteria.d.ts.map +1 -1
  15. package/dist/criteria.js +99 -76
  16. package/dist/criteria.js.map +1 -1
  17. package/dist/entity-schema-registry.d.ts +133 -3
  18. package/dist/entity-schema-registry.d.ts.map +1 -1
  19. package/dist/entity-schema-registry.js +155 -4
  20. package/dist/entity-schema-registry.js.map +1 -1
  21. package/dist/paginated-result.d.ts +4 -4
  22. package/dist/paginated-result.d.ts.map +1 -1
  23. package/dist/paginated-result.js +6 -20
  24. package/dist/paginated-result.js.map +1 -1
  25. package/dist/types/change-tracker.d.ts +30 -0
  26. package/dist/types/change-tracker.d.ts.map +1 -1
  27. package/dist/types/criteria.d.ts +1 -4
  28. package/dist/types/criteria.d.ts.map +1 -1
  29. package/dist/types/domain.d.ts +2 -0
  30. package/dist/types/domain.d.ts.map +1 -1
  31. package/dist/types/utils.d.ts +2 -2
  32. package/dist/utils/helpers.d.ts +1 -0
  33. package/dist/utils/helpers.d.ts.map +1 -1
  34. package/dist/utils/helpers.js +23 -0
  35. package/dist/utils/helpers.js.map +1 -1
  36. package/dist/value-object.d.ts +1 -1
  37. package/dist/value-object.js +1 -1
  38. package/package.json +17 -3
  39. package/src/aggregate-changes.ts +133 -24
  40. package/src/base-entity.ts +22 -13
  41. package/src/change-tracker.ts +110 -39
  42. package/src/criteria.ts +151 -109
  43. package/src/entity-schema-registry.ts +253 -6
  44. package/src/paginated-result.ts +10 -30
  45. package/src/types/change-tracker.ts +31 -0
  46. package/src/types/criteria.ts +1 -4
  47. package/src/types/domain.ts +2 -0
  48. package/src/types/utils.ts +2 -2
  49. package/src/utils/helpers.ts +28 -0
  50. package/src/value-object.ts +1 -1
  51. package/.versionrc.json +0 -21
  52. package/CHANGELOG.md +0 -132
  53. package/tests/aggregate-changes.test.ts +0 -284
  54. package/tests/criteria.test.ts +0 -716
  55. package/tests/domain-events.test.ts +0 -431
  56. package/tests/entity-equality.test.ts +0 -464
  57. package/tests/entity-schema-registry.test.ts +0 -382
  58. package/tests/entity-validation.test.ts +0 -252
  59. package/tests/history-tracker.spec.ts +0 -439
  60. package/tests/id.test.ts +0 -338
  61. package/tests/load-test/data.json +0 -346041
  62. package/tests/load-test/entities.ts +0 -97
  63. package/tests/load-test/generate-data.ts +0 -81
  64. package/tests/load-test/lead-to-domain.mapper.ts +0 -24
  65. package/tests/load-test/load.test.ts +0 -38
  66. package/tests/repository.test.ts +0 -635
  67. package/tests/to-json.test.ts +0 -99
  68. package/tests/utils.ts +0 -290
  69. package/tests/value-object-validation.test.ts +0 -219
  70. package/tests/value-objects.test.ts +0 -80
  71. package/tsconfig.json +0 -9
package/CHANGELOG.md DELETED
@@ -1,132 +0,0 @@
1
- # Changelog
2
-
3
- All notable changes to this project will be documented in this file. See [standard-version](https://github.com/conventional-changelog/standard-version) for commit guidelines.
4
-
5
- ### [1.2.2](https://github.com/tarcisioandrade/rich-domain/compare/v0.3.0...v1.2.2) (2025-11-30)
6
-
7
-
8
- ### Features
9
-
10
- * filter component with criteria ([a6e5a66](https://github.com/tarcisioandrade/rich-domain/commit/a6e5a664e7b175f8156832b2d42821f73efec522))
11
-
12
-
13
- ### Bug Fixes
14
-
15
- * incorrect verification in criteria operator validation ([8b4da16](https://github.com/tarcisioandrade/rich-domain/commit/8b4da16ded67a55afe64adbe60d52359cd168a17))
16
-
17
-
18
- ### Chores
19
-
20
- * update dependencies and refactor mappers to use PrismaToPersistence ([5281716](https://github.com/tarcisioandrade/rich-domain/commit/52817161667ce473ffe2f7991d55c03190d097c0))
21
-
22
-
23
- ### Refactoring
24
-
25
- * enhance criteria validation by adding field value sanitization ([9991a6b](https://github.com/tarcisioandrade/rich-domain/commit/9991a6b54dbfa8b5d2505ed17897b76f6b794a33))
26
- * improve filter component logic and UI interactions for adding filters ([3dd8d9f](https://github.com/tarcisioandrade/rich-domain/commit/3dd8d9fb8934ca15357ad62485f74c541c01a680))
27
- * migrate filter types to filter-utils for improved organization and clarity ([eeb2af4](https://github.com/tarcisioandrade/rich-domain/commit/eeb2af4f563ff7f8dc3eb1836b22392bc229f3af))
28
- * update FilterDateValue component to handle string dates and improve date handling logic ([6cb0b42](https://github.com/tarcisioandrade/rich-domain/commit/6cb0b4245ce9394389ee208c16035bc49e4aa912))
29
-
30
- ### [1.2.1](https://github.com/tarcisioandrade/rich-domain/compare/v1.2.0...v1.2.1) (2025-11-29)
31
-
32
-
33
- ### Features
34
-
35
- * implement custom UUID generation in crypto module ([4fc732a](https://github.com/tarcisioandrade/rich-domain/commit/4fc732a00237a9d63e320555acf4a8a14c941f13))
36
- * introduce TypedOrder and enhance Criteria to support nested object field paths ([9857624](https://github.com/tarcisioandrade/rich-domain/commit/9857624608eda1320ec813b165e9183f1e28a0d1))
37
- * new history-tracker structure ([aaea6d4](https://github.com/tarcisioandrade/rich-domain/commit/aaea6d4c8aed575f50917cba41936d1b1560ed3d))
38
- * update Id constructor to handle optional isNew parameter ([0aae58b](https://github.com/tarcisioandrade/rich-domain/commit/0aae58b02da69e0cb7421663335ac305ade5afa7))
39
-
40
-
41
- ### Bug Fixes
42
-
43
- * aggregate-changes type inference ([aa62138](https://github.com/tarcisioandrade/rich-domain/commit/aa62138b31cdedb63c0e1be8a19271f4ada2e71c))
44
- * history tracker not tracking when nested created ([2900e20](https://github.com/tarcisioandrade/rich-domain/commit/2900e20c18da7aa0ee960c9e5d0f5e2524e3d6f3))
45
- * uow in persistence fastify example ([fcef611](https://github.com/tarcisioandrade/rich-domain/commit/fcef61103f160c0b1adbee9cbce821ce047189eb))
46
-
47
-
48
- ### Refactoring
49
-
50
- * change prisma example to new architecture ([92dc179](https://github.com/tarcisioandrade/rich-domain/commit/92dc1798ee5a64439d6633ab5b0cc77b42265ed6))
51
- * improve deepClone method for better object handling and error management ([52ae893](https://github.com/tarcisioandrade/rich-domain/commit/52ae8930d112ba018f7944f69837edea82d45309))
52
- * prisma example unit of work with async local storage ([ca6fc16](https://github.com/tarcisioandrade/rich-domain/commit/ca6fc1663ca251dac020939080b126f3984b0eae))
53
- * remove unnosed code from ancient architecture ([19a41b1](https://github.com/tarcisioandrade/rich-domain/commit/19a41b121b4adaaea5525da20b068f43cc674db8))
54
- * update repository methods to use 'save' consistently across use cases ([aab323b](https://github.com/tarcisioandrade/rich-domain/commit/aab323be2fff86246dc0c1a14cb7ba56bf11efa7))
55
- * update repository methods to use 'save' instead of 'create' and 'update' ([6766970](https://github.com/tarcisioandrade/rich-domain/commit/6766970019781215019b492963e020ef4babcd89))
56
-
57
-
58
- ### Tests
59
-
60
- * add load test ([9bd07fb](https://github.com/tarcisioandrade/rich-domain/commit/9bd07fb019e2cc0eef3f0573af8f4dad1d8b385f))
61
- * new history-tracker structure ([d31ad0b](https://github.com/tarcisioandrade/rich-domain/commit/d31ad0b109bfa5e5ff62743d70dd51bb64107f10))
62
-
63
- ## [1.2.0](https://github.com/tarcisioandrade/rich-domain/compare/v1.1.0...v1.2.0) (2025-11-23)
64
-
65
-
66
- ### Features
67
-
68
- * add getFilterByField and getSortByField methods to useCriteria hook ([8db041a](https://github.com/tarcisioandrade/rich-domain/commit/8db041a8fbe359ec6fe3ff70e22ac87a6c221686))
69
- * add options parameter to addFilter method for enhanced filtering capabilities ([c43c164](https://github.com/tarcisioandrade/rich-domain/commit/c43c164f56831b9f0b06f2e5981ba3f540cf3378))
70
- * add quantifier support to Criteria class for enhanced filtering options ([322adf3](https://github.com/tarcisioandrade/rich-domain/commit/322adf36dfedcdc21ae6fc85c3577e1fb2835019))
71
- * criteria adapter implementation ([f387f0b](https://github.com/tarcisioandrade/rich-domain/commit/f387f0bcd601a69da74d5fff33b9d9872e049179))
72
- * enhance criteria operators with new types and validation functions ([8d30c14](https://github.com/tarcisioandrade/rich-domain/commit/8d30c1448e3d25c790d667b43d17546a441e710d))
73
-
74
-
75
- ### Bug Fixes
76
-
77
- * FieldPath infer type to array methods ([844a56e](https://github.com/tarcisioandrade/rich-domain/commit/844a56ef2cdc65fac0cb228c6936808a92ffa440))
78
- * search filter not reset page ([c8c9d8b](https://github.com/tarcisioandrade/rich-domain/commit/c8c9d8b1fecc1ce8c568f43710c8748ad9ad1f1e))
79
- * types in react-with-react-query ([6eeeb74](https://github.com/tarcisioandrade/rich-domain/commit/6eeeb74715bef50d3c5dec8dcc3e37484e0f511b))
80
-
81
-
82
- ### Chores
83
-
84
- * add TypeScript check script and enhance ([9f6347c](https://github.com/tarcisioandrade/rich-domain/commit/9f6347c4707adebb9e03bcdc952791a9c3279038))
85
- * enhance useCriteria hook and update TypeScript types for better type safety ([2bbc699](https://github.com/tarcisioandrade/rich-domain/commit/2bbc699aab415a52502ad50e7fbf67bcd52ae222))
86
- * refactor frontend example to integrate useCriteria hook ([48dccc7](https://github.com/tarcisioandrade/rich-domain/commit/48dccc7df870024f280902c22c77dcf531702781))
87
- * update ESLint configurations to include tsconfigRootDir ([b019884](https://github.com/tarcisioandrade/rich-domain/commit/b01988496b91c6c9d3e6d3d89597cf9f23e54729))
88
- * update rich domain version in examples app ([4be8a1b](https://github.com/tarcisioandrade/rich-domain/commit/4be8a1bdd4473d8274c0ec8bec1d4ec1d70109ea))
89
- * update target paths in registry.json and use-criteria.json for improved file structure ([d2f9348](https://github.com/tarcisioandrade/rich-domain/commit/d2f93486158e465d8878d92f1d80e259cc62f701))
90
-
91
-
92
- ### Refactoring
93
-
94
- * criteria adapter in fastify-with-prisma example ([8adef1e](https://github.com/tarcisioandrade/rich-domain/commit/8adef1e0c62ab94bd3665f91ca6ea29d568ea24f))
95
-
96
- ## [1.1.0](https://github.com/tarcisioandrade/rich-domain/compare/v1.0.0...v1.1.0) (2025-11-22)
97
-
98
-
99
- ### Features
100
-
101
- * add examples in monorepo ([2cba30e](https://github.com/tarcisioandrade/rich-domain/commit/2cba30e318be2575b007c8b4f9a00e35561eec5e))
102
- * add react-with-react-query example ([439589e](https://github.com/tarcisioandrade/rich-domain/commit/439589e1abbc47b1b03efab3b9dde53fe8d6253c))
103
- * domain execeptions ([e206274](https://github.com/tarcisioandrade/rich-domain/commit/e206274d8a93c09cc91b2c0514be62cb986caaa3))
104
- * implement useCriteria hook ([c00441a](https://github.com/tarcisioandrade/rich-domain/commit/c00441a105391fd06ef39b8c7e3aa0907c3ff508))
105
-
106
-
107
- ### Bug Fixes
108
-
109
- * infinite loop when multiples order are add ([2c13dc4](https://github.com/tarcisioandrade/rich-domain/commit/2c13dc472d009bf328ad45222ea1c921f588ff9a))
110
- * search not working in fromArray PaginatedResult method ([b79b2aa](https://github.com/tarcisioandrade/rich-domain/commit/b79b2aa2aa17e8b98bf2c2fcb4823452db99dcc1))
111
-
112
-
113
- ### Refactoring
114
-
115
- * clean exports ([3f8e16a](https://github.com/tarcisioandrade/rich-domain/commit/3f8e16a2103514ef7a2a928dff20f674d1e39506))
116
- * migrate eslint configuration to ES module syntax ([3ea4366](https://github.com/tarcisioandrade/rich-domain/commit/3ea43667b0857b7ae2afcbad858c16384035dc50))
117
- * monorepo implementation ([e0b1e68](https://github.com/tarcisioandrade/rich-domain/commit/e0b1e683d5ee95b54b79510d43dd957af49c65e2))
118
- * remove defaultValues from value objects ([0523479](https://github.com/tarcisioandrade/rich-domain/commit/0523479c6f272d7ea5d678638fa666e3fde71c7a))
119
- * streamline domain event imports and consolidate IDomainEvent interface ([573c689](https://github.com/tarcisioandrade/rich-domain/commit/573c689a762baecd9fb8791e66300f663d7a58c1))
120
-
121
-
122
- ### Chores
123
-
124
- * add postinstall script to generate Prisma client ([ffbe310](https://github.com/tarcisioandrade/rich-domain/commit/ffbe310db4ff9e6918d4e55096b8939c1f8f4543))
125
- * enhance linting setup for react-rich-domain and update package.json ([ac2803b](https://github.com/tarcisioandrade/rich-domain/commit/ac2803bac0d907c0513d249fc5e14390418c7530))
126
- * modify check script to target specific workspace ([d39d6d8](https://github.com/tarcisioandrade/rich-domain/commit/d39d6d8ef3a04d17207a3a0e795027040b3cf707))
127
- * update clean script to remove node_modules in backend and rich-domain packages ([2a10cab](https://github.com/tarcisioandrade/rich-domain/commit/2a10cab193ba00e504a9a301cc00cda4b68b8573))
128
- * update module entry points in package.json for react-rich-domain ([03be9ac](https://github.com/tarcisioandrade/rich-domain/commit/03be9aced05160c97eb74f771b770283c07e046b))
129
- * update package configuration and add new dependencies ([4135393](https://github.com/tarcisioandrade/rich-domain/commit/4135393ae78b1b8e6b0c9306f0ac0872971ddb3f))
130
- * update package.json scripts to target specific workspace ([8f46868](https://github.com/tarcisioandrade/rich-domain/commit/8f468688017f49cae58dff5d1a9818fe64b38017))
131
- * update package.json with homepage and repository detail ([bfb387c](https://github.com/tarcisioandrade/rich-domain/commit/bfb387cb2076989a808f25be09337a24ed3a8ff6))
132
- * update test script to target specific workspace ([6783208](https://github.com/tarcisioandrade/rich-domain/commit/678320803828a43ef588c366e59bf62c2851c126))
@@ -1,284 +0,0 @@
1
- // ============================================================================
2
- // Tests: AggregateChanges
3
- // ============================================================================
4
-
5
- import { AggregateChanges } from "../src/aggregate-changes";
6
-
7
- describe("AggregateChanges", () => {
8
- let changes: AggregateChanges;
9
-
10
- beforeEach(() => {
11
- changes = new AggregateChanges();
12
- });
13
-
14
- describe("isEmpty / hasChanges", () => {
15
- it("should be empty initially", () => {
16
- expect(changes.isEmpty()).toBe(true);
17
- expect(changes.hasChanges()).toBe(false);
18
- expect(changes.count).toBe(0);
19
- });
20
-
21
- it("should not be empty after adding operation", () => {
22
- changes.addCreate("User", { id: "1", name: "Test" }, 0);
23
- expect(changes.isEmpty()).toBe(false);
24
- expect(changes.hasChanges()).toBe(true);
25
- expect(changes.count).toBe(1);
26
- });
27
- });
28
-
29
- describe("addCreate", () => {
30
- it("should add create operation", () => {
31
- const data = { id: "1", name: "Test User" };
32
- changes.addCreate("User", data, 0);
33
-
34
- expect(changes.hasCreates()).toBe(true);
35
- expect(changes.creates()).toHaveLength(1);
36
- expect(changes.creates()[0]).toMatchObject({
37
- type: "create",
38
- entity: "User",
39
- data,
40
- depth: 0,
41
- });
42
- });
43
-
44
- it("should add create with parent info", () => {
45
- const data = { id: "post-1", title: "Test Post" };
46
- changes.addCreate("Post", data, 1, "user-1", "User");
47
-
48
- const create = changes.creates()[0];
49
- expect(create.parentId).toBe("user-1");
50
- expect(create.parentEntity).toBe("User");
51
- });
52
- });
53
-
54
- describe("addUpdate", () => {
55
- it("should add update operation", () => {
56
- const data = { id: "1", name: "Updated" };
57
- const changedFields = { name: "Updated" };
58
- changes.addUpdate("User", "1", data, changedFields, 0);
59
-
60
- expect(changes.hasUpdates()).toBe(true);
61
- expect(changes.updates()).toHaveLength(1);
62
- expect(changes.updates()[0]).toMatchObject({
63
- type: "update",
64
- entity: "User",
65
- id: "1",
66
- changedFields,
67
- depth: 0,
68
- });
69
- });
70
- });
71
-
72
- describe("addDelete", () => {
73
- it("should add delete operation", () => {
74
- const data = { id: "1", name: "To Delete" };
75
- changes.addDelete("User", "1", data, 0);
76
-
77
- expect(changes.hasDeletes()).toBe(true);
78
- expect(changes.deletes()).toHaveLength(1);
79
- expect(changes.deletes()[0]).toMatchObject({
80
- type: "delete",
81
- entity: "User",
82
- id: "1",
83
- depth: 0,
84
- });
85
- });
86
- });
87
-
88
- describe("ordering", () => {
89
- beforeEach(() => {
90
- // Add operations in random order
91
- changes.addCreate("Comment", { id: "c1" }, 2);
92
- changes.addDelete("Like", "l1", { id: "l1" }, 3);
93
- changes.addCreate("User", { id: "u1" }, 0);
94
- changes.addDelete("Comment", "c2", { id: "c2" }, 2);
95
- changes.addCreate("Post", { id: "p1" }, 1);
96
- changes.addDelete("Post", "p2", { id: "p2" }, 1);
97
- });
98
-
99
- it("should order creates by depth ASC (root → leaf)", () => {
100
- const creates = changes.creates();
101
- expect(creates[0].entity).toBe("User"); // depth 0
102
- expect(creates[1].entity).toBe("Post"); // depth 1
103
- expect(creates[2].entity).toBe("Comment"); // depth 2
104
- });
105
-
106
- it("should order deletes by depth DESC (leaf → root)", () => {
107
- const deletes = changes.deletes();
108
- expect(deletes[0].entity).toBe("Like"); // depth 3
109
- expect(deletes[1].entity).toBe("Comment"); // depth 2
110
- expect(deletes[2].entity).toBe("Post"); // depth 1
111
- });
112
- });
113
-
114
- describe("operations iterator", () => {
115
- it("should yield operations in correct order: deletes, creates, updates", () => {
116
- changes.addCreate("Post", { id: "p1" }, 1);
117
- changes.addUpdate("User", "u1", { id: "u1" }, { name: "New" }, 0);
118
- changes.addDelete("Comment", "c1", { id: "c1" }, 2);
119
-
120
- const ops = [...changes.operations()];
121
-
122
- expect(ops[0].type).toBe("delete"); // deletes first
123
- expect(ops[1].type).toBe("create"); // creates second
124
- expect(ops[2].type).toBe("update"); // updates last
125
- });
126
- });
127
-
128
- describe("toBatchOperations", () => {
129
- beforeEach(() => {
130
- // Creates
131
- changes.addCreate("User", { id: "u1" }, 0);
132
- changes.addCreate("Post", { id: "p1" }, 1, "u1", "User");
133
- changes.addCreate("Post", { id: "p2" }, 1, "u1", "User");
134
- changes.addCreate("Comment", { id: "c1" }, 2, "p1", "Post");
135
-
136
- // Updates
137
- changes.addUpdate("User", "u2", { id: "u2" }, { name: "Updated" }, 0);
138
- changes.addUpdate("Post", "p3", { id: "p3" }, { title: "New Title" }, 1);
139
-
140
- // Deletes
141
- changes.addDelete("Comment", "c2", { id: "c2" }, 2);
142
- changes.addDelete("Comment", "c3", { id: "c3" }, 2);
143
- changes.addDelete("Post", "p4", { id: "p4" }, 1);
144
- });
145
-
146
- it("should group deletes by entity and order by depth DESC", () => {
147
- const batch = changes.toBatchOperations();
148
-
149
- expect(batch.deletes).toHaveLength(2); // Comment and Post
150
-
151
- // Comments should come first (depth 2)
152
- expect(batch.deletes[0].entity).toBe("Comment");
153
- expect(batch.deletes[0].ids).toEqual(["c2", "c3"]);
154
-
155
- // Posts second (depth 1)
156
- expect(batch.deletes[1].entity).toBe("Post");
157
- expect(batch.deletes[1].ids).toEqual(["p4"]);
158
- });
159
-
160
- it("should group creates by entity and order by depth ASC", () => {
161
- const batch = changes.toBatchOperations();
162
-
163
- expect(batch.creates).toHaveLength(3); // User, Post, Comment
164
-
165
- // User first (depth 0)
166
- expect(batch.creates[0].entity).toBe("User");
167
- expect(batch.creates[0].items).toHaveLength(1);
168
-
169
- // Post second (depth 1)
170
- expect(batch.creates[1].entity).toBe("Post");
171
- expect(batch.creates[1].items).toHaveLength(2);
172
-
173
- // Comment last (depth 2)
174
- expect(batch.creates[2].entity).toBe("Comment");
175
- expect(batch.creates[2].items).toHaveLength(1);
176
- });
177
-
178
- it("should group updates by entity", () => {
179
- const batch = changes.toBatchOperations();
180
-
181
- expect(batch.updates).toHaveLength(2); // User and Post
182
-
183
- const userUpdates = batch.updates.find((u) => u.entity === "User");
184
- expect(userUpdates?.items).toHaveLength(1);
185
- expect(userUpdates?.items[0].id).toBe("u2");
186
-
187
- const postUpdates = batch.updates.find((u) => u.entity === "Post");
188
- expect(postUpdates?.items).toHaveLength(1);
189
- expect(postUpdates?.items[0].id).toBe("p3");
190
- });
191
-
192
- it("should include parentId in create items", () => {
193
- const batch = changes.toBatchOperations();
194
-
195
- const postCreates = batch.creates.find((c) => c.entity === "Post");
196
- expect(postCreates?.items[0].parentId).toBe("u1");
197
- expect(postCreates?.items[1].parentId).toBe("u1");
198
-
199
- const commentCreates = batch.creates.find((c) => c.entity === "Comment");
200
- expect(commentCreates?.items[0].parentId).toBe("p1");
201
- });
202
- });
203
-
204
- describe("for (filter by entity)", () => {
205
- beforeEach(() => {
206
- changes.addCreate("Post", { id: "p1", title: "Post 1" }, 1);
207
- changes.addCreate("Post", { id: "p2", title: "Post 2" }, 1);
208
- changes.addUpdate("Post", "p3", { id: "p3" }, { title: "Updated" }, 1);
209
- changes.addDelete("Post", "p4", { id: "p4" }, 1);
210
- changes.addCreate("Comment", { id: "c1" }, 2);
211
- });
212
-
213
- it("should filter creates by entity", () => {
214
- const postChanges = changes.for("Post");
215
- expect(postChanges.creates).toHaveLength(2);
216
- expect(postChanges.creates[0].id).toBe("p1");
217
- });
218
-
219
- it("should filter updates by entity", () => {
220
- const postChanges = changes.for("Post");
221
- expect(postChanges.updates).toHaveLength(1);
222
- expect(postChanges.updates[0].entity.id).toBe("p3");
223
- });
224
-
225
- it("should filter deletes by entity", () => {
226
- const postChanges = changes.for("Post");
227
- expect(postChanges.deletes).toHaveLength(1);
228
- expect(postChanges.deletes[0].id).toBe("p4");
229
- });
230
-
231
- it("should return empty for non-existent entity", () => {
232
- const userChanges = changes.for("User");
233
- expect(userChanges.isEmpty()).toBe(true);
234
- });
235
-
236
- it("should have helper methods", () => {
237
- const postChanges = changes.for("Post");
238
- expect(postChanges.hasCreates()).toBe(true);
239
- expect(postChanges.hasUpdates()).toBe(true);
240
- expect(postChanges.hasDeletes()).toBe(true);
241
- expect(postChanges.hasChanges()).toBe(true);
242
- });
243
- });
244
-
245
- describe("getAffectedEntities", () => {
246
- it("should return list of unique entities", () => {
247
- changes.addCreate("User", { id: "u1" }, 0);
248
- changes.addCreate("Post", { id: "p1" }, 1);
249
- changes.addUpdate("Post", "p2", { id: "p2" }, {}, 1);
250
- changes.addDelete("Comment", "c1", { id: "c1" }, 2);
251
-
252
- const entities = changes.getAffectedEntities();
253
-
254
- expect(entities).toContain("User");
255
- expect(entities).toContain("Post");
256
- expect(entities).toContain("Comment");
257
- expect(entities).toHaveLength(3);
258
- });
259
- });
260
-
261
- describe("clone", () => {
262
- it("should create independent copy", () => {
263
- changes.addCreate("User", { id: "u1" }, 0);
264
-
265
- const cloned = changes.clone();
266
- cloned.addCreate("Post", { id: "p1" }, 1);
267
-
268
- expect(changes.count).toBe(1);
269
- expect(cloned.count).toBe(2);
270
- });
271
- });
272
-
273
- describe("clear", () => {
274
- it("should remove all operations", () => {
275
- changes.addCreate("User", { id: "u1" }, 0);
276
- changes.addUpdate("Post", "p1", { id: "p1" }, {}, 1);
277
-
278
- changes.clear();
279
-
280
- expect(changes.isEmpty()).toBe(true);
281
- expect(changes.count).toBe(0);
282
- });
283
- });
284
- });