@tanstack/db
Version:
A reactive client store for building super fast apps on sync
570 lines (569 loc) • 27.5 kB
text/typescript
import { CollectionImpl } from '../../collection/index.js';
import { SingleResult, StringCollationConfig } from '../../types.js';
import { Aggregate, BasicExpression, Func, OrderByDirection, PropRef, Value } from '../ir.js';
import { QueryBuilder } from './index.js';
/**
* Context - The central state container for query builder operations
*
* This interface tracks all the information needed to build and type-check queries:
*
* **Schema Management**:
* - `baseSchema`: The original tables/collections from the `from()` clause
* - `schema`: Current available tables (expands with joins, contracts with subqueries)
*
* **Query State**:
* - `fromSourceName`: Which table was used in `from()` - needed for optionality logic
* - `hasJoins`: Whether any joins have been added (affects result type inference)
* - `joinTypes`: Maps table aliases to their join types for optionality calculations
*
* **Result Tracking**:
* - `result`: The final shape after `select()` - undefined until select is called
*
* The context evolves through the query builder chain:
* 1. `from()` sets baseSchema and schema to the same thing
* 2. `join()` expands schema and sets hasJoins/joinTypes
* 3. `select()` sets result to the projected shape
*/
export interface Context {
baseSchema: ContextSchema;
schema: ContextSchema;
fromSourceName: string;
hasJoins?: boolean;
joinTypes?: Record<string, `inner` | `left` | `right` | `full` | `outer` | `cross`>;
result?: any;
singleResult?: boolean;
}
/**
* ContextSchema - The shape of available tables/collections in a query context
*
* This is simply a record mapping table aliases to their TypeScript types.
* It evolves as the query progresses:
* - Initial: Just the `from()` table
* - After joins: Includes all joined tables with proper optionality
* - In subqueries: May be a subset of the outer query's schema
*/
export type ContextSchema = Record<string, unknown>;
/**
* Source - Input definition for query builder `from()` clause
*
* Maps table aliases to either:
* - `CollectionImpl`: A database collection/table
* - `QueryBuilder`: A subquery that can be used as a table
*
* Example: `{ users: usersCollection, orders: ordersCollection }`
*/
export type Source = {
[alias: string]: CollectionImpl<any, any> | QueryBuilder<Context>;
};
/**
* InferCollectionType - Extracts the TypeScript type from a CollectionImpl
*
* This helper ensures we get the same type that was used when creating the collection itself.
* This can be an explicit type passed by the user or the schema output type.
*/
export type InferCollectionType<T> = T extends CollectionImpl<infer TOutput, any, any, any, any> ? TOutput : never;
/**
* SchemaFromSource - Converts a Source definition into a ContextSchema
*
* This transforms the input to `from()` into the schema format used throughout
* the query builder. For each alias in the source:
* - Collections → their inferred TypeScript type
* - Subqueries → their result type (what they would return if executed)
*
* The `Prettify` wrapper ensures clean type display in IDEs.
*/
export type SchemaFromSource<T extends Source> = Prettify<{
[K in keyof T]: T[K] extends CollectionImpl<any, any, any, any, any> ? InferCollectionType<T[K]> : T[K] extends QueryBuilder<infer TContext> ? GetResult<TContext> : never;
}>;
/**
* GetAliases - Extracts all table aliases available in a query context
*
* Simple utility type that returns the keys of the schema, representing
* all table/source aliases that can be referenced in the current query.
*/
export type GetAliases<TContext extends Context> = keyof TContext[`schema`];
/**
* WhereCallback - Type for where/having clause callback functions
*
* These callbacks receive a `refs` object containing RefProxy instances for
* all available tables. The callback should return a boolean expression
* that will be used to filter query results.
*
* Example: `(refs) => eq(refs.users.age, 25)`
*/
export type WhereCallback<TContext extends Context> = (refs: RefsForContext<TContext>) => any;
/**
* SelectValue - Union of all valid values in a select clause
*
* This type defines what can be used as values in the object passed to `select()`.
*
* **Core Expression Types**:
* - `BasicExpression`: Function calls like `upper(users.name)`
* - `Aggregate`: Aggregations like `count()`, `avg()`
* - `RefProxy/Ref`: Direct field references like `users.name`
*
* **JavaScript Literals** (for constant values in projections):
* - `string`: String literals like `'active'`, `'N/A'`
* - `number`: Numeric literals like `0`, `42`, `3.14`
* - `boolean`: Boolean literals `true`, `false`
* - `null`: Explicit null values
*
* **Advanced Features**:
* - `undefined`: Allows optional projection values
* - `{ [key: string]: SelectValue }`: Nested object projection
*
* The clean Ref type ensures no internal properties are visible to users.
*
* Examples:
* ```typescript
* select({
* id: users.id,
* name: users.name,
* status: 'active', // string literal
* priority: 1, // number literal
* verified: true, // boolean literal
* notes: null, // explicit null
* profile: {
* name: users.name,
* email: users.email
* }
* })
* ```
*/
type SelectValue = BasicExpression | Aggregate | Ref | RefLeaf<any> | string | number | boolean | null | undefined | {
[key: string]: SelectValue;
} | Array<RefLeaf<any>>;
type SelectShape = {
[key: string]: SelectValue | SelectShape;
};
/**
* SelectObject - Wrapper type for select clause objects
*
* This ensures that objects passed to `select()` have valid SelectValue types
* for all their properties. It's a simple wrapper that provides better error
* messages when invalid selections are attempted.
*/
export type SelectObject<T extends SelectShape = SelectShape> = T;
/**
* ResultTypeFromSelect - Infers the result type from a select object
*
* This complex type transforms the input to `select()` into the actual TypeScript
* type that the query will return. It handles all the different kinds of values
* that can appear in a select clause:
*
* **Ref/RefProxy Extraction**:
* - `RefProxy<T>` → `T`: Extracts the underlying type
* - `Ref<T> | undefined` → `T | undefined`: Preserves optionality
* - `Ref<T> | null` → `T | null`: Preserves nullability
*
* **Expression Types**:
* - `BasicExpression<T>` → `T`: Function results like `upper()` → `string`
* - `Aggregate<T>` → `T`: Aggregation results like `count()` → `number`
*
* **JavaScript Literals** (pass through as-is):
* - `string` → `string`: String literals remain strings
* - `number` → `number`: Numeric literals remain numbers
* - `boolean` → `boolean`: Boolean literals remain booleans
* - `null` → `null`: Explicit null remains null
* - `undefined` → `undefined`: Direct undefined values
*
* **Nested Objects** (recursive):
* - Plain objects are recursively processed to handle nested projections
* - RefProxy objects are detected and their types extracted
*
* Example transformation:
* ```typescript
* // Input:
* { id: Ref<number>, name: Ref<string>, status: 'active', count: 42, profile: { bio: Ref<string> } }
*
* // Output:
* { id: number, name: string, status: 'active', count: 42, profile: { bio: string } }
* ```
*/
export type ResultTypeFromSelect<TSelectObject> = WithoutRefBrand<Prettify<{
[K in keyof TSelectObject]: NeedsExtraction<TSelectObject[K]> extends true ? ExtractExpressionType<TSelectObject[K]> : TSelectObject[K] extends Ref<infer _T> ? ExtractRef<TSelectObject[K]> : TSelectObject[K] extends RefLeaf<infer T> ? T : TSelectObject[K] extends RefLeaf<infer T> | undefined ? T | undefined : TSelectObject[K] extends RefLeaf<infer T> | null ? T | null : TSelectObject[K] extends Ref<infer _T> | undefined ? ExtractRef<TSelectObject[K]> | undefined : TSelectObject[K] extends Ref<infer _T> | null ? ExtractRef<TSelectObject[K]> | null : TSelectObject[K] extends Aggregate<infer T> ? T : TSelectObject[K] extends string | number | boolean | null | undefined ? TSelectObject[K] : TSelectObject[K] extends Record<string, any> ? ResultTypeFromSelect<TSelectObject[K]> : never;
}>>;
type ExtractRef<T> = Prettify<ResultTypeFromSelect<WithoutRefBrand<T>>>;
type ExtractExpressionType<T> = T extends PropRef<infer U> ? U : T extends Value<infer U> ? U : T extends Func<infer U> ? U : T extends Aggregate<infer U> ? U : T extends BasicExpression<infer U> ? U : T;
type NeedsExtraction<T> = T extends PropRef<any> | Value<any> | Func<any> | Aggregate<any> | BasicExpression<any> ? true : false;
/**
* OrderByCallback - Type for orderBy clause callback functions
*
* Similar to WhereCallback, these receive refs for all available tables
* and should return expressions that will be used for sorting.
*
* Example: `(refs) => refs.users.createdAt`
*/
export type OrderByCallback<TContext extends Context> = (refs: RefsForContext<TContext>) => any;
/**
* OrderByOptions - Configuration for orderBy operations
*
* Combines direction and null handling with string-specific sorting options.
* The intersection with StringSortOpts allows for either simple lexical sorting
* or locale-aware sorting with customizable options.
*/
export type OrderByOptions = {
direction?: OrderByDirection;
nulls?: `first` | `last`;
} & StringCollationConfig;
/**
* CompareOptions - Final resolved options for comparison operations
*
* This is the internal type used after all orderBy options have been resolved
* to their concrete values. Unlike OrderByOptions, all fields are required
* since defaults have been applied.
*/
export type CompareOptions = StringCollationConfig & {
direction: OrderByDirection;
nulls: `first` | `last`;
};
/**
* GroupByCallback - Type for groupBy clause callback functions
*
* These callbacks receive refs for all available tables and should return
* expressions that will be used for grouping query results.
*
* Example: `(refs) => refs.orders.status`
*/
export type GroupByCallback<TContext extends Context> = (refs: RefsForContext<TContext>) => any;
/**
* JoinOnCallback - Type for join condition callback functions
*
* These callbacks receive refs for all available tables (including the newly
* joined table) and should return a boolean expression defining the join condition.
*
* Important: The newly joined table is NOT marked as optional in this callback,
* even for left/right/full joins, because optionality is applied AFTER the join
* condition is evaluated.
*
* Example: `(refs) => eq(refs.users.id, refs.orders.userId)`
*/
export type JoinOnCallback<TContext extends Context> = (refs: RefsForContext<TContext>) => any;
/**
* RefProxyForContext - Creates ref proxies for all tables/collections in a query context
*
* This is the main entry point for creating ref objects in query builder callbacks.
* It handles optionality by placing undefined/null OUTSIDE the RefProxy to enable
* JavaScript's optional chaining operator (?.):
*
* Examples:
* - Required field: `RefProxy<User>` → user.name works
* - Optional field: `RefProxy<User> | undefined` → user?.name works
* - Nullable field: `RefProxy<User> | null` → user?.name works
* - Both optional and nullable: `RefProxy<User> | undefined` → user?.name works
*
* The key insight is that `RefProxy<User | undefined>` would NOT allow `user?.name`
* because the undefined is "inside" the proxy, but `RefProxy<User> | undefined`
* does allow it because the undefined is "outside" the proxy.
*
* The logic prioritizes optional chaining by always placing `undefined` outside when
* a type is both optional and nullable (e.g., `string | null | undefined`).
*/
export type RefsForContext<TContext extends Context> = {
[K in keyof TContext[`schema`]]: IsNonExactOptional<TContext[`schema`][K]> extends true ? IsNonExactNullable<TContext[`schema`][K]> extends true ? // T is both non-exact optional and non-exact nullable (e.g., string | null | undefined)
Ref<NonNullable<TContext[`schema`][K]>> | undefined : // T is optional (T | undefined) but not exactly undefined, and not nullable
Ref<NonUndefined<TContext[`schema`][K]>> | undefined : IsNonExactNullable<TContext[`schema`][K]> extends true ? // T is nullable (T | null) but not exactly null, and not optional
Ref<NonNull<TContext[`schema`][K]>> | null : Ref<TContext[`schema`][K]>;
};
/**
* Type Detection Helpers
*
* These helpers distinguish between different kinds of optionality/nullability:
* - IsExactlyUndefined: T is literally `undefined` (not `string | undefined`)
* - IsOptional: T includes undefined (like `string | undefined`)
* - IsExactlyNull: T is literally `null` (not `string | null`)
* - IsNullable: T includes null (like `string | null`)
* - IsNonExactOptional: T includes undefined but is not exactly undefined
* - IsNonExactNullable: T includes null but is not exactly null
*
* The [T] extends [undefined] pattern prevents distributive conditional types,
* ensuring we check the exact type rather than distributing over union members.
*/
type IsExactlyUndefined<T> = [T] extends [undefined] ? true : false;
type IsExactlyNull<T> = [T] extends [null] ? true : false;
type IsOptional<T> = undefined extends T ? true : false;
type IsNullable<T> = null extends T ? true : false;
type IsNonExactOptional<T> = IsOptional<T> extends true ? IsExactlyUndefined<T> extends false ? true : false : false;
type IsNonExactNullable<T> = IsNullable<T> extends true ? IsExactlyNull<T> extends false ? true : false : false;
/**
* Type Extraction Helpers
*
* These helpers extract the "useful" part of a type by removing null/undefined:
* - NonUndefined: `string | undefined` → `string` (preserves null if present)
* - NonNull: `string | null` → `string` (preserves undefined if present)
*
* These are used when we need to handle optional and nullable types separately.
* For cases where both null and undefined should be removed, use TypeScript's
* built-in NonNullable<T> instead.
*/
type NonUndefined<T> = T extends undefined ? never : T;
type NonNull<T> = T extends null ? never : T;
/**
* Ref - The user-facing ref interface for the query builder
*
* This is a clean type that represents a reference to a value in the query,
* designed for optimal IDE experience without internal implementation details.
* It provides a recursive interface that allows nested property access while
* preserving optionality and nullability correctly.
*
* When spread in select clauses, it correctly produces the underlying data type
* without Ref wrappers, enabling clean spread operations.
*
* Example usage:
* ```typescript
* // Clean interface - no internal properties visible
* const users: Ref<{ id: number; profile?: { bio: string } }> = { ... }
* users.id // Ref<number> - clean display
* users.profile?.bio // Ref<string> - nested optional access works
*
* // Spread operations work cleanly:
* select(({ user }) => ({ ...user })) // Returns User type, not Ref types
* ```
*/
export type Ref<T = any> = {
[K in keyof T]: IsNonExactOptional<T[K]> extends true ? IsNonExactNullable<T[K]> extends true ? IsPlainObject<NonNullable<T[K]>> extends true ? Ref<NonNullable<T[K]>> | undefined : RefLeaf<NonNullable<T[K]>> | undefined : IsPlainObject<NonUndefined<T[K]>> extends true ? Ref<NonUndefined<T[K]>> | undefined : RefLeaf<NonUndefined<T[K]>> | undefined : IsNonExactNullable<T[K]> extends true ? IsPlainObject<NonNull<T[K]>> extends true ? Ref<NonNull<T[K]>> | null : RefLeaf<NonNull<T[K]>> | null : IsPlainObject<T[K]> extends true ? Ref<T[K]> : RefLeaf<T[K]>;
} & RefLeaf<T>;
/**
* Ref - The user-facing ref type with clean IDE display
*
* An opaque branded type that represents a reference to a value in a query.
* This shows as `Ref<T>` in the IDE without exposing internal structure.
*
* Example usage:
* - Ref<number> displays as `Ref<number>` in IDE
* - Ref<string> displays as `Ref<string>` in IDE
* - No internal properties like __refProxy, __path, __type are visible
*/
declare const RefBrand: unique symbol;
export type RefLeaf<T = any> = {
readonly [RefBrand]?: T;
};
type WithoutRefBrand<T> = T extends Record<string, any> ? Omit<T, typeof RefBrand> : T;
/**
* PreserveSingleResultFlag - Conditionally includes the singleResult flag
*
* This helper type ensures the singleResult flag is only added to the context when it's
* explicitly true. It uses a non-distributive conditional (tuple wrapper) to prevent
* unexpected behavior when TFlag is a union type.
*
* @template TFlag - The singleResult flag value to check
* @returns { singleResult: true } if TFlag is true, otherwise {}
*/
type PreserveSingleResultFlag<TFlag> = [TFlag] extends [true] ? {
singleResult: true;
} : {};
/**
* MergeContextWithJoinType - Creates a new context after a join operation
*
* This is the core type that handles the complex logic of merging schemas
* when tables are joined, applying the correct optionality based on join type.
*
* **Key Responsibilities**:
* 1. **Schema Merging**: Combines existing schema with newly joined tables
* 2. **Optionality Logic**: Applies join-specific optionality rules:
* - `LEFT JOIN`: New table becomes optional
* - `RIGHT JOIN`: Existing tables become optional
* - `FULL JOIN`: Both existing and new become optional
* - `INNER JOIN`: No tables become optional
* 3. **State Tracking**: Updates hasJoins and joinTypes for future operations
*
* **Context Evolution**:
* - `baseSchema`: Unchanged (always the original `from()` tables)
* - `schema`: Expanded with new tables and proper optionality
* - `hasJoins`: Set to true
* - `joinTypes`: Updated to track this join type
* - `result`: Preserved from previous operations
* - `singleResult`: Preserved only if already true (via PreserveSingleResultFlag)
*/
export type MergeContextWithJoinType<TContext extends Context, TNewSchema extends ContextSchema, TJoinType extends `inner` | `left` | `right` | `full` | `outer` | `cross`> = {
baseSchema: TContext[`baseSchema`];
schema: ApplyJoinOptionalityToMergedSchema<TContext[`schema`], TNewSchema, TJoinType, TContext[`fromSourceName`]>;
fromSourceName: TContext[`fromSourceName`];
hasJoins: true;
joinTypes: (TContext[`joinTypes`] extends Record<string, any> ? TContext[`joinTypes`] : {}) & {
[K in keyof TNewSchema & string]: TJoinType;
};
result: TContext[`result`];
} & PreserveSingleResultFlag<TContext[`singleResult`]>;
/**
* ApplyJoinOptionalityToMergedSchema - Applies optionality rules when merging schemas
*
* This type implements the SQL join optionality semantics:
*
* **For Existing Tables**:
* - `RIGHT JOIN` or `FULL JOIN`: Main table (from fromSourceName) becomes optional
* - Other join types: Existing tables keep their current optionality
* - Previously joined tables: Keep their already-applied optionality
*
* **For New Tables**:
* - `LEFT JOIN` or `FULL JOIN`: New table becomes optional
* - `INNER JOIN` or `RIGHT JOIN`: New table remains required
*
* **Examples**:
* ```sql
* FROM users LEFT JOIN orders -- orders becomes optional
* FROM users RIGHT JOIN orders -- users becomes optional
* FROM users FULL JOIN orders -- both become optional
* FROM users INNER JOIN orders -- both remain required
* ```
*
* The intersection (&) ensures both existing and new schemas are merged
* into a single type while preserving all table references.
*/
export type ApplyJoinOptionalityToMergedSchema<TExistingSchema extends ContextSchema, TNewSchema extends ContextSchema, TJoinType extends `inner` | `left` | `right` | `full` | `outer` | `cross`, TFromSourceName extends string> = {
[K in keyof TExistingSchema]: K extends TFromSourceName ? TJoinType extends `right` | `full` ? TExistingSchema[K] | undefined : TExistingSchema[K] : TExistingSchema[K];
} & {
[K in keyof TNewSchema]: TJoinType extends `left` | `full` ? // New table becomes optional for left and full joins
TNewSchema[K] | undefined : TNewSchema[K];
};
/**
* Utility type to infer the query result size (single row or an array)
*/
export type InferResultType<TContext extends Context> = TContext extends SingleResult ? GetResult<TContext> | undefined : Array<GetResult<TContext>>;
/**
* GetResult - Determines the final result type of a query
*
* This type implements the logic for what a query returns based on its current state:
*
* **Priority Order**:
* 1. **Explicit Result**: If `select()` was called, use the projected type
* 2. **Join Query**: If joins exist, return all tables with proper optionality
* 3. **Single Table**: Return just the main table from `from()`
*
* **Examples**:
* ```typescript
* // Single table query:
* from({ users }).where(...) // → User[]
*
* // Join query without select:
* from({ users }).leftJoin({ orders }, ...) // → { users: User, orders: Order | undefined }[]
*
* // Query with select:
* from({ users }).select({ id: users.id, name: users.name }) // → { id: number, name: string }[]
* ```
*
* The `Prettify` wrapper ensures clean type display in IDEs by flattening
* complex intersection types into readable object types.
*/
export type GetResult<TContext extends Context> = Prettify<TContext[`result`] extends object ? TContext[`result`] : TContext[`hasJoins`] extends true ? TContext[`schema`] : TContext[`schema`][TContext[`fromSourceName`]]>;
/**
* ApplyJoinOptionalityToSchema - Legacy helper for complex join scenarios
*
* This type was designed to handle complex scenarios with multiple joins
* where the optionality of tables might be affected by subsequent joins.
* Currently used in advanced join logic, but most cases are handled by
* the simpler `ApplyJoinOptionalityToMergedSchema`.
*
* **Logic**:
* 1. **Main Table**: Becomes optional if ANY right or full join exists in the chain
* 2. **Joined Tables**: Check their specific join type for optionality
* 3. **Complex Cases**: Handle scenarios where subsequent joins affect earlier tables
*
* This is primarily used for edge cases and may be simplified in future versions
* as the simpler merge-based approach covers most real-world scenarios.
*/
export type ApplyJoinOptionalityToSchema<TSchema extends ContextSchema, TJoinTypes extends Record<string, string>, TFromSourceName extends string> = {
[K in keyof TSchema]: K extends TFromSourceName ? HasJoinType<TJoinTypes, `right` | `full`> extends true ? TSchema[K] | undefined : TSchema[K] : K extends keyof TJoinTypes ? TJoinTypes[K] extends `left` | `full` ? TSchema[K] | undefined : IsTableMadeOptionalBySubsequentJoins<K, TJoinTypes, TFromSourceName> extends true ? TSchema[K] | undefined : TSchema[K] : TSchema[K];
};
/**
* IsTableMadeOptionalBySubsequentJoins - Checks if later joins affect table optionality
*
* This helper determines if a table that was initially required becomes optional
* due to joins that happen later in the query chain.
*
* **Current Implementation**:
* - Main table: Becomes optional if any right/full joins exist
* - Joined tables: Not affected by subsequent joins (simplified model)
*
* This is a conservative approach that may be extended in the future to handle
* more complex join interaction scenarios.
*/
type IsTableMadeOptionalBySubsequentJoins<TTableAlias extends string | number | symbol, TJoinTypes extends Record<string, string>, TFromSourceName extends string> = TTableAlias extends TFromSourceName ? HasJoinType<TJoinTypes, `right` | `full`> : false;
/**
* HasJoinType - Utility to check if any join in a chain matches target types
*
* This type searches through all recorded join types to see if any match
* the specified target types. It's used to implement logic like "becomes optional
* if ANY right or full join exists in the chain".
*
* **How it works**:
* 1. Maps over all join types, checking each against target types
* 2. Creates a union of boolean results
* 3. Uses `true extends Union` pattern to check if any were true
*
* **Example**:
* ```typescript
* HasJoinType<{ orders: 'left', products: 'right' }, 'right' | 'full'>
* // → true (because products is a right join)
* ```
*/
export type HasJoinType<TJoinTypes extends Record<string, string>, TTargetTypes extends string> = true extends {
[K in keyof TJoinTypes]: TJoinTypes[K] extends TTargetTypes ? true : false;
}[keyof TJoinTypes] ? true : false;
/**
* MergeContextForJoinCallback - Special context for join condition callbacks
*
* This type creates a context specifically for the `onCallback` parameter of join operations.
* The key difference from `MergeContextWithJoinType` is that NO optionality is applied here.
*
* **Why No Optionality?**
* In SQL, join conditions are evaluated BEFORE optionality is determined. Both tables
* must be treated as available (non-optional) within the join condition itself.
* Optionality is only applied to the result AFTER the join logic executes.
*
* **Example**:
* ```typescript
* .from({ users })
* .leftJoin({ orders }, ({ users, orders }) => {
* // users is NOT optional here - we can access users.id directly
* // orders is NOT optional here - we can access orders.userId directly
* return eq(users.id, orders.userId)
* })
* .where(({ orders }) => {
* // NOW orders is optional because it's after the LEFT JOIN
* return orders?.status === 'pending'
* })
* ```
*
* The simple intersection (&) merges schemas without any optionality transformation.
*/
export type MergeContextForJoinCallback<TContext extends Context, TNewSchema extends ContextSchema> = {
baseSchema: TContext[`baseSchema`];
schema: TContext[`schema`] & TNewSchema;
fromSourceName: TContext[`fromSourceName`];
hasJoins: true;
joinTypes: TContext[`joinTypes`] extends Record<string, any> ? TContext[`joinTypes`] : {};
result: TContext[`result`];
};
/**
* WithResult - Updates a context with a new result type after select()
*
* This utility type is used internally when the `select()` method is called
* to update the context with the projected result type. It preserves all
* other context properties while replacing the `result` field.
*
* **Usage**:
* When `select()` is called, the query builder uses this type to create
* a new context where `result` contains the shape of the selected fields.
*
* The double `Prettify` ensures both the overall context and the nested
* result type display cleanly in IDEs.
*/
export type WithResult<TContext extends Context, TResult> = Prettify<Omit<TContext, `result`> & {
result: Prettify<TResult>;
}>;
/**
* Prettify - Utility type for clean IDE display
*/
export type Prettify<T> = {
[K in keyof T]: T[K];
} & {};
/**
* IsPlainObject - Utility type to check if T is a plain object
*/
type IsPlainObject<T> = T extends unknown ? T extends object ? T extends ReadonlyArray<any> ? false : T extends JsBuiltIns ? false : true : false : false;
/**
* JsBuiltIns - List of JavaScript built-ins
*/
type JsBuiltIns = ArrayBuffer | ArrayBufferLike | AsyncGenerator<any, any, any> | BigInt64Array | BigUint64Array | DataView | Date | Error | Float32Array | Float64Array | Function | Generator<any, any, any> | Int16Array | Int32Array | Int8Array | Map<any, any> | Promise<any> | RegExp | Set<any> | string | Uint16Array | Uint32Array | Uint8Array | Uint8ClampedArray | WeakMap<any, any> | WeakSet<any>;
export {};