mirror of
https://github.com/zoriya/typebox.git
synced 2026-06-01 10:35:14 +00:00
Enhanced Indexed Access Types (#413)
This commit is contained in:
@@ -62,3 +62,164 @@ import { Type, Static } from '@sinclair/typebox'
|
||||
|
||||
Expect(R).ToInfer<never>()
|
||||
}
|
||||
// ------------------------------------------------------------------
|
||||
// Intersections
|
||||
// ------------------------------------------------------------------
|
||||
{
|
||||
type A = { x: string; y: 1 }
|
||||
type B = { x: string; y: number }
|
||||
type C = A & B
|
||||
type R = C['y']
|
||||
|
||||
const A = Type.Object({ x: Type.String(), y: Type.Literal(1) })
|
||||
const B = Type.Object({ x: Type.String(), y: Type.Number() })
|
||||
const C = Type.Intersect([A, B])
|
||||
const R = Type.Index(C, ['y'])
|
||||
Expect(R).ToBe<1>()
|
||||
}
|
||||
{
|
||||
type A = { x: string; y: 1 }
|
||||
type B = { x: string; y: number }
|
||||
type C = A & B
|
||||
type R = C['x']
|
||||
|
||||
const A = Type.Object({ x: Type.String(), y: Type.Literal(1) })
|
||||
const B = Type.Object({ x: Type.String(), y: Type.Number() })
|
||||
const C = Type.Intersect([A, B])
|
||||
const R = Type.Index(C, ['x'])
|
||||
Expect(R).ToBe<string>()
|
||||
}
|
||||
{
|
||||
type A = { x: string; y: 1 }
|
||||
type B = { x: string; y: number }
|
||||
type C = A & B
|
||||
type R = C['x' | 'y']
|
||||
|
||||
const A = Type.Object({ x: Type.String(), y: Type.Literal(1) })
|
||||
const B = Type.Object({ x: Type.String(), y: Type.Number() })
|
||||
const C = Type.Intersect([A, B])
|
||||
const R = Type.Index(C, ['x', 'y'])
|
||||
Expect(R).ToBe<string | 1>()
|
||||
}
|
||||
{
|
||||
type A = { x: string; y: number }
|
||||
type B = { x: number; y: number }
|
||||
type C = A & B
|
||||
type R = C['x']
|
||||
|
||||
const A = Type.Object({ x: Type.String(), y: Type.Number() })
|
||||
const B = Type.Object({ x: Type.Number(), y: Type.Number() })
|
||||
const C = Type.Intersect([A, B])
|
||||
const R = Type.Index(C, ['x'])
|
||||
Expect(R).ToBe<never>()
|
||||
}
|
||||
{
|
||||
type A = { x: string; y: number }
|
||||
type B = { x: number; y: number }
|
||||
type C = A & B
|
||||
type R = C['y']
|
||||
|
||||
const A = Type.Object({ x: Type.String(), y: Type.Number() })
|
||||
const B = Type.Object({ x: Type.Number(), y: Type.Number() })
|
||||
const C = Type.Intersect([A, B])
|
||||
const R = Type.Index(C, ['y'])
|
||||
Expect(R).ToBe<number>()
|
||||
}
|
||||
{
|
||||
type A = { x: string; y: number }
|
||||
type B = { x: number; y: number }
|
||||
type C = A & B
|
||||
type R = C['x' | 'y']
|
||||
|
||||
const A = Type.Object({ x: Type.String(), y: Type.Number() })
|
||||
const B = Type.Object({ x: Type.Number(), y: Type.Number() })
|
||||
const C = Type.Intersect([A, B])
|
||||
const R = Type.Index(C, ['x', 'y'])
|
||||
Expect(R).ToBe<number>()
|
||||
}
|
||||
{
|
||||
type A = { x: string; y: 1 }
|
||||
type B = { x: string; y: number }
|
||||
type C = { x: string; y: number }
|
||||
type D = { x: string }
|
||||
type I = (A & B) & (C & D)
|
||||
type R = I['x' | 'y']
|
||||
|
||||
const A = Type.Object({ x: Type.String(), y: Type.Literal(1) })
|
||||
const B = Type.Object({ x: Type.String(), y: Type.Number() })
|
||||
const C = Type.Object({ x: Type.String(), y: Type.Number() })
|
||||
const D = Type.Object({ x: Type.String() })
|
||||
const I = Type.Intersect([Type.Intersect([A, B]), Type.Intersect([C, D])])
|
||||
const R = Type.Index(I, ['x', 'y'])
|
||||
type O = Static<typeof R>
|
||||
Expect(R).ToBe<string | 1>()
|
||||
}
|
||||
{
|
||||
type A = { x: string; y: 1 }
|
||||
type B = { x: number; y: number }
|
||||
type C = { x: string; y: number }
|
||||
type D = { x: string }
|
||||
type I = (A & B) & (C & D)
|
||||
type R = I['x' | 'y']
|
||||
|
||||
const A = Type.Object({ x: Type.String(), y: Type.Literal(1) })
|
||||
const B = Type.Object({ x: Type.Number(), y: Type.Number() })
|
||||
const C = Type.Object({ x: Type.String(), y: Type.Number() })
|
||||
const D = Type.Object({ x: Type.String() })
|
||||
const I = Type.Intersect([Type.Intersect([A, B]), Type.Intersect([C, D])])
|
||||
const R = Type.Index(I, ['x', 'y'])
|
||||
type O = Static<typeof R>
|
||||
Expect(R).ToBe<1>()
|
||||
}
|
||||
{
|
||||
type A = { x: string; y: 1 }
|
||||
type B = { x: number; y: number }
|
||||
type C = { x: string; y: number }
|
||||
type D = { x: string }
|
||||
type I = (A | B) & (C & D)
|
||||
type R = I['x' | 'y']
|
||||
|
||||
const A = Type.Object({ x: Type.String(), y: Type.Literal(1) })
|
||||
const B = Type.Object({ x: Type.Number(), y: Type.Number() })
|
||||
const C = Type.Object({ x: Type.String(), y: Type.Number() })
|
||||
const D = Type.Object({ x: Type.String() })
|
||||
const I = Type.Intersect([Type.Union([A, B]), Type.Intersect([C, D])])
|
||||
const R = Type.Index(I, ['x', 'y'])
|
||||
type O = Static<typeof R>
|
||||
Expect(R).ToBe<number | string>()
|
||||
}
|
||||
{
|
||||
type A = { x: 'A'; y: 1 }
|
||||
type B = { x: 'B'; y: number }
|
||||
type C = { x: 'C'; y: number }
|
||||
type D = { x: 'D' }
|
||||
type I = A | B | C | D
|
||||
type R = I['x']
|
||||
|
||||
const A = Type.Object({ x: Type.Literal('A'), y: Type.Literal(1) })
|
||||
const B = Type.Object({ x: Type.Literal('B'), y: Type.Number() })
|
||||
const C = Type.Object({ x: Type.Literal('C'), y: Type.Number() })
|
||||
const D = Type.Object({ x: Type.Literal('D') })
|
||||
const I = Type.Union([A, B, C, D])
|
||||
const R = Type.Index(I, Type.Union([Type.Literal('x')]))
|
||||
type O = Static<typeof R>
|
||||
Expect(R).ToBe<'A' | 'B' | 'C' | 'D'>()
|
||||
}
|
||||
{
|
||||
type I = {
|
||||
x: string
|
||||
y: number
|
||||
z: I
|
||||
}
|
||||
type R = I['x' | 'y' | 'z']
|
||||
const I = Type.Recursive((This) =>
|
||||
Type.Object({
|
||||
x: Type.String(),
|
||||
y: Type.Number(),
|
||||
z: This,
|
||||
}),
|
||||
)
|
||||
const R = Type.Index(I, ['x', 'y', 'z']) // z unresolvable
|
||||
type O = Static<typeof R>
|
||||
Expect(R).ToBe<string | number>()
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user