metal-orm 1.0.60 → 1.0.62

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/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "metal-orm",
3
- "version": "1.0.60",
3
+ "version": "1.0.62",
4
4
  "type": "module",
5
5
  "types": "./dist/index.d.ts",
6
6
  "engines": {
@@ -3,25 +3,25 @@ import {
3
3
  hasMany,
4
4
  hasOne,
5
5
  belongsTo,
6
- belongsToMany,
7
- RelationKinds,
8
- type HasManyRelation,
9
- type HasOneRelation,
10
- type BelongsToRelation,
6
+ belongsToMany,
7
+ RelationKinds,
8
+ type HasManyRelation,
9
+ type HasOneRelation,
10
+ type BelongsToRelation,
11
11
  type BelongsToManyRelation,
12
12
  type RelationDef
13
13
  } from '../schema/relation.js';
14
- import { TableDef } from '../schema/table.js';
15
- import { isTableDef } from '../schema/table-guards.js';
16
- import {
17
- buildTableDef,
18
- EntityConstructor,
19
- EntityMetadata,
20
- EntityOrTableTarget,
21
- EntityOrTableTargetResolver,
22
- getAllEntityMetadata,
23
- getEntityMetadata
24
- } from '../orm/entity-metadata.js';
14
+ import { TableDef } from '../schema/table.js';
15
+ import { isTableDef } from '../schema/table-guards.js';
16
+ import {
17
+ buildTableDef,
18
+ EntityConstructor,
19
+ EntityMetadata,
20
+ EntityOrTableTarget,
21
+ EntityOrTableTargetResolver,
22
+ getAllEntityMetadata,
23
+ getEntityMetadata
24
+ } from '../orm/entity-metadata.js';
25
25
 
26
26
  import { tableRef, type TableRef } from '../schema/table.js';
27
27
  import {
@@ -41,10 +41,10 @@ const unwrapTarget = (target: EntityOrTableTargetResolver): EntityOrTableTarget
41
41
  return target as EntityOrTableTarget;
42
42
  };
43
43
 
44
- const resolveTableTarget = (
45
- target: EntityOrTableTargetResolver,
46
- tableMap: Map<EntityConstructor, TableDef>
47
- ): TableDef => {
44
+ const resolveTableTarget = (
45
+ target: EntityOrTableTargetResolver,
46
+ tableMap: Map<EntityConstructor, TableDef>
47
+ ): TableDef => {
48
48
  const resolved = unwrapTarget(target);
49
49
  if (isTableDef(resolved)) {
50
50
  return resolved;
@@ -53,65 +53,65 @@ const resolveTableTarget = (
53
53
  if (!table) {
54
54
  throw new Error(`Entity '${(resolved as EntityConstructor).name}' is not registered with decorators`);
55
55
  }
56
- return table;
57
- };
58
-
59
- const toSnakeCase = (value: string): string => {
60
- return value
61
- .replace(/([a-z0-9])([A-Z])/g, '$1_$2')
62
- .replace(/[^a-z0-9_]+/gi, '_')
63
- .replace(/__+/g, '_')
64
- .replace(/^_|_$/g, '')
65
- .toLowerCase();
66
- };
67
-
68
- const normalizeEntityName = (value: string): string => {
69
- const stripped = value.replace(/Entity$/i, '');
70
- const normalized = toSnakeCase(stripped || value);
71
- return normalized || 'unknown';
72
- };
73
-
74
- const getPivotKeyBaseFromTarget = (target: EntityOrTableTargetResolver): string => {
75
- const resolved = unwrapTarget(target);
76
- if (isTableDef(resolved)) {
77
- return toSnakeCase(resolved.name || 'unknown');
78
- }
79
- const ctor = resolved as EntityConstructor;
80
- return normalizeEntityName(ctor.name || 'unknown');
81
- };
82
-
83
- const getPivotKeyBaseFromRoot = (meta: EntityMetadata): string => {
84
- return normalizeEntityName(meta.target.name || meta.tableName || 'unknown');
85
- };
86
-
87
- const buildRelationDefinitions = (
88
- meta: EntityMetadata,
89
- tableMap: Map<EntityConstructor, TableDef>
90
- ): Record<string, RelationDef> => {
91
- const relations: Record<string, RelationDef> = {};
92
-
93
- for (const [name, relation] of Object.entries(meta.relations)) {
56
+ return table;
57
+ };
58
+
59
+ const toSnakeCase = (value: string): string => {
60
+ return value
61
+ .replace(/([a-z0-9])([A-Z])/g, '$1_$2')
62
+ .replace(/[^a-z0-9_]+/gi, '_')
63
+ .replace(/__+/g, '_')
64
+ .replace(/^_|_$/g, '')
65
+ .toLowerCase();
66
+ };
67
+
68
+ const normalizeEntityName = (value: string): string => {
69
+ const stripped = value.replace(/Entity$/i, '');
70
+ const normalized = toSnakeCase(stripped || value);
71
+ return normalized || 'unknown';
72
+ };
73
+
74
+ const getPivotKeyBaseFromTarget = (target: EntityOrTableTargetResolver): string => {
75
+ const resolved = unwrapTarget(target);
76
+ if (isTableDef(resolved)) {
77
+ return toSnakeCase(resolved.name || 'unknown');
78
+ }
79
+ const ctor = resolved as EntityConstructor;
80
+ return normalizeEntityName(ctor.name || 'unknown');
81
+ };
82
+
83
+ const getPivotKeyBaseFromRoot = (meta: EntityMetadata): string => {
84
+ return normalizeEntityName(meta.target.name || meta.tableName || 'unknown');
85
+ };
86
+
87
+ const buildRelationDefinitions = (
88
+ meta: EntityMetadata,
89
+ tableMap: Map<EntityConstructor, TableDef>
90
+ ): Record<string, RelationDef> => {
91
+ const relations: Record<string, RelationDef> = {};
92
+
93
+ for (const [name, relation] of Object.entries(meta.relations)) {
94
94
  switch (relation.kind) {
95
- case RelationKinds.HasOne: {
96
- const foreignKey = relation.foreignKey ?? `${getPivotKeyBaseFromRoot(meta)}_id`;
97
- relations[name] = hasOne(
98
- resolveTableTarget(relation.target, tableMap),
99
- foreignKey,
100
- relation.localKey,
101
- relation.cascade
102
- );
103
- break;
104
- }
105
- case RelationKinds.HasMany: {
106
- const foreignKey = relation.foreignKey ?? `${getPivotKeyBaseFromRoot(meta)}_id`;
107
- relations[name] = hasMany(
108
- resolveTableTarget(relation.target, tableMap),
109
- foreignKey,
110
- relation.localKey,
111
- relation.cascade
112
- );
113
- break;
114
- }
95
+ case RelationKinds.HasOne: {
96
+ const foreignKey = relation.foreignKey ?? `${getPivotKeyBaseFromRoot(meta)}_id`;
97
+ relations[name] = hasOne(
98
+ resolveTableTarget(relation.target, tableMap),
99
+ foreignKey,
100
+ relation.localKey,
101
+ relation.cascade
102
+ );
103
+ break;
104
+ }
105
+ case RelationKinds.HasMany: {
106
+ const foreignKey = relation.foreignKey ?? `${getPivotKeyBaseFromRoot(meta)}_id`;
107
+ relations[name] = hasMany(
108
+ resolveTableTarget(relation.target, tableMap),
109
+ foreignKey,
110
+ relation.localKey,
111
+ relation.cascade
112
+ );
113
+ break;
114
+ }
115
115
  case RelationKinds.BelongsTo: {
116
116
  relations[name] = belongsTo(
117
117
  resolveTableTarget(relation.target, tableMap),
@@ -121,22 +121,22 @@ const buildRelationDefinitions = (
121
121
  );
122
122
  break;
123
123
  }
124
- case RelationKinds.BelongsToMany: {
125
- const pivotForeignKeyToRoot =
126
- relation.pivotForeignKeyToRoot ?? `${getPivotKeyBaseFromRoot(meta)}_id`;
127
- const pivotForeignKeyToTarget =
128
- relation.pivotForeignKeyToTarget ?? `${getPivotKeyBaseFromTarget(relation.target)}_id`;
129
- relations[name] = belongsToMany(
130
- resolveTableTarget(relation.target, tableMap),
131
- resolveTableTarget(relation.pivotTable, tableMap),
132
- {
133
- pivotForeignKeyToRoot,
134
- pivotForeignKeyToTarget,
135
- localKey: relation.localKey,
136
- targetKey: relation.targetKey,
137
- pivotPrimaryKey: relation.pivotPrimaryKey,
138
- defaultPivotColumns: relation.defaultPivotColumns,
139
- cascade: relation.cascade
124
+ case RelationKinds.BelongsToMany: {
125
+ const pivotForeignKeyToRoot =
126
+ relation.pivotForeignKeyToRoot ?? `${getPivotKeyBaseFromRoot(meta)}_id`;
127
+ const pivotForeignKeyToTarget =
128
+ relation.pivotForeignKeyToTarget ?? `${getPivotKeyBaseFromTarget(relation.target)}_id`;
129
+ relations[name] = belongsToMany(
130
+ resolveTableTarget(relation.target, tableMap),
131
+ resolveTableTarget(relation.pivotTable, tableMap),
132
+ {
133
+ pivotForeignKeyToRoot,
134
+ pivotForeignKeyToTarget,
135
+ localKey: relation.localKey,
136
+ targetKey: relation.targetKey,
137
+ pivotPrimaryKey: relation.pivotPrimaryKey,
138
+ defaultPivotColumns: relation.defaultPivotColumns,
139
+ cascade: relation.cascade
140
140
  }
141
141
  );
142
142
  break;
@@ -200,31 +200,39 @@ type EntityTable<TEntity extends object> =
200
200
  Omit<TableDef<{ [K in SelectableKeys<TEntity>]: ColumnDef }>, 'relations'> & {
201
201
  relations: {
202
202
  [K in RelationKeys<TEntity>]:
203
- NonNullable<TEntity[K]> extends HasManyCollection<infer TChild>
204
- ? HasManyRelation<EntityTable<NonNullable<TChild> & object>>
205
- : NonNullable<TEntity[K]> extends ManyToManyCollection<infer TTarget, infer TPivot>
206
- ? BelongsToManyRelation<
207
- EntityTable<NonNullable<TTarget> & object>,
208
- TPivot extends object ? EntityTable<NonNullable<TPivot> & object> : TableDef
209
- >
210
- : NonNullable<TEntity[K]> extends HasOneReference<infer TChild>
211
- ? HasOneRelation<EntityTable<NonNullable<TChild> & object>>
212
- : NonNullable<TEntity[K]> extends BelongsToReference<infer TParent>
213
- ? BelongsToRelation<EntityTable<NonNullable<TParent> & object>>
214
- : NonNullable<TEntity[K]> extends object
215
- ? BelongsToRelation<EntityTable<NonNullable<TEntity[K]> & object>>
216
- : never;
203
+ NonNullable<TEntity[K]> extends HasManyCollection<infer TChild>
204
+ ? HasManyRelation<EntityTable<NonNullable<TChild> & object>>
205
+ : NonNullable<TEntity[K]> extends ManyToManyCollection<infer TTarget, infer TPivot>
206
+ ? BelongsToManyRelation<
207
+ EntityTable<NonNullable<TTarget> & object>,
208
+ TPivot extends object ? EntityTable<NonNullable<TPivot> & object> : TableDef
209
+ >
210
+ : NonNullable<TEntity[K]> extends HasOneReference<infer TChild>
211
+ ? HasOneRelation<EntityTable<NonNullable<TChild> & object>>
212
+ : NonNullable<TEntity[K]> extends BelongsToReference<infer TParent>
213
+ ? BelongsToRelation<EntityTable<NonNullable<TParent> & object>>
214
+ : NonNullable<TEntity[K]> extends object
215
+ ? BelongsToRelation<EntityTable<NonNullable<TEntity[K]> & object>>
216
+ : never;
217
217
  };
218
218
  };
219
219
 
220
220
  export const selectFromEntity = <TEntity extends object>(
221
221
  ctor: EntityConstructor<TEntity>
222
- ): SelectQueryBuilder<unknown, EntityTable<TEntity>> => {
222
+ ): SelectQueryBuilder<TEntity, EntityTable<TEntity>> => {
223
223
  const table = getTableDefFromEntity(ctor);
224
224
  if (!table) {
225
225
  throw new Error(`Entity '${ctor.name}' is not registered with decorators or has not been bootstrapped`);
226
226
  }
227
- return new SelectQueryBuilder(table as unknown as EntityTable<TEntity>);
227
+ return new SelectQueryBuilder(
228
+ table as unknown as EntityTable<TEntity>,
229
+ undefined,
230
+ undefined,
231
+ undefined,
232
+ undefined,
233
+ undefined,
234
+ ctor
235
+ );
228
236
  };
229
237
 
230
238
  /**
@@ -233,12 +241,12 @@ export const selectFromEntity = <TEntity extends object>(
233
241
  * Lazily bootstraps entity metadata (via getTableDefFromEntity) and returns a
234
242
  * `tableRef(...)`-style proxy so users can write `u.id` instead of `u.columns.id`.
235
243
  */
236
- export const entityRef = <TEntity extends object>(
237
- ctor: EntityConstructor<TEntity>
238
- ): TableRef<EntityTable<TEntity>> => {
239
- const table = getTableDefFromEntity(ctor);
240
- if (!table) {
241
- throw new Error(`Entity '${ctor.name}' is not registered with decorators or has not been bootstrapped`);
242
- }
243
- return tableRef(table as EntityTable<TEntity>);
244
- };
244
+ export const entityRef = <TEntity extends object>(
245
+ ctor: EntityConstructor<TEntity>
246
+ ): TableRef<EntityTable<TEntity>> => {
247
+ const table = getTableDefFromEntity(ctor);
248
+ if (!table) {
249
+ throw new Error(`Entity '${ctor.name}' is not registered with decorators or has not been bootstrapped`);
250
+ }
251
+ return tableRef(table as EntityTable<TEntity>);
252
+ };
@@ -6,3 +6,7 @@ export * from './column-decorator.js';
6
6
  export * from './relations.js';
7
7
  export * from './bootstrap.js';
8
8
  export { getDecoratorMetadata } from './decorator-metadata.js';
9
+
10
+ // Entity Materialization - convert query results to real class instances
11
+ export { materializeAs, DefaultEntityMaterializer, PrototypeMaterializationStrategy, ConstructorMaterializationStrategy } from '../orm/entity-materializer.js';
12
+ export type { EntityMaterializer, EntityMaterializationStrategy } from '../orm/entity-materializer.js';
@@ -0,0 +1,159 @@
1
+ import { EntityConstructor } from './entity-metadata.js';
2
+ import { rebuildRegistry } from './entity-registry.js';
3
+ import { hasEntityMeta } from './entity-meta.js';
4
+
5
+ /**
6
+ * Strategy interface for materializing entity instances (Open/Closed Principle).
7
+ */
8
+ export interface EntityMaterializationStrategy {
9
+ /**
10
+ * Creates an instance of the entity class and populates it with data.
11
+ * @param ctor - The entity constructor
12
+ * @param data - The raw data to populate
13
+ * @returns The materialized entity instance
14
+ */
15
+ materialize<T>(ctor: EntityConstructor<T>, data: Record<string, unknown>): T;
16
+ }
17
+
18
+ /**
19
+ * Default strategy: Uses Object.create() to create instance without calling constructor.
20
+ * Safe for classes with required constructor parameters.
21
+ */
22
+ export class PrototypeMaterializationStrategy implements EntityMaterializationStrategy {
23
+ materialize<T>(ctor: EntityConstructor<T>, data: Record<string, unknown>): T {
24
+ const instance = Object.create(ctor.prototype) as T;
25
+ Object.assign(instance, data);
26
+ return instance;
27
+ }
28
+ }
29
+
30
+ /**
31
+ * Alternative strategy: Calls default constructor then assigns properties.
32
+ * Use when class constructor initializes important state.
33
+ */
34
+ export class ConstructorMaterializationStrategy implements EntityMaterializationStrategy {
35
+ materialize<T>(ctor: EntityConstructor<T>, data: Record<string, unknown>): T {
36
+ const instance = Reflect.construct(ctor, []) as T & Record<string, unknown>;
37
+ Object.assign(instance, data);
38
+ return instance;
39
+ }
40
+ }
41
+
42
+ /**
43
+ * Interface for materializing query results into real entity class instances.
44
+ */
45
+ export interface EntityMaterializer {
46
+ /**
47
+ * Materializes a single row into a real entity instance.
48
+ * @param entityClass - The entity constructor
49
+ * @param row - The raw data row
50
+ * @returns The materialized entity instance
51
+ */
52
+ materialize<T>(entityClass: EntityConstructor<T>, row: Record<string, unknown>): T;
53
+
54
+ /**
55
+ * Materializes multiple rows into real entity instances.
56
+ * @param entityClass - The entity constructor
57
+ * @param rows - The raw data rows
58
+ * @returns Array of materialized entity instances
59
+ */
60
+ materializeMany<T>(entityClass: EntityConstructor<T>, rows: Record<string, unknown>[]): T[];
61
+ }
62
+
63
+ /**
64
+ * Default implementation of EntityMaterializer.
65
+ * Converts query results into actual class instances with working methods.
66
+ *
67
+ * @example
68
+ * const materializer = new DefaultEntityMaterializer();
69
+ * const users = materializer.materializeMany(User, queryResults);
70
+ * users[0] instanceof User; // true
71
+ * users[0].getFullName(); // works!
72
+ */
73
+ export class DefaultEntityMaterializer implements EntityMaterializer {
74
+ constructor(
75
+ private readonly strategy: EntityMaterializationStrategy = new ConstructorMaterializationStrategy()
76
+ ) { }
77
+
78
+ materialize<T>(ctor: EntityConstructor<T>, row: Record<string, unknown>): T {
79
+ if (hasEntityMeta(row)) {
80
+ return this.materializeEntityProxy(ctor, row);
81
+ }
82
+
83
+ const instance = this.strategy.materialize(ctor, row);
84
+ this.materializeRelations(instance as Record<string, unknown>);
85
+ return instance;
86
+ }
87
+
88
+ materializeMany<T>(ctor: EntityConstructor<T>, rows: Record<string, unknown>[]): T[] {
89
+ return rows.map(row => this.materialize(ctor, row));
90
+ }
91
+
92
+ /**
93
+ * Recursively materializes nested relation data.
94
+ */
95
+ private materializeRelations(instance: Record<string, unknown>): void {
96
+ // Rebuild registry to ensure we have latest metadata
97
+ rebuildRegistry();
98
+
99
+ for (const value of Object.values(instance)) {
100
+ if (value === null || value === undefined) continue;
101
+
102
+ // Handle has-one / belongs-to (single object)
103
+ if (typeof value === 'object' && !Array.isArray(value)) {
104
+ const nested = value as Record<string, unknown>;
105
+ // Check if this looks like an entity (has common entity patterns)
106
+ if (this.isEntityLike(nested)) {
107
+ // For now, keep as-is since we don't have relation metadata here
108
+ // Future: use relation metadata to get target constructor
109
+ }
110
+ }
111
+
112
+ // Handle has-many / belongs-to-many (array)
113
+ if (Array.isArray(value) && value.length > 0) {
114
+ const first = value[0];
115
+ if (typeof first === 'object' && first !== null && this.isEntityLike(first)) {
116
+ // For now, keep array as-is
117
+ // Future: materialize each item if we can resolve the target class
118
+ }
119
+ }
120
+ }
121
+ }
122
+
123
+ /**
124
+ * Simple heuristic to check if an object looks like an entity.
125
+ */
126
+ private isEntityLike(obj: Record<string, unknown>): boolean {
127
+ return 'id' in obj || Object.keys(obj).some(k =>
128
+ k.endsWith('Id') || k === 'createdAt' || k === 'updatedAt'
129
+ );
130
+ }
131
+
132
+ private materializeEntityProxy<T>(ctor: EntityConstructor<T>, row: Record<string, unknown>): T {
133
+ const proxy = row as Record<string, unknown>;
134
+ const baseline = this.strategy.materialize(ctor, {}) as Record<string, unknown>;
135
+ for (const key of Object.keys(baseline)) {
136
+ if (!Object.prototype.hasOwnProperty.call(proxy, key)) {
137
+ proxy[key] = baseline[key];
138
+ }
139
+ }
140
+ Object.setPrototypeOf(proxy, ctor.prototype);
141
+ return proxy as T;
142
+ }
143
+ }
144
+
145
+ /**
146
+ * Convenience function to materialize query results as real class instances.
147
+ *
148
+ * @example
149
+ * const results = await selectFromEntity(User).execute(session);
150
+ * const users = materializeAs(User, results);
151
+ * users[0] instanceof User; // true!
152
+ */
153
+ export const materializeAs = <TEntity extends object>(
154
+ ctor: EntityConstructor<TEntity>,
155
+ results: Record<string, unknown>[]
156
+ ): TEntity[] => {
157
+ const materializer = new DefaultEntityMaterializer();
158
+ return materializer.materializeMany(ctor, results);
159
+ };
@@ -0,0 +1,39 @@
1
+ import { TableDef } from '../schema/table.js';
2
+ import { EntityConstructor, getAllEntityMetadata } from './entity-metadata.js';
3
+
4
+ /**
5
+ * Reverse lookup registry: TableDef → EntityConstructor
6
+ */
7
+ const tableToConstructor = new Map<TableDef, EntityConstructor>();
8
+
9
+ /**
10
+ * Gets the entity constructor for a given table definition.
11
+ * @param table - The table definition
12
+ * @returns The entity constructor or undefined if not found
13
+ */
14
+ export const getConstructorForTable = (table: TableDef): EntityConstructor | undefined => {
15
+ if (!tableToConstructor.size) {
16
+ rebuildRegistry();
17
+ }
18
+ return tableToConstructor.get(table);
19
+ };
20
+
21
+ /**
22
+ * Rebuilds the registry from entity metadata.
23
+ * Called automatically on first lookup or when metadata changes.
24
+ */
25
+ export const rebuildRegistry = (): void => {
26
+ tableToConstructor.clear();
27
+ for (const meta of getAllEntityMetadata()) {
28
+ if (meta.table) {
29
+ tableToConstructor.set(meta.table, meta.target);
30
+ }
31
+ }
32
+ };
33
+
34
+ /**
35
+ * Clears the entity registry.
36
+ */
37
+ export const clearEntityRegistry = (): void => {
38
+ tableToConstructor.clear();
39
+ };
@@ -61,26 +61,56 @@ const executeWithContexts = async <TTable extends TableDef>(
61
61
  await loadLazyRelationsForTable(entityCtx, qb.getTable(), lazyRelations, lazyRelationOptions);
62
62
  return entities;
63
63
  };
64
-
65
- /**
66
- * Executes a hydrated query using the ORM session.
67
- * @template TTable - The table type
68
- * @param session - The ORM session
69
- * @param qb - The select query builder
64
+
65
+ const executePlainWithContexts = async <TTable extends TableDef>(
66
+ execCtx: ExecutionContext,
67
+ qb: SelectQueryBuilder<unknown, TTable>
68
+ ): Promise<Record<string, unknown>[]> => {
69
+ const ast = qb.getAST();
70
+ const compiled = execCtx.dialect.compileSelect(ast);
71
+ const executed = await execCtx.interceptors.run({ sql: compiled.sql, params: compiled.params }, execCtx.executor);
72
+ const rows = flattenResults(executed);
73
+
74
+ if (ast.setOps && ast.setOps.length > 0) {
75
+ return rows;
76
+ }
77
+
78
+ return hydrateRows(rows, qb.getHydrationPlan());
79
+ };
80
+
81
+ /**
82
+ * Executes a hydrated query using the ORM session.
83
+ * @template TTable - The table type
84
+ * @param session - The ORM session
85
+ * @param qb - The select query builder
70
86
  * @returns Promise resolving to array of entity instances
71
87
  */
72
- export async function executeHydrated<TTable extends TableDef>(
73
- session: OrmSession,
74
- qb: SelectQueryBuilder<unknown, TTable>
75
- ): Promise<EntityInstance<TTable>[]> {
76
- return executeWithContexts(session.getExecutionContext(), session, qb);
77
- }
78
-
79
- /**
80
- * Executes a hydrated query using execution and hydration contexts.
81
- * @template TTable - The table type
82
- * @param _execCtx - The execution context (unused)
83
- * @param hydCtx - The hydration context
88
+ export async function executeHydrated<TTable extends TableDef>(
89
+ session: OrmSession,
90
+ qb: SelectQueryBuilder<unknown, TTable>
91
+ ): Promise<EntityInstance<TTable>[]> {
92
+ return executeWithContexts(session.getExecutionContext(), session, qb);
93
+ }
94
+
95
+ /**
96
+ * Executes a hydrated query and returns plain row objects (no entity proxies).
97
+ * @template TTable - The table type
98
+ * @param session - The ORM session
99
+ * @param qb - The select query builder
100
+ * @returns Promise resolving to array of plain row objects
101
+ */
102
+ export async function executeHydratedPlain<TTable extends TableDef>(
103
+ session: OrmSession,
104
+ qb: SelectQueryBuilder<unknown, TTable>
105
+ ): Promise<Record<string, unknown>[]> {
106
+ return executePlainWithContexts(session.getExecutionContext(), qb);
107
+ }
108
+
109
+ /**
110
+ * Executes a hydrated query using execution and hydration contexts.
111
+ * @template TTable - The table type
112
+ * @param _execCtx - The execution context (unused)
113
+ * @param hydCtx - The hydration context
84
114
  * @param qb - The select query builder
85
115
  * @returns Promise resolving to array of entity instances
86
116
  */
@@ -96,6 +126,20 @@ export async function executeHydratedWithContexts<TTable extends TableDef>(
96
126
  return executeWithContexts(execCtx, entityCtx, qb);
97
127
  }
98
128
 
129
+ /**
130
+ * Executes a hydrated query using execution context and returns plain row objects.
131
+ * @template TTable - The table type
132
+ * @param execCtx - The execution context
133
+ * @param qb - The select query builder
134
+ * @returns Promise resolving to array of plain row objects
135
+ */
136
+ export async function executeHydratedPlainWithContexts<TTable extends TableDef>(
137
+ execCtx: ExecutionContext,
138
+ qb: SelectQueryBuilder<unknown, TTable>
139
+ ): Promise<Record<string, unknown>[]> {
140
+ return executePlainWithContexts(execCtx, qb);
141
+ }
142
+
99
143
  const loadLazyRelationsForTable = async <TTable extends TableDef>(
100
144
  ctx: EntityContext,
101
145
  table: TTable,
@@ -9,7 +9,6 @@ import { SelectPredicateFacet } from './predicate-facet.js';
9
9
  import { SelectRelationFacet } from './relation-facet.js';
10
10
  import { ORDER_DIRECTIONS, OrderDirection } from '../../core/sql/sql.js';
11
11
  import { OrmSession } from '../../orm/orm-session.js';
12
- import { EntityInstance } from '../../schema/types.js';
13
12
  import type { SelectQueryBuilder } from '../select.js';
14
13
 
15
14
  export type WhereHasOptions = {
@@ -85,7 +84,7 @@ export async function executePagedQuery<T, TTable extends TableDef>(
85
84
  session: OrmSession,
86
85
  options: { page: number; pageSize: number },
87
86
  countCallback: (session: OrmSession) => Promise<number>
88
- ): Promise<{ items: EntityInstance<TTable>[]; totalItems: number }> {
87
+ ): Promise<{ items: T[]; totalItems: number }> {
89
88
  const { page, pageSize } = options;
90
89
 
91
90
  if (!Number.isInteger(page) || page < 1) {