Skip to content

Commit

Permalink
fix(types): MergeSchemaPath infer inputs not only params (honojs#2154)
Browse files Browse the repository at this point in the history
* fix(types): `MergeSchemaPath` infer inputs not only params

* denoify

* format
  • Loading branch information
yusukebe authored Feb 5, 2024
1 parent 97dd369 commit 09845e5
Show file tree
Hide file tree
Showing 3 changed files with 80 additions and 10 deletions.
12 changes: 9 additions & 3 deletions deno_dist/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1627,7 +1627,9 @@ type ExtractParams<Path extends string> = string extends Path
? { [K in Param | keyof ExtractParams<`/${Rest}`>]: string }
: Path extends `${infer Start}:${infer Param}`
? { [K in Param]: string }
: {}
: never

type FlattenIfIntersect<T> = T extends infer O ? { [K in keyof O]: O[K] } : never

export type MergeSchemaPath<OrigSchema extends Schema, SubPath extends string> = {
[P in keyof OrigSchema as MergePath<SubPath, P & string>]: {
Expand All @@ -1636,8 +1638,10 @@ export type MergeSchemaPath<OrigSchema extends Schema, SubPath extends string> =
output: infer Output
}
? {
input: Input extends { param: infer Params }
? { param: Params & ExtractParams<SubPath> }
input: Input extends { param: infer _ }
? ExtractParams<SubPath> extends never
? Input
: FlattenIfIntersect<Input & { param: ExtractParams<SubPath> }>
: RemoveBlankRecord<ExtractParams<SubPath>> extends never
? Input
: Input & { param: ExtractParams<SubPath> }
Expand All @@ -1648,6 +1652,8 @@ export type MergeSchemaPath<OrigSchema extends Schema, SubPath extends string> =
}

export type AddParam<I, P extends string> = ParamKeys<P> extends never
? I
: I extends { param: infer _ }
? I
: I & { param: UnionToIntersection<ParamKeyToRecord<ParamKeys<P>>> }

Expand Down
66 changes: 62 additions & 4 deletions src/types.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ import { createMiddleware } from './helper'
import { Hono } from './hono'
import { poweredBy } from './middleware/powered-by'
import type {
AddParam,
Env,
ExtractSchema,
Handler,
Expand Down Expand Up @@ -475,8 +476,57 @@ describe('For HonoRequest', () => {
})
})

describe('merge path', () => {
test('MergePath', () => {
describe('AddParam', () => {
it('Should add params to input correctly', () => {
type Actual = AddParam<
{
param: {
id: string
}
} & {
query: {
page: string
}
},
'/:id'
>
type Expected = {
query: {
page: string
}
} & {
param: {
id: string
}
}
type verify = Expect<Equal<Expected, Actual>>
})
})

describe('ToSchema', () => {
it('Should convert parameters to schema correctly', () => {
type Actual = ToSchema<'get', '/:id', { param: { id: string }; query: { page: string } }, {}>
type Expected = {
'/:id': {
$get: {
input: {
param: {
id: string
}
query: {
page: string
}
}
output: {}
}
}
}
type verify = Expect<Equal<Expected, Actual>>
})
})

describe('MergePath', () => {
it('Should merge paths correctly', () => {
type path1 = MergePath<'/api', '/book'>
type verify1 = Expect<Equal<'/api/book', path1>>
type path2 = MergePath<'/api/', '/book'>
Expand All @@ -486,8 +536,10 @@ describe('merge path', () => {
type path4 = MergePath<'/api', '/'>
type verify4 = Expect<Equal<'/api', path4>>
})
})

test('MergeSchemaPath', () => {
describe('MergeSchemaPath', () => {
it('Should merge schema and sub path correctly', () => {
type Sub = ToSchema<
'post',
'/posts',
Expand Down Expand Up @@ -537,7 +589,7 @@ describe('merge path', () => {
type verify = Expect<Equal<Expected, Actual>>
})

test('MergeSchemePath - with params and the subpath does not have params', () => {
it('Should merge schema which has params and sub path does not have params', () => {
type Actual = MergeSchemaPath<
{
'/': {
Expand All @@ -546,6 +598,9 @@ describe('merge path', () => {
param: {
id: string
}
query: {
page: string
}
}
output: {}
}
Expand All @@ -560,6 +615,9 @@ describe('merge path', () => {
param: {
id: string
}
query: {
page: string
}
}
output: {}
}
Expand Down
12 changes: 9 additions & 3 deletions src/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -1627,7 +1627,9 @@ type ExtractParams<Path extends string> = string extends Path
? { [K in Param | keyof ExtractParams<`/${Rest}`>]: string }
: Path extends `${infer Start}:${infer Param}`
? { [K in Param]: string }
: {}
: never

type FlattenIfIntersect<T> = T extends infer O ? { [K in keyof O]: O[K] } : never

export type MergeSchemaPath<OrigSchema extends Schema, SubPath extends string> = {
[P in keyof OrigSchema as MergePath<SubPath, P & string>]: {
Expand All @@ -1636,8 +1638,10 @@ export type MergeSchemaPath<OrigSchema extends Schema, SubPath extends string> =
output: infer Output
}
? {
input: Input extends { param: infer Params }
? { param: Params & ExtractParams<SubPath> }
input: Input extends { param: infer _ }
? ExtractParams<SubPath> extends never
? Input
: FlattenIfIntersect<Input & { param: ExtractParams<SubPath> }>
: RemoveBlankRecord<ExtractParams<SubPath>> extends never
? Input
: Input & { param: ExtractParams<SubPath> }
Expand All @@ -1648,6 +1652,8 @@ export type MergeSchemaPath<OrigSchema extends Schema, SubPath extends string> =
}

export type AddParam<I, P extends string> = ParamKeys<P> extends never
? I
: I extends { param: infer _ }
? I
: I & { param: UnionToIntersection<ParamKeyToRecord<ParamKeys<P>>> }

Expand Down

0 comments on commit 09845e5

Please sign in to comment.