Skip to content

Commit daa8705

Browse files
committed
feat(types): allow jsonpath selector types overrides
1 parent 8bdb969 commit daa8705

File tree

5 files changed

+139
-7
lines changed

5 files changed

+139
-7
lines changed

src/select-query-parser/result.ts

+28-2
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,8 @@ import {
1515
GetFieldNodeResultName,
1616
IsAny,
1717
IsRelationNullable,
18+
IsStringUnion,
19+
JsonPathToType,
1820
ResolveRelationship,
1921
SelectQueryError,
2022
} from './utils'
@@ -239,6 +241,30 @@ type ProcessFieldNode<
239241
? ProcessEmbeddedResource<Schema, Relationships, Field, RelationName>
240242
: ProcessSimpleField<Row, RelationName, Field>
241243

244+
type ResolveJsonPathType<
245+
Value,
246+
Path extends string | undefined,
247+
CastType extends PostgreSQLTypes
248+
> = Path extends string
249+
? JsonPathToType<Value, Path> extends never
250+
? // Always fallback if JsonPathToType returns never
251+
TypeScriptTypes<CastType>
252+
: JsonPathToType<Value, Path> extends infer PathResult
253+
? PathResult extends string
254+
? // Use the result if it's a string as we know that even with the string accessor ->> it's a valid type
255+
PathResult
256+
: IsStringUnion<PathResult> extends true
257+
? // Use the result if it's a union of strings
258+
PathResult
259+
: CastType extends 'json'
260+
? // If the type is not a string, ensure it was accessed with json accessor ->
261+
PathResult
262+
: // Otherwise it means non-string value accessed with string accessor ->> use the TypeScriptTypes result
263+
TypeScriptTypes<CastType>
264+
: TypeScriptTypes<CastType>
265+
: // No json path, use regular type casting
266+
TypeScriptTypes<CastType>
267+
242268
/**
243269
* Processes a simple field (without embedded resources).
244270
*
@@ -261,8 +287,8 @@ type ProcessSimpleField<
261287
}
262288
: {
263289
// Aliases override the property name in the result
264-
[K in GetFieldNodeResultName<Field>]: Field['castType'] extends PostgreSQLTypes // We apply the detected casted as the result type
265-
? TypeScriptTypes<Field['castType']>
290+
[K in GetFieldNodeResultName<Field>]: Field['castType'] extends PostgreSQLTypes
291+
? ResolveJsonPathType<Row[Field['name']], Field['jsonPath'], Field['castType']>
266292
: Row[Field['name']]
267293
}
268294
: SelectQueryError<`column '${Field['name']}' does not exist on '${RelationName}'.`>

src/select-query-parser/utils.ts

+27-3
Original file line numberDiff line numberDiff line change
@@ -546,9 +546,33 @@ export type FindFieldMatchingRelationships<
546546
: SelectQueryError<'Failed to find matching relation via name'>
547547

548548
export type JsonPathToAccessor<Path extends string> = Path extends `${infer P1}->${infer P2}`
549-
? P2 extends `>${infer Rest}` // Check if P2 starts with > (from ->>)
549+
? P2 extends `>${infer Rest}` // Handle ->> operator
550550
? JsonPathToAccessor<`${P1}.${Rest}`>
551-
: JsonPathToAccessor<`${P1}.${P2}`>
551+
: P2 extends string // Handle -> operator
552+
? JsonPathToAccessor<`${P1}.${P2}`>
553+
: Path
554+
: Path extends `>${infer Rest}` // Clean up any remaining > characters
555+
? JsonPathToAccessor<Rest>
552556
: Path extends `${infer P1}::${infer _}` // Handle type casting
553-
? JsonPathToAccessor<P1> // Process the path without the cast type
557+
? JsonPathToAccessor<P1>
554558
: Path
559+
560+
export type JsonPathToType<T, Path extends string> = Path extends ''
561+
? T
562+
: ContainsNull<T> extends true
563+
? JsonPathToType<Exclude<T, null>, Path>
564+
: Path extends `${infer Key}.${infer Rest}`
565+
? Key extends keyof T
566+
? JsonPathToType<T[Key], Rest>
567+
: never
568+
: Path extends keyof T
569+
? T[Path]
570+
: never
571+
572+
export type IsStringUnion<T> = string extends T
573+
? false
574+
: T extends string
575+
? [T] extends [never]
576+
? false
577+
: true
578+
: false

test/index.test-d.ts

+28-2
Original file line numberDiff line numberDiff line change
@@ -214,11 +214,37 @@ const postgrest = new PostgrestClient<Database>(REST_URL)
214214

215215
// Json Accessor with custom types overrides
216216
{
217-
const { error } = await postgrest
217+
const { data, error } = await postgrest
218218
.schema('personal')
219219
.from('users')
220-
.select('data->foo->bar, data->foo->>baz')
220+
.select('data->bar->baz, data->en, data->bar')
221+
if (error) {
222+
throw new Error(error.message)
223+
}
224+
expectType<
225+
{
226+
baz: number
227+
en: 'ONE' | 'TWO' | 'THREE'
228+
bar: {
229+
baz: number
230+
}
231+
}[]
232+
>(data)
233+
}
234+
// Json string Accessor with custom types overrides
235+
{
236+
const { data, error } = await postgrest
237+
.schema('personal')
238+
.from('users')
239+
.select('data->bar->>baz, data->>en, data->>bar')
221240
if (error) {
222241
throw new Error(error.message)
223242
}
243+
expectType<
244+
{
245+
baz: string
246+
en: 'ONE' | 'TWO' | 'THREE'
247+
bar: string
248+
}[]
249+
>(data)
224250
}

test/select-query-parser/parser.test-d.ts

+25
Original file line numberDiff line numberDiff line change
@@ -103,6 +103,31 @@ import { selectParams } from '../relationships'
103103
},
104104
])
105105
}
106+
{
107+
expectType<ParseQuery<'data->preferences->>theme, data->>some, data->foo->bar->>biz'>>([
108+
{
109+
type: 'field',
110+
name: 'data',
111+
alias: 'theme',
112+
castType: 'text',
113+
jsonPath: 'preferences.theme',
114+
},
115+
{
116+
type: 'field',
117+
name: 'data',
118+
alias: 'some',
119+
castType: 'text',
120+
jsonPath: 'some',
121+
},
122+
{
123+
type: 'field',
124+
name: 'data',
125+
alias: 'biz',
126+
castType: 'text',
127+
jsonPath: 'foo.bar.biz',
128+
},
129+
])
130+
}
106131
// Select with spread
107132
{
108133
expectType<ParseQuery<'username, ...posts(id, title)'>>([

test/select-query-parser/result.test-d.ts

+31
Original file line numberDiff line numberDiff line change
@@ -118,3 +118,34 @@ type SelectQueryFromTableResult<
118118
expectType<typeof result1>(result2!)
119119
expectType<typeof result2>(result3!)
120120
}
121+
122+
{
123+
type SelectQueryFromTableResult<
124+
TableName extends keyof Database['personal']['Tables'],
125+
Q extends string
126+
> = GetResult<
127+
Database['personal'],
128+
Database['personal']['Tables'][TableName]['Row'],
129+
TableName,
130+
Database['personal']['Tables'][TableName]['Relationships'],
131+
Q
132+
>
133+
134+
let result: SelectQueryFromTableResult<'users', `data->bar->baz, data->en, data->bar`>
135+
let expected: {
136+
baz: number
137+
en: 'ONE' | 'TWO' | 'THREE'
138+
bar: {
139+
baz: number
140+
}
141+
}
142+
expectType<TypeEqual<typeof result, typeof expected>>(true)
143+
144+
let result2: SelectQueryFromTableResult<'users', `data->bar->>baz, data->>en, data->>bar`>
145+
let expected2: {
146+
baz: string
147+
en: 'ONE' | 'TWO' | 'THREE'
148+
bar: string
149+
}
150+
expectType<TypeEqual<typeof result2, typeof expected2>>(true)
151+
}

0 commit comments

Comments
 (0)