metal-orm 1.1.8 → 1.1.10

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 (75) hide show
  1. package/README.md +769 -764
  2. package/dist/index.cjs +2352 -226
  3. package/dist/index.cjs.map +1 -1
  4. package/dist/index.d.cts +605 -40
  5. package/dist/index.d.ts +605 -40
  6. package/dist/index.js +2324 -226
  7. package/dist/index.js.map +1 -1
  8. package/package.json +22 -17
  9. package/src/bulk/bulk-context.ts +83 -0
  10. package/src/bulk/bulk-delete-executor.ts +89 -0
  11. package/src/bulk/bulk-executor.base.ts +73 -0
  12. package/src/bulk/bulk-insert-executor.ts +74 -0
  13. package/src/bulk/bulk-types.ts +70 -0
  14. package/src/bulk/bulk-update-executor.ts +192 -0
  15. package/src/bulk/bulk-upsert-executor.ts +95 -0
  16. package/src/bulk/bulk-utils.ts +91 -0
  17. package/src/bulk/index.ts +18 -0
  18. package/src/codegen/typescript.ts +30 -21
  19. package/src/core/ast/expression-builders.ts +107 -10
  20. package/src/core/ast/expression-nodes.ts +52 -22
  21. package/src/core/ast/expression-visitor.ts +23 -13
  22. package/src/core/dialect/abstract.ts +30 -17
  23. package/src/core/dialect/mysql/index.ts +20 -5
  24. package/src/core/execution/db-executor.ts +96 -64
  25. package/src/core/execution/executors/better-sqlite3-executor.ts +94 -0
  26. package/src/core/execution/executors/mssql-executor.ts +66 -34
  27. package/src/core/execution/executors/mysql-executor.ts +98 -66
  28. package/src/core/execution/executors/postgres-executor.ts +33 -11
  29. package/src/core/execution/executors/sqlite-executor.ts +86 -30
  30. package/src/decorators/bootstrap.ts +482 -398
  31. package/src/decorators/column-decorator.ts +87 -96
  32. package/src/decorators/decorator-metadata.ts +100 -24
  33. package/src/decorators/entity.ts +27 -24
  34. package/src/decorators/relations.ts +231 -149
  35. package/src/decorators/transformers/transformer-decorators.ts +26 -29
  36. package/src/decorators/validators/country-validators-decorators.ts +9 -15
  37. package/src/dto/apply-filter.ts +568 -551
  38. package/src/index.ts +16 -9
  39. package/src/orm/entity-hydration.ts +116 -72
  40. package/src/orm/entity-metadata.ts +347 -301
  41. package/src/orm/entity-relations.ts +264 -207
  42. package/src/orm/entity.ts +199 -199
  43. package/src/orm/execute.ts +13 -13
  44. package/src/orm/lazy-batch/morph-many.ts +70 -0
  45. package/src/orm/lazy-batch/morph-one.ts +69 -0
  46. package/src/orm/lazy-batch/morph-to.ts +59 -0
  47. package/src/orm/lazy-batch.ts +4 -1
  48. package/src/orm/orm-session.ts +170 -104
  49. package/src/orm/pooled-executor-factory.ts +99 -58
  50. package/src/orm/query-logger.ts +49 -40
  51. package/src/orm/relation-change-processor.ts +198 -96
  52. package/src/orm/relations/belongs-to.ts +143 -143
  53. package/src/orm/relations/has-many.ts +204 -204
  54. package/src/orm/relations/has-one.ts +174 -174
  55. package/src/orm/relations/many-to-many.ts +288 -288
  56. package/src/orm/relations/morph-many.ts +156 -0
  57. package/src/orm/relations/morph-one.ts +151 -0
  58. package/src/orm/relations/morph-to.ts +162 -0
  59. package/src/orm/save-graph.ts +116 -1
  60. package/src/query-builder/expression-table-mapper.ts +5 -0
  61. package/src/query-builder/hydration-manager.ts +345 -345
  62. package/src/query-builder/hydration-planner.ts +178 -148
  63. package/src/query-builder/relation-conditions.ts +171 -151
  64. package/src/query-builder/relation-cte-builder.ts +5 -1
  65. package/src/query-builder/relation-filter-utils.ts +9 -6
  66. package/src/query-builder/relation-include-strategies.ts +44 -2
  67. package/src/query-builder/relation-join-strategies.ts +8 -1
  68. package/src/query-builder/relation-service.ts +250 -241
  69. package/src/query-builder/select/cursor-pagination.ts +323 -0
  70. package/src/query-builder/select/select-operations.ts +110 -105
  71. package/src/query-builder/select.ts +42 -1
  72. package/src/query-builder/update-include.ts +4 -0
  73. package/src/schema/relation.ts +296 -188
  74. package/src/schema/types.ts +138 -123
  75. package/src/tree/tree-decorator.ts +127 -137
@@ -1,288 +1,288 @@
1
- import { ManyToManyCollection } from '../../schema/types.js';
2
- import { EntityContext } from '../entity-context.js';
3
- import { RelationKey } from '../runtime-types.js';
4
- import { BelongsToManyRelation } from '../../schema/relation.js';
5
- import { TableDef } from '../../schema/table.js';
6
- import { findPrimaryKey } from '../../query-builder/hydration-planner.js';
7
- import { EntityMeta, getHydrationRows } from '../entity-meta.js';
8
-
9
- type Rows = Record<string, unknown>[];
10
-
11
- const toKey = (value: unknown): string => (value === null || value === undefined ? '' : String(value));
12
-
13
- const hideInternal = (obj: object, keys: string[]): void => {
14
- for (const key of keys) {
15
- Object.defineProperty(obj, key, {
16
- value: obj[key],
17
- writable: false,
18
- configurable: false,
19
- enumerable: false
20
- });
21
- }
22
- };
23
-
24
- const hideWritable = (obj: object, keys: string[]): void => {
25
- for (const key of keys) {
26
- const value = obj[key as keyof typeof obj];
27
- Object.defineProperty(obj, key, {
28
- value,
29
- writable: true,
30
- configurable: true,
31
- enumerable: false
32
- });
33
- }
34
- };
35
-
36
- const normalizePivot = (pivot: unknown): Record<string, unknown> | undefined => {
37
- if (!pivot || typeof pivot !== 'object' || Array.isArray(pivot)) return undefined;
38
- const entries = Object.entries(pivot as Record<string, unknown>)
39
- .filter(([, value]) => value !== undefined);
40
- if (!entries.length) return undefined;
41
- return Object.fromEntries(entries);
42
- };
43
-
44
- const applyPivot = (entity: Record<string, unknown>, pivot?: Record<string, unknown>): void => {
45
- if (!pivot) return;
46
- const current = entity._pivot;
47
- if (current && typeof current === 'object' && !Array.isArray(current)) {
48
- entity._pivot = { ...(current as Record<string, unknown>), ...pivot };
49
- return;
50
- }
51
- entity._pivot = { ...pivot };
52
- };
53
-
54
- /**
55
- * Default implementation of a many-to-many collection.
56
- * Manages the relationship between two entities through a pivot table.
57
- * Supports lazy loading, attaching/detaching entities, and syncing by IDs.
58
- *
59
- * @template TTarget The type of the target entities in the collection.
60
- */
61
- export class DefaultManyToManyCollection<TTarget, TPivot extends object | undefined = undefined>
62
- implements ManyToManyCollection<TTarget, TPivot> {
63
- private loaded = false;
64
- private items: TTarget[] = [];
65
-
66
- /**
67
- * @param ctx The entity context for tracking changes.
68
- * @param meta Metadata for the root entity.
69
- * @param root The root entity instance.
70
- * @param relationName The name of the relation.
71
- * @param relation Relation definition.
72
- * @param rootTable Table definition of the root entity.
73
- * @param loader Function to load the collection items.
74
- * @param createEntity Function to create entity instances from rows.
75
- * @param localKey The local key used for joining.
76
- */
77
- constructor(
78
- private readonly ctx: EntityContext,
79
- private readonly meta: EntityMeta<TableDef>,
80
- private readonly root: unknown,
81
- private readonly relationName: string,
82
- private readonly relation: BelongsToManyRelation,
83
- private readonly rootTable: TableDef,
84
- private readonly loader: () => Promise<Map<string, Rows>>,
85
- private readonly createEntity: (row: Record<string, unknown>) => TTarget,
86
- private readonly localKey: string
87
- ) {
88
- hideInternal(this, ['ctx', 'meta', 'root', 'relationName', 'relation', 'rootTable', 'loader', 'createEntity', 'localKey']);
89
- hideWritable(this, ['loaded', 'items']);
90
- this.hydrateFromCache();
91
- }
92
-
93
- /**
94
- * Loads the collection items if not already loaded.
95
- * @returns A promise that resolves to the array of target entities.
96
- */
97
- async load(): Promise<TTarget[]> {
98
- if (this.loaded) return this.items;
99
- const map = await this.loader();
100
- const key = toKey(this.root[this.localKey]);
101
- const rows = map.get(key) ?? [];
102
- this.items = rows.map(row => {
103
- const entity = this.createEntity(row);
104
- if ((row as { _pivot?: unknown })._pivot) {
105
- (entity as { _pivot?: unknown })._pivot = (row as { _pivot?: unknown })._pivot;
106
- }
107
- return entity;
108
- });
109
- this.loaded = true;
110
- return this.items;
111
- }
112
-
113
- /**
114
- * Returns the currently loaded items.
115
- * @returns Array of target entities.
116
- */
117
- getItems(): TTarget[] {
118
- return this.items;
119
- }
120
-
121
- /**
122
- * Array-compatible length for testing frameworks.
123
- */
124
- get length(): number {
125
- return this.items.length;
126
- }
127
-
128
- /**
129
- * Enables iteration over the collection like an array.
130
- */
131
- [Symbol.iterator](): Iterator<TTarget> {
132
- return this.items[Symbol.iterator]();
133
- }
134
-
135
- /**
136
- * Attaches an entity to the collection.
137
- * Registers an 'attach' change in the entity context.
138
- * @param target Entity instance or its primary key value.
139
- */
140
- attach(target: TTarget | number | string, pivot?: Partial<TPivot> | Record<string, unknown>): void {
141
- const entity = this.ensureEntity(target);
142
- const id = this.extractId(entity);
143
- const pivotPayload = this.filterPivotPayload(normalizePivot(pivot));
144
- const existing = id != null
145
- ? this.items.find(item => this.extractId(item) === id)
146
- : this.items.find(item => item === entity);
147
-
148
- if (existing) {
149
- if (pivotPayload) {
150
- applyPivot(existing as Record<string, unknown>, pivotPayload);
151
- this.ctx.registerRelationChange(
152
- this.root,
153
- this.relationKey,
154
- this.rootTable,
155
- this.relationName,
156
- this.relation,
157
- { kind: 'update', entity: existing, pivot: pivotPayload }
158
- );
159
- }
160
- return;
161
- }
162
-
163
- if (pivotPayload) {
164
- applyPivot(entity as Record<string, unknown>, pivotPayload);
165
- }
166
- this.items.push(entity);
167
- this.ctx.registerRelationChange(
168
- this.root,
169
- this.relationKey,
170
- this.rootTable,
171
- this.relationName,
172
- this.relation,
173
- { kind: 'attach', entity, pivot: pivotPayload }
174
- );
175
- }
176
-
177
- /**
178
- * Detaches an entity from the collection.
179
- * Registers a 'detach' change in the entity context.
180
- * @param target Entity instance or its primary key value.
181
- */
182
- detach(target: TTarget | number | string): void {
183
- const id = typeof target === 'number' || typeof target === 'string'
184
- ? target
185
- : this.extractId(target);
186
-
187
- if (id == null) return;
188
-
189
- const existing = this.items.find(item => this.extractId(item) === id);
190
- if (!existing) return;
191
-
192
- this.items = this.items.filter(item => this.extractId(item) !== id);
193
- this.ctx.registerRelationChange(
194
- this.root,
195
- this.relationKey,
196
- this.rootTable,
197
- this.relationName,
198
- this.relation,
199
- { kind: 'detach', entity: existing }
200
- );
201
- }
202
-
203
- /**
204
- * Syncs the collection with a list of IDs.
205
- * Attaches missing IDs and detaches IDs not in the list.
206
- * @param ids Array of primary key values to sync with.
207
- */
208
- async syncByIds(ids: (number | string)[]): Promise<void> {
209
- await this.load();
210
- const normalized = new Set(ids.map(id => toKey(id)));
211
- const currentIds = new Set(this.items.map(item => toKey(this.extractId(item))));
212
-
213
- for (const id of normalized) {
214
- if (!currentIds.has(id)) {
215
- this.attach(id);
216
- }
217
- }
218
-
219
- for (const item of [...this.items]) {
220
- const itemId = toKey(this.extractId(item));
221
- if (!normalized.has(itemId)) {
222
- this.detach(item);
223
- }
224
- }
225
- }
226
-
227
- private ensureEntity(target: TTarget | number | string): TTarget {
228
- if (typeof target === 'number' || typeof target === 'string') {
229
- const stub: Record<string, unknown> = {
230
- [this.targetKey]: target
231
- };
232
- return this.createEntity(stub);
233
- }
234
- return target;
235
- }
236
-
237
- private extractId(entity: TTarget | number | string | null | undefined): number | string | null {
238
- if (entity === null || entity === undefined) return null;
239
- if (typeof entity === 'number' || typeof entity === 'string') {
240
- return entity;
241
- }
242
- return (entity as Record<string, unknown>)[this.targetKey] as string | number | null ?? null;
243
- }
244
-
245
- private get relationKey(): RelationKey {
246
- return `${this.rootTable.name}.${this.relationName}`;
247
- }
248
-
249
- private get targetKey(): string {
250
- return this.relation.targetKey || findPrimaryKey(this.relation.target);
251
- }
252
-
253
- private filterPivotPayload(pivot?: Record<string, unknown>): Record<string, unknown> | undefined {
254
- if (!pivot) return undefined;
255
- const payload: Record<string, unknown> = {};
256
- for (const [key, value] of Object.entries(pivot)) {
257
- if (value === undefined) continue;
258
- if (!this.relation.pivotTable.columns[key]) continue;
259
- if (key === this.relation.pivotForeignKeyToRoot || key === this.relation.pivotForeignKeyToTarget) continue;
260
- payload[key] = value;
261
- }
262
- return Object.keys(payload).length ? payload : undefined;
263
- }
264
-
265
- private hydrateFromCache(): void {
266
- const keyValue = (this.root as Record<string, unknown>)[this.localKey];
267
- if (keyValue === undefined || keyValue === null) return;
268
- const rows = getHydrationRows(this.meta, this.relationName, keyValue);
269
- if (!rows?.length) return;
270
- this.items = rows.map(row => {
271
- const entity = this.createEntity(row);
272
- if ((row as { _pivot?: unknown })._pivot) {
273
- (entity as { _pivot?: unknown })._pivot = (row as { _pivot?: unknown })._pivot;
274
- }
275
- return entity;
276
- });
277
- this.loaded = true;
278
- }
279
-
280
- toJSON(): unknown[] {
281
- return this.items.map(item => {
282
- const entityWithToJSON = item as { toJSON?: () => unknown };
283
- return typeof entityWithToJSON.toJSON === 'function'
284
- ? entityWithToJSON.toJSON()
285
- : item;
286
- });
287
- }
288
- }
1
+ import { ManyToManyCollection } from '../../schema/types.js';
2
+ import { EntityContext } from '../entity-context.js';
3
+ import { RelationKey } from '../runtime-types.js';
4
+ import { BelongsToManyRelation } from '../../schema/relation.js';
5
+ import { TableDef } from '../../schema/table.js';
6
+ import { findPrimaryKey } from '../../query-builder/hydration-planner.js';
7
+ import { EntityMeta, getHydrationRows } from '../entity-meta.js';
8
+
9
+ type Rows = Record<string, unknown>[];
10
+
11
+ const toKey = (value: unknown): string => (value === null || value === undefined ? '' : String(value));
12
+
13
+ const hideInternal = (obj: object, keys: string[]): void => {
14
+ for (const key of keys) {
15
+ Object.defineProperty(obj, key, {
16
+ value: obj[key],
17
+ writable: false,
18
+ configurable: false,
19
+ enumerable: false
20
+ });
21
+ }
22
+ };
23
+
24
+ const hideWritable = (obj: object, keys: string[]): void => {
25
+ for (const key of keys) {
26
+ const value = obj[key as keyof typeof obj];
27
+ Object.defineProperty(obj, key, {
28
+ value,
29
+ writable: true,
30
+ configurable: true,
31
+ enumerable: false
32
+ });
33
+ }
34
+ };
35
+
36
+ const normalizePivot = (pivot: unknown): Record<string, unknown> | undefined => {
37
+ if (!pivot || typeof pivot !== 'object' || Array.isArray(pivot)) return undefined;
38
+ const entries = Object.entries(pivot as Record<string, unknown>)
39
+ .filter(([, value]) => value !== undefined);
40
+ if (!entries.length) return undefined;
41
+ return Object.fromEntries(entries);
42
+ };
43
+
44
+ const applyPivot = (entity: Record<string, unknown>, pivot?: Record<string, unknown>): void => {
45
+ if (!pivot) return;
46
+ const current = entity._pivot;
47
+ if (current && typeof current === 'object' && !Array.isArray(current)) {
48
+ entity._pivot = { ...(current as Record<string, unknown>), ...pivot };
49
+ return;
50
+ }
51
+ entity._pivot = { ...pivot };
52
+ };
53
+
54
+ /**
55
+ * Default implementation of a many-to-many collection.
56
+ * Manages the relationship between two entities through a pivot table.
57
+ * Supports lazy loading, attaching/detaching entities, and syncing by IDs.
58
+ *
59
+ * @template TTarget The type of the target entities in the collection.
60
+ */
61
+ export class DefaultManyToManyCollection<TTarget, TPivot extends object | undefined = undefined>
62
+ implements ManyToManyCollection<TTarget, TPivot> {
63
+ private loaded = false;
64
+ private items: TTarget[] = [];
65
+
66
+ /**
67
+ * @param ctx The entity context for tracking changes.
68
+ * @param meta Metadata for the root entity.
69
+ * @param root The root entity instance.
70
+ * @param relationName The name of the relation.
71
+ * @param relation Relation definition.
72
+ * @param rootTable Table definition of the root entity.
73
+ * @param loader Function to load the collection items.
74
+ * @param createEntity Function to create entity instances from rows.
75
+ * @param localKey The local key used for joining.
76
+ */
77
+ constructor(
78
+ private readonly ctx: EntityContext,
79
+ private readonly meta: EntityMeta<TableDef>,
80
+ private readonly root: unknown,
81
+ private readonly relationName: string,
82
+ private readonly relation: BelongsToManyRelation,
83
+ private readonly rootTable: TableDef,
84
+ private readonly loader: () => Promise<Map<string, Rows>>,
85
+ private readonly createEntity: (row: Record<string, unknown>) => TTarget,
86
+ private readonly localKey: string
87
+ ) {
88
+ hideInternal(this, ['ctx', 'meta', 'root', 'relationName', 'relation', 'rootTable', 'loader', 'createEntity', 'localKey']);
89
+ hideWritable(this, ['loaded', 'items']);
90
+ this.hydrateFromCache();
91
+ }
92
+
93
+ /**
94
+ * Loads the collection items if not already loaded.
95
+ * @returns A promise that resolves to the array of target entities.
96
+ */
97
+ async load(): Promise<TTarget[]> {
98
+ if (this.loaded) return this.items;
99
+ const map = await this.loader();
100
+ const key = toKey(this.root[this.localKey]);
101
+ const rows = map.get(key) ?? [];
102
+ this.items = rows.map(row => {
103
+ const entity = this.createEntity(row);
104
+ if ((row as { _pivot?: unknown })._pivot) {
105
+ (entity as { _pivot?: unknown })._pivot = (row as { _pivot?: unknown })._pivot;
106
+ }
107
+ return entity;
108
+ });
109
+ this.loaded = true;
110
+ return this.items;
111
+ }
112
+
113
+ /**
114
+ * Returns the currently loaded items.
115
+ * @returns Array of target entities.
116
+ */
117
+ getItems(): TTarget[] {
118
+ return this.items;
119
+ }
120
+
121
+ /**
122
+ * Array-compatible length for testing frameworks.
123
+ */
124
+ get length(): number {
125
+ return this.items.length;
126
+ }
127
+
128
+ /**
129
+ * Enables iteration over the collection like an array.
130
+ */
131
+ [Symbol.iterator](): Iterator<TTarget> {
132
+ return this.items[Symbol.iterator]();
133
+ }
134
+
135
+ /**
136
+ * Attaches an entity to the collection.
137
+ * Registers an 'attach' change in the entity context.
138
+ * @param target Entity instance or its primary key value.
139
+ */
140
+ attach(target: TTarget | number | string, pivot?: Partial<TPivot> | Record<string, unknown>): void {
141
+ const entity = this.ensureEntity(target);
142
+ const id = this.extractId(entity);
143
+ const pivotPayload = this.filterPivotPayload(normalizePivot(pivot));
144
+ const existing = id != null
145
+ ? this.items.find(item => this.extractId(item) === id)
146
+ : this.items.find(item => item === entity);
147
+
148
+ if (existing) {
149
+ if (pivotPayload) {
150
+ applyPivot(existing as Record<string, unknown>, pivotPayload);
151
+ this.ctx.registerRelationChange(
152
+ this.root,
153
+ this.relationKey,
154
+ this.rootTable,
155
+ this.relationName,
156
+ this.relation,
157
+ { kind: 'update', entity: existing, pivot: pivotPayload }
158
+ );
159
+ }
160
+ return;
161
+ }
162
+
163
+ if (pivotPayload) {
164
+ applyPivot(entity as Record<string, unknown>, pivotPayload);
165
+ }
166
+ this.items.push(entity);
167
+ this.ctx.registerRelationChange(
168
+ this.root,
169
+ this.relationKey,
170
+ this.rootTable,
171
+ this.relationName,
172
+ this.relation,
173
+ { kind: 'attach', entity, pivot: pivotPayload }
174
+ );
175
+ }
176
+
177
+ /**
178
+ * Detaches an entity from the collection.
179
+ * Registers a 'detach' change in the entity context.
180
+ * @param target Entity instance or its primary key value.
181
+ */
182
+ detach(target: TTarget | number | string): void {
183
+ const id = typeof target === 'number' || typeof target === 'string'
184
+ ? target
185
+ : this.extractId(target);
186
+
187
+ if (id == null) return;
188
+
189
+ const existing = this.items.find(item => this.extractId(item) === id);
190
+ if (!existing) return;
191
+
192
+ this.items = this.items.filter(item => this.extractId(item) !== id);
193
+ this.ctx.registerRelationChange(
194
+ this.root,
195
+ this.relationKey,
196
+ this.rootTable,
197
+ this.relationName,
198
+ this.relation,
199
+ { kind: 'detach', entity: existing }
200
+ );
201
+ }
202
+
203
+ /**
204
+ * Syncs the collection with a list of IDs.
205
+ * Attaches missing IDs and detaches IDs not in the list.
206
+ * @param ids Array of primary key values to sync with.
207
+ */
208
+ async syncByIds(ids: (number | string)[]): Promise<void> {
209
+ await this.load();
210
+ const normalized = new Set(ids.map(id => toKey(id)));
211
+ const currentIds = new Set(this.items.map(item => toKey(this.extractId(item))));
212
+
213
+ for (const id of normalized) {
214
+ if (!currentIds.has(id)) {
215
+ this.attach(id);
216
+ }
217
+ }
218
+
219
+ for (const item of [...this.items]) {
220
+ const itemId = toKey(this.extractId(item));
221
+ if (!normalized.has(itemId)) {
222
+ this.detach(item);
223
+ }
224
+ }
225
+ }
226
+
227
+ private ensureEntity(target: TTarget | number | string): TTarget {
228
+ if (typeof target === 'number' || typeof target === 'string') {
229
+ const stub: Record<string, unknown> = {
230
+ [this.targetKey]: target
231
+ };
232
+ return this.createEntity(stub);
233
+ }
234
+ return target;
235
+ }
236
+
237
+ private extractId(entity: TTarget | number | string | null | undefined): number | string | null {
238
+ if (entity === null || entity === undefined) return null;
239
+ if (typeof entity === 'number' || typeof entity === 'string') {
240
+ return entity;
241
+ }
242
+ return (entity as Record<string, unknown>)[this.targetKey] as string | number | null ?? null;
243
+ }
244
+
245
+ private get relationKey(): RelationKey {
246
+ return `${this.rootTable.name}.${this.relationName}`;
247
+ }
248
+
249
+ private get targetKey(): string {
250
+ return this.relation.targetKey || findPrimaryKey(this.relation.target);
251
+ }
252
+
253
+ private filterPivotPayload(pivot?: Record<string, unknown>): Record<string, unknown> | undefined {
254
+ if (!pivot) return undefined;
255
+ const payload: Record<string, unknown> = {};
256
+ for (const [key, value] of Object.entries(pivot)) {
257
+ if (value === undefined) continue;
258
+ if (!this.relation.pivotTable.columns[key]) continue;
259
+ if (key === this.relation.pivotForeignKeyToRoot || key === this.relation.pivotForeignKeyToTarget) continue;
260
+ payload[key] = value;
261
+ }
262
+ return Object.keys(payload).length ? payload : undefined;
263
+ }
264
+
265
+ private hydrateFromCache(): void {
266
+ const keyValue = (this.root as Record<string, unknown>)[this.localKey];
267
+ if (keyValue === undefined || keyValue === null) return;
268
+ const rows = getHydrationRows(this.meta, this.relationName, keyValue);
269
+ if (!rows?.length) return;
270
+ this.items = rows.map(row => {
271
+ const entity = this.createEntity(row);
272
+ if ((row as { _pivot?: unknown })._pivot) {
273
+ (entity as { _pivot?: unknown })._pivot = (row as { _pivot?: unknown })._pivot;
274
+ }
275
+ return entity;
276
+ });
277
+ this.loaded = true;
278
+ }
279
+
280
+ toJSON(): unknown[] {
281
+ return this.items.map(item => {
282
+ const entityWithToJSON = item as { toJSON?: () => unknown };
283
+ return typeof entityWithToJSON.toJSON === 'function'
284
+ ? entityWithToJSON.toJSON()
285
+ : item;
286
+ });
287
+ }
288
+ }