metal-orm 1.0.96 → 1.0.98

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.
@@ -1,108 +1,108 @@
1
- import { TableDef } from '../../schema/table.js';
2
- import { BelongsToRelation } from '../../schema/relation.js';
3
- import { findPrimaryKey } from '../../query-builder/hydration-planner.js';
4
- import { RelationIncludeOptions } from '../../query-builder/relation-types.js';
5
- import { EntityContext } from '../entity-context.js';
6
- import {
7
- buildColumnSelection,
8
- collectKeysFromRoots,
9
- fetchRowsForKeys,
10
- filterRow,
11
- groupRowsByUnique,
12
- hasColumns
13
- } from './shared.js';
14
-
15
- /**
16
- * Loads related entities for a belongs-to relation in batch.
17
- * @param ctx - The entity context.
18
- * @param rootTable - The root table of the relation.
19
- * @param _relationName - The name of the relation (unused).
20
- * @param relation - The belongs-to relation definition.
21
- * @returns A promise resolving to a map of foreign keys to single related rows.
22
- */
23
- export const loadBelongsToRelation = async (
24
- ctx: EntityContext,
25
- rootTable: TableDef,
26
- relationName: string,
27
- relation: BelongsToRelation,
28
- options?: RelationIncludeOptions
29
- ): Promise<Map<string, Record<string, unknown>>> => {
30
- const roots = ctx.getEntitiesForTable(rootTable);
31
-
32
- const getForeignKeys = (): Set<unknown> => collectKeysFromRoots(roots, relation.foreignKey);
33
- let foreignKeys = getForeignKeys();
34
-
35
- if (!foreignKeys.size) {
36
- const pkName = findPrimaryKey(rootTable);
37
- const pkColumn = rootTable.columns[pkName];
38
- const fkColumn = rootTable.columns[relation.foreignKey];
39
-
40
- if (pkColumn && fkColumn) {
41
- const missingKeys = new Set<unknown>();
42
- const entityByPk = new Map<unknown, Record<string, unknown>>();
43
-
44
- for (const tracked of roots) {
45
- const entity = tracked.entity as Record<string, unknown>;
46
- const pkValue = entity[pkName];
47
- if (pkValue === undefined || pkValue === null) continue;
48
- const fkValue = entity[relation.foreignKey];
49
- if (fkValue === undefined || fkValue === null) {
50
- missingKeys.add(pkValue);
51
- entityByPk.set(pkValue, entity);
52
- }
53
- }
54
-
55
- if (missingKeys.size) {
56
- const selection = buildColumnSelection(
57
- rootTable,
58
- [pkName, relation.foreignKey],
59
- column => `Column '${column}' not found on table '${rootTable.name}'`
60
- );
61
- const keyRows = await fetchRowsForKeys(ctx, rootTable, pkColumn, missingKeys, selection);
62
- for (const row of keyRows) {
63
- const pkValue = row[pkName];
64
- if (pkValue === undefined || pkValue === null) continue;
65
- const entity = entityByPk.get(pkValue);
66
- if (!entity) continue;
67
- const fkValue = row[relation.foreignKey];
68
- if (fkValue !== undefined && fkValue !== null) {
69
- entity[relation.foreignKey] = fkValue;
70
- }
71
- }
72
- foreignKeys = getForeignKeys();
73
- }
74
- }
75
- }
76
-
77
- if (!foreignKeys.size) {
78
- return new Map();
79
- }
80
-
81
- const targetKey = relation.localKey || findPrimaryKey(relation.target);
82
- const pkColumn = relation.target.columns[targetKey];
83
- if (!pkColumn) return new Map();
84
-
85
- const requestedColumns = hasColumns(options?.columns) ? [...options!.columns] : undefined;
86
- const selectedColumns = requestedColumns ? [...requestedColumns] : Object.keys(relation.target.columns);
87
- if (!selectedColumns.includes(targetKey)) {
88
- selectedColumns.push(targetKey);
89
- }
90
-
91
- const selection = buildColumnSelection(
92
- relation.target,
93
- selectedColumns,
94
- column => `Column '${column}' not found on relation '${relationName}'`
95
- );
96
-
97
- const rows = await fetchRowsForKeys(ctx, relation.target, pkColumn, foreignKeys, selection, options?.filter);
98
- const grouped = groupRowsByUnique(rows, targetKey);
99
-
100
- if (!requestedColumns) return grouped;
101
-
102
- const visibleColumns = new Set(selectedColumns);
103
- const filtered = new Map<string, Record<string, unknown>>();
104
- for (const [key, row] of grouped.entries()) {
105
- filtered.set(key, filterRow(row, visibleColumns));
106
- }
107
- return filtered;
108
- };
1
+ import { TableDef } from '../../schema/table.js';
2
+ import { BelongsToRelation } from '../../schema/relation.js';
3
+ import { findPrimaryKey } from '../../query-builder/hydration-planner.js';
4
+ import { RelationIncludeOptions } from '../../query-builder/relation-types.js';
5
+ import { EntityContext } from '../entity-context.js';
6
+ import {
7
+ buildColumnSelection,
8
+ collectKeysFromRoots,
9
+ fetchRowsForKeys,
10
+ filterRow,
11
+ groupRowsByUnique,
12
+ hasColumns
13
+ } from './shared.js';
14
+
15
+ /**
16
+ * Loads related entities for a belongs-to relation in batch.
17
+ * @param ctx - The entity context.
18
+ * @param rootTable - The root table of the relation.
19
+ * @param _relationName - The name of the relation (unused).
20
+ * @param relation - The belongs-to relation definition.
21
+ * @returns A promise resolving to a map of foreign keys to single related rows.
22
+ */
23
+ export const loadBelongsToRelation = async (
24
+ ctx: EntityContext,
25
+ rootTable: TableDef,
26
+ relationName: string,
27
+ relation: BelongsToRelation,
28
+ options?: RelationIncludeOptions
29
+ ): Promise<Map<string, Record<string, unknown>>> => {
30
+ const roots = ctx.getEntitiesForTable(rootTable);
31
+
32
+ const getForeignKeys = (): Set<unknown> => collectKeysFromRoots(roots, relation.foreignKey);
33
+ let foreignKeys = getForeignKeys();
34
+
35
+ if (!foreignKeys.size) {
36
+ const pkName = findPrimaryKey(rootTable);
37
+ const pkColumn = rootTable.columns[pkName];
38
+ const fkColumn = rootTable.columns[relation.foreignKey];
39
+
40
+ if (pkColumn && fkColumn) {
41
+ const missingKeys = new Set<unknown>();
42
+ const entityByPk = new Map<unknown, Record<string, unknown>>();
43
+
44
+ for (const tracked of roots) {
45
+ const entity = tracked.entity as Record<string, unknown>;
46
+ const pkValue = entity[pkName];
47
+ if (pkValue === undefined || pkValue === null) continue;
48
+ const fkValue = entity[relation.foreignKey];
49
+ if (fkValue === undefined || fkValue === null) {
50
+ missingKeys.add(pkValue);
51
+ entityByPk.set(pkValue, entity);
52
+ }
53
+ }
54
+
55
+ if (missingKeys.size) {
56
+ const selection = buildColumnSelection(
57
+ rootTable,
58
+ [pkName, relation.foreignKey],
59
+ column => `Column '${column}' not found on table '${rootTable.name}'`
60
+ );
61
+ const keyRows = await fetchRowsForKeys(ctx, rootTable, pkColumn, missingKeys, selection);
62
+ for (const row of keyRows) {
63
+ const pkValue = row[pkName];
64
+ if (pkValue === undefined || pkValue === null) continue;
65
+ const entity = entityByPk.get(pkValue);
66
+ if (!entity) continue;
67
+ const fkValue = row[relation.foreignKey];
68
+ if (fkValue !== undefined && fkValue !== null) {
69
+ entity[relation.foreignKey] = fkValue;
70
+ }
71
+ }
72
+ foreignKeys = getForeignKeys();
73
+ }
74
+ }
75
+ }
76
+
77
+ if (!foreignKeys.size) {
78
+ return new Map();
79
+ }
80
+
81
+ const targetKey = relation.localKey || findPrimaryKey(relation.target);
82
+ const pkColumn = relation.target.columns[targetKey];
83
+ if (!pkColumn) return new Map();
84
+
85
+ const requestedColumns = hasColumns(options?.columns) ? [...options!.columns] : undefined;
86
+ const selectedColumns = requestedColumns ? [...requestedColumns] : Object.keys(relation.target.columns);
87
+ if (!selectedColumns.includes(targetKey)) {
88
+ selectedColumns.push(targetKey);
89
+ }
90
+
91
+ const selection = buildColumnSelection(
92
+ relation.target,
93
+ selectedColumns,
94
+ column => `Column '${column}' not found on relation '${relationName}'`
95
+ );
96
+
97
+ const rows = await fetchRowsForKeys(ctx, relation.target, pkColumn, foreignKeys, selection, options?.filter);
98
+ const grouped = groupRowsByUnique(rows, targetKey);
99
+
100
+ if (!requestedColumns) return grouped;
101
+
102
+ const visibleColumns = new Set(selectedColumns);
103
+ const filtered = new Map<string, Record<string, unknown>>();
104
+ for (const [key, row] of grouped.entries()) {
105
+ filtered.set(key, filterRow(row, visibleColumns));
106
+ }
107
+ return filtered;
108
+ };
@@ -1,69 +1,69 @@
1
- import { TableDef } from '../../schema/table.js';
2
- import { HasManyRelation } from '../../schema/relation.js';
3
- import { findPrimaryKey } from '../../query-builder/hydration-planner.js';
4
- import { RelationIncludeOptions } from '../../query-builder/relation-types.js';
5
- import { EntityContext } from '../entity-context.js';
6
- import {
7
- buildColumnSelection,
8
- collectKeysFromRoots,
9
- fetchRowsForKeys,
10
- filterRows,
11
- groupRowsByMany,
12
- hasColumns,
13
- Rows
14
- } from './shared.js';
15
-
16
- /**
17
- * Loads related entities for a has-many relation in batch.
18
- * @param ctx - The entity context.
19
- * @param rootTable - The root table of the relation.
20
- * @param _relationName - The name of the relation (unused).
21
- * @param relation - The has-many relation definition.
22
- * @returns A promise resolving to a map of root keys to arrays of related rows.
23
- */
24
- export const loadHasManyRelation = async (
25
- ctx: EntityContext,
26
- rootTable: TableDef,
27
- relationName: string,
28
- relation: HasManyRelation,
29
- options?: RelationIncludeOptions
30
- ): Promise<Map<string, Rows>> => {
31
- const localKey = relation.localKey || findPrimaryKey(rootTable);
32
- const roots = ctx.getEntitiesForTable(rootTable);
33
- const keys = collectKeysFromRoots(roots, localKey);
34
-
35
- if (!keys.size) {
36
- return new Map();
37
- }
38
-
39
- const fkColumn = relation.target.columns[relation.foreignKey];
40
- if (!fkColumn) return new Map();
41
-
42
- const requestedColumns = hasColumns(options?.columns) ? [...options!.columns] : undefined;
43
- const targetPrimaryKey = findPrimaryKey(relation.target);
44
- const selectedColumns = requestedColumns ? [...requestedColumns] : Object.keys(relation.target.columns);
45
- if (!selectedColumns.includes(targetPrimaryKey)) {
46
- selectedColumns.push(targetPrimaryKey);
47
- }
48
-
49
- const queryColumns = new Set(selectedColumns);
50
- queryColumns.add(relation.foreignKey);
51
-
52
- const selection = buildColumnSelection(
53
- relation.target,
54
- Array.from(queryColumns),
55
- column => `Column '${column}' not found on relation '${relationName}'`
56
- );
57
-
58
- const rows = await fetchRowsForKeys(ctx, relation.target, fkColumn, keys, selection, options?.filter);
59
- const grouped = groupRowsByMany(rows, relation.foreignKey);
60
-
61
- if (!requestedColumns) return grouped;
62
-
63
- const visibleColumns = new Set(selectedColumns);
64
- const filtered = new Map<string, Rows>();
65
- for (const [key, bucket] of grouped.entries()) {
66
- filtered.set(key, filterRows(bucket, visibleColumns));
67
- }
68
- return filtered;
69
- };
1
+ import { TableDef } from '../../schema/table.js';
2
+ import { HasManyRelation } from '../../schema/relation.js';
3
+ import { findPrimaryKey } from '../../query-builder/hydration-planner.js';
4
+ import { RelationIncludeOptions } from '../../query-builder/relation-types.js';
5
+ import { EntityContext } from '../entity-context.js';
6
+ import {
7
+ buildColumnSelection,
8
+ collectKeysFromRoots,
9
+ fetchRowsForKeys,
10
+ filterRows,
11
+ groupRowsByMany,
12
+ hasColumns,
13
+ Rows
14
+ } from './shared.js';
15
+
16
+ /**
17
+ * Loads related entities for a has-many relation in batch.
18
+ * @param ctx - The entity context.
19
+ * @param rootTable - The root table of the relation.
20
+ * @param _relationName - The name of the relation (unused).
21
+ * @param relation - The has-many relation definition.
22
+ * @returns A promise resolving to a map of root keys to arrays of related rows.
23
+ */
24
+ export const loadHasManyRelation = async (
25
+ ctx: EntityContext,
26
+ rootTable: TableDef,
27
+ relationName: string,
28
+ relation: HasManyRelation,
29
+ options?: RelationIncludeOptions
30
+ ): Promise<Map<string, Rows>> => {
31
+ const localKey = relation.localKey || findPrimaryKey(rootTable);
32
+ const roots = ctx.getEntitiesForTable(rootTable);
33
+ const keys = collectKeysFromRoots(roots, localKey);
34
+
35
+ if (!keys.size) {
36
+ return new Map();
37
+ }
38
+
39
+ const fkColumn = relation.target.columns[relation.foreignKey];
40
+ if (!fkColumn) return new Map();
41
+
42
+ const requestedColumns = hasColumns(options?.columns) ? [...options!.columns] : undefined;
43
+ const targetPrimaryKey = findPrimaryKey(relation.target);
44
+ const selectedColumns = requestedColumns ? [...requestedColumns] : Object.keys(relation.target.columns);
45
+ if (!selectedColumns.includes(targetPrimaryKey)) {
46
+ selectedColumns.push(targetPrimaryKey);
47
+ }
48
+
49
+ const queryColumns = new Set(selectedColumns);
50
+ queryColumns.add(relation.foreignKey);
51
+
52
+ const selection = buildColumnSelection(
53
+ relation.target,
54
+ Array.from(queryColumns),
55
+ column => `Column '${column}' not found on relation '${relationName}'`
56
+ );
57
+
58
+ const rows = await fetchRowsForKeys(ctx, relation.target, fkColumn, keys, selection, options?.filter);
59
+ const grouped = groupRowsByMany(rows, relation.foreignKey);
60
+
61
+ if (!requestedColumns) return grouped;
62
+
63
+ const visibleColumns = new Set(selectedColumns);
64
+ const filtered = new Map<string, Rows>();
65
+ for (const [key, bucket] of grouped.entries()) {
66
+ filtered.set(key, filterRows(bucket, visibleColumns));
67
+ }
68
+ return filtered;
69
+ };
@@ -1,68 +1,68 @@
1
- import { TableDef } from '../../schema/table.js';
2
- import { HasOneRelation } from '../../schema/relation.js';
3
- import { findPrimaryKey } from '../../query-builder/hydration-planner.js';
4
- import { RelationIncludeOptions } from '../../query-builder/relation-types.js';
5
- import { EntityContext } from '../entity-context.js';
6
- import {
7
- buildColumnSelection,
8
- collectKeysFromRoots,
9
- fetchRowsForKeys,
10
- filterRow,
11
- groupRowsByUnique,
12
- hasColumns
13
- } from './shared.js';
14
-
15
- /**
16
- * Loads related entities for a has-one relation in batch.
17
- * @param ctx - The entity context.
18
- * @param rootTable - The root table of the relation.
19
- * @param _relationName - The name of the relation (unused).
20
- * @param relation - The has-one relation definition.
21
- * @returns A promise resolving to a map of root keys to single related rows.
22
- */
23
- export const loadHasOneRelation = async (
24
- ctx: EntityContext,
25
- rootTable: TableDef,
26
- relationName: string,
27
- relation: HasOneRelation,
28
- options?: RelationIncludeOptions
29
- ): Promise<Map<string, Record<string, unknown>>> => {
30
- const localKey = relation.localKey || findPrimaryKey(rootTable);
31
- const roots = ctx.getEntitiesForTable(rootTable);
32
- const keys = collectKeysFromRoots(roots, localKey);
33
-
34
- if (!keys.size) {
35
- return new Map();
36
- }
37
-
38
- const fkColumn = relation.target.columns[relation.foreignKey];
39
- if (!fkColumn) return new Map();
40
-
41
- const requestedColumns = hasColumns(options?.columns) ? [...options!.columns] : undefined;
42
- const targetPrimaryKey = findPrimaryKey(relation.target);
43
- const selectedColumns = requestedColumns ? [...requestedColumns] : Object.keys(relation.target.columns);
44
- if (!selectedColumns.includes(targetPrimaryKey)) {
45
- selectedColumns.push(targetPrimaryKey);
46
- }
47
-
48
- const queryColumns = new Set(selectedColumns);
49
- queryColumns.add(relation.foreignKey);
50
-
51
- const selection = buildColumnSelection(
52
- relation.target,
53
- Array.from(queryColumns),
54
- column => `Column '${column}' not found on relation '${relationName}'`
55
- );
56
-
57
- const rows = await fetchRowsForKeys(ctx, relation.target, fkColumn, keys, selection, options?.filter);
58
- const grouped = groupRowsByUnique(rows, relation.foreignKey);
59
-
60
- if (!requestedColumns) return grouped;
61
-
62
- const visibleColumns = new Set(selectedColumns);
63
- const filtered = new Map<string, Record<string, unknown>>();
64
- for (const [key, row] of grouped.entries()) {
65
- filtered.set(key, filterRow(row, visibleColumns));
66
- }
67
- return filtered;
68
- };
1
+ import { TableDef } from '../../schema/table.js';
2
+ import { HasOneRelation } from '../../schema/relation.js';
3
+ import { findPrimaryKey } from '../../query-builder/hydration-planner.js';
4
+ import { RelationIncludeOptions } from '../../query-builder/relation-types.js';
5
+ import { EntityContext } from '../entity-context.js';
6
+ import {
7
+ buildColumnSelection,
8
+ collectKeysFromRoots,
9
+ fetchRowsForKeys,
10
+ filterRow,
11
+ groupRowsByUnique,
12
+ hasColumns
13
+ } from './shared.js';
14
+
15
+ /**
16
+ * Loads related entities for a has-one relation in batch.
17
+ * @param ctx - The entity context.
18
+ * @param rootTable - The root table of the relation.
19
+ * @param _relationName - The name of the relation (unused).
20
+ * @param relation - The has-one relation definition.
21
+ * @returns A promise resolving to a map of root keys to single related rows.
22
+ */
23
+ export const loadHasOneRelation = async (
24
+ ctx: EntityContext,
25
+ rootTable: TableDef,
26
+ relationName: string,
27
+ relation: HasOneRelation,
28
+ options?: RelationIncludeOptions
29
+ ): Promise<Map<string, Record<string, unknown>>> => {
30
+ const localKey = relation.localKey || findPrimaryKey(rootTable);
31
+ const roots = ctx.getEntitiesForTable(rootTable);
32
+ const keys = collectKeysFromRoots(roots, localKey);
33
+
34
+ if (!keys.size) {
35
+ return new Map();
36
+ }
37
+
38
+ const fkColumn = relation.target.columns[relation.foreignKey];
39
+ if (!fkColumn) return new Map();
40
+
41
+ const requestedColumns = hasColumns(options?.columns) ? [...options!.columns] : undefined;
42
+ const targetPrimaryKey = findPrimaryKey(relation.target);
43
+ const selectedColumns = requestedColumns ? [...requestedColumns] : Object.keys(relation.target.columns);
44
+ if (!selectedColumns.includes(targetPrimaryKey)) {
45
+ selectedColumns.push(targetPrimaryKey);
46
+ }
47
+
48
+ const queryColumns = new Set(selectedColumns);
49
+ queryColumns.add(relation.foreignKey);
50
+
51
+ const selection = buildColumnSelection(
52
+ relation.target,
53
+ Array.from(queryColumns),
54
+ column => `Column '${column}' not found on relation '${relationName}'`
55
+ );
56
+
57
+ const rows = await fetchRowsForKeys(ctx, relation.target, fkColumn, keys, selection, options?.filter);
58
+ const grouped = groupRowsByUnique(rows, relation.foreignKey);
59
+
60
+ if (!requestedColumns) return grouped;
61
+
62
+ const visibleColumns = new Set(selectedColumns);
63
+ const filtered = new Map<string, Record<string, unknown>>();
64
+ for (const [key, row] of grouped.entries()) {
65
+ filtered.set(key, filterRow(row, visibleColumns));
66
+ }
67
+ return filtered;
68
+ };