Skip to content

Commit 77eec8d

Browse files
committed
🎉 feat: module
1 parent f390af4 commit 77eec8d

File tree

6 files changed

+152
-291
lines changed

6 files changed

+152
-291
lines changed

CHANGELOG.md

+9-5
Original file line numberDiff line numberDiff line change
@@ -1,21 +1,25 @@
1-
# 0.0.4 - 4 Mar 2024
1+
# 0.0.5 - 5 Mar 2025
2+
Feature:
3+
- support `t.Module`, `t.Ref`
4+
5+
# 0.0.4 - 4 Mar 2025
26
Bug fix:
37
- handle undefined union
48

5-
# 0.0.3 - 4 Mar 2024
9+
# 0.0.3 - 4 Mar 2025
610
Bug fix:
711
- handle root array
812

9-
# 0.0.2 - 4 Mar 2024
13+
# 0.0.2 - 4 Mar 2025
1014
Feature:
1115
- support Record, Tuple, Union
1216

13-
# 0.0.1 - 4 Mar 2024
17+
# 0.0.1 - 4 Mar 2025
1418
Bug fix:
1519
- incorrect array bracket limit
1620
- handle deep nested optional object property
1721
- using pointer instead of created value for
1822

19-
# 0.0.0 - 4 Mar 2024
23+
# 0.0.0 - 4 Mar 2025
2024
Feature:
2125
- initial release

example/index.ts

+14-10
Original file line numberDiff line numberDiff line change
@@ -2,21 +2,25 @@ import { t } from 'elysia'
22
import { createMirror } from '../src/index'
33
import { TypeCompiler } from '@sinclair/typebox/compiler'
44

5-
const shape = t.Union([
6-
t.Undefined(),
7-
t.Object({
5+
const v = t.Module({
6+
a: t.Object({
87
name: t.String(),
9-
job: t.String(),
8+
job: t.Optional(t.Ref('job')),
109
trait: t.Optional(t.String())
11-
})
12-
])
10+
}),
11+
job: t.String()
12+
})
13+
14+
const shape = v.Import('a')
1315

14-
const value = undefined satisfies typeof shape.static
16+
const value = {
17+
name: 'Jane Doe',
18+
job: 'Software Engineer',
19+
trait: 'Friendly'
20+
} satisfies typeof shape.static
1521

1622
const mirror = createMirror(shape, {
1723
TypeCompiler
1824
})
1925

20-
console.dir(mirror(value), {
21-
depth: null
22-
})
26+
console.log(mirror(value))

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"name": "exact-mirror",
3-
"version": "0.0.4",
3+
"version": "0.0.5",
44
"description": "Mirror exact value to TypeBox/OpenAPI model",
55
"license": "MIT",
66
"scripts": {

src/index.ts

+26-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,9 @@
11
import { TypeCompiler, type TypeCheck } from '@sinclair/typebox/compiler'
22
import type { TAnySchema, TRecord } from '@sinclair/typebox'
3+
import { Insert } from '@sinclair/typebox/build/cjs/value'
34

45
const Kind = Symbol.for('TypeBox.Kind')
6+
const OptionalKind = Symbol.for('TypeBox.Optional')
57

68
const isSpecialProperty = (name: string) => /(\ |-|\t|\n)/.test(name)
79

@@ -66,6 +68,7 @@ interface Instruction {
6668
*/
6769
TypeCompiler?: typeof TypeCompiler
6870
typeCompilerWanred?: boolean
71+
definitions: Record<string, TAnySchema>
6972
}
7073

7174
const handleRecord = (
@@ -176,6 +179,16 @@ const mirror = (
176179

177180
const isRoot = property === 'v' && !instruction.unions.length
178181

182+
if (
183+
Kind in schema &&
184+
schema[Kind] === 'Import' &&
185+
schema.$ref in schema.$defs
186+
)
187+
return mirror(schema.$defs[schema.$ref], property, {
188+
...instruction,
189+
definitions: Object.assign(instruction.definitions, schema.$defs)
190+
})
191+
179192
if (
180193
isRoot &&
181194
schema.type !== 'object' &&
@@ -295,6 +308,13 @@ const mirror = (
295308
break
296309

297310
default:
311+
if (schema.$ref && schema.$ref in instruction.definitions)
312+
return mirror(
313+
instruction.definitions[schema.$ref],
314+
property,
315+
instruction
316+
)
317+
298318
if (Array.isArray(schema.anyOf)) {
299319
v = handleUnion(schema.anyOf, property, instruction)
300320

@@ -327,7 +347,10 @@ const mirror = (
327347

328348
export const createMirror = <T extends TAnySchema>(
329349
schema: T,
330-
{ TypeCompiler }: Pick<Instruction, 'TypeCompiler'> = {}
350+
{
351+
TypeCompiler,
352+
definitions = {}
353+
}: Partial<Pick<Instruction, 'TypeCompiler' | 'definitions'>> = {}
331354
): ((v: T['static']) => T['static']) => {
332355
const unions = <Instruction['unions']>[]
333356

@@ -338,7 +361,8 @@ export const createMirror = <T extends TAnySchema>(
338361
parentIsOptional: false,
339362
unions,
340363
unionKeys: {},
341-
TypeCompiler
364+
TypeCompiler,
365+
definitions
342366
})
343367

344368
if (!unions.length) return Function('v', f) as any

test/ref.test.ts

+102
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,102 @@
1+
import { t } from 'elysia'
2+
3+
import createMirror from '../src'
4+
5+
import { describe, it, expect } from 'bun:test'
6+
import { isEqual } from './utils'
7+
8+
describe('Ref', () => {
9+
it('handle module', () => {
10+
const modules = t.Module({
11+
object: t.Object({
12+
name: t.String(),
13+
optional: t.Optional(t.String())
14+
})
15+
})
16+
17+
const shape = modules.Import('object')
18+
19+
const value = {
20+
name: 'salt'
21+
} satisfies typeof shape.static
22+
23+
isEqual(shape, value)
24+
})
25+
26+
it('handle nested ref', () => {
27+
const modules = t.Module({
28+
object: t.Object({
29+
name: t.String(),
30+
info: t.Ref('info')
31+
}),
32+
info: t.Object({
33+
id: t.Number(),
34+
name: t.String()
35+
})
36+
})
37+
38+
const shape = modules.Import('object')
39+
40+
const value = {
41+
name: 'salt',
42+
info: {
43+
id: 123,
44+
name: 'salt'
45+
}
46+
} satisfies typeof shape.static
47+
48+
isEqual(shape, value)
49+
})
50+
51+
it('handle optional ref', () => {
52+
const modules = t.Module({
53+
object: t.Object({
54+
name: t.String(),
55+
info: t.Optional(t.Ref('info'))
56+
}),
57+
info: t.Object({
58+
id: t.Number(),
59+
name: t.String()
60+
})
61+
})
62+
63+
const shape = modules.Import('object')
64+
65+
const value = {
66+
name: 'salt'
67+
} satisfies typeof shape.static
68+
69+
isEqual(shape, {
70+
name: 'salt'
71+
})
72+
73+
isEqual(shape, {
74+
name: 'salt',
75+
info: {
76+
id: 123,
77+
name: 'salt'
78+
}
79+
})
80+
})
81+
82+
it('handle custom modules', () => {
83+
const definitions = {
84+
object: t.Object({
85+
name: t.String(),
86+
optional: t.Optional(t.String())
87+
})
88+
}
89+
90+
const shape = definitions.object
91+
92+
const value = {
93+
name: 'salt'
94+
} satisfies typeof shape.static
95+
96+
expect(
97+
createMirror(shape, {
98+
definitions
99+
})(value)
100+
).toEqual(value)
101+
})
102+
})

0 commit comments

Comments
 (0)