Skip to content

Commit b658d26

Browse files
committed
additional properties -> json input
1 parent 309978e commit b658d26

File tree

3 files changed

+49
-3
lines changed

3 files changed

+49
-3
lines changed

src/index.ts

+19-3
Original file line numberDiff line numberDiff line change
@@ -97,7 +97,10 @@ export function createCli<R extends AnyRouter>({router, ...params}: TrpcCliParam
9797
},
9898
},
9999
},
100-
getPojoInput: parsedCliParams => JSON.parse(parsedCliParams.options.input as string) as {},
100+
getPojoInput: parsedCliParams => {
101+
if (parsedCliParams.options.input == null) return parsedCliParams.options.input
102+
return JSON.parse(parsedCliParams.options.input as string) as {}
103+
},
101104
},
102105
incompatiblePairs: [],
103106
type,
@@ -182,7 +185,7 @@ export function createCli<R extends AnyRouter>({router, ...params}: TrpcCliParam
182185
})
183186

184187
const unusedFlagAliases: Record<string, string> = {...meta.aliases?.flags}
185-
Object.entries(flagJsonSchemaProperties).forEach(([propertyKey, propertyValue]) => {
188+
const addOptionForProperty = ([propertyKey, propertyValue]: [string, JsonSchema7Type]) => {
186189
const description = getDescription(propertyValue)
187190

188191
const longOption = `--${kebabCase(propertyKey)}`
@@ -356,7 +359,19 @@ export function createCli<R extends AnyRouter>({router, ...params}: TrpcCliParam
356359
)
357360

358361
command.addOption(option)
359-
})
362+
}
363+
364+
if (
365+
'additionalProperties' in procedureInputs.optionsJsonSchema &&
366+
procedureInputs.optionsJsonSchema.additionalProperties !== false
367+
) {
368+
program.allowUnknownOption()
369+
program.allowExcessArguments()
370+
command.allowUnknownOption()
371+
command.allowExcessArguments()
372+
}
373+
374+
Object.entries(flagJsonSchemaProperties).forEach(addOptionForProperty)
360375

361376
const invalidFlagAliases = Object.entries(unusedFlagAliases).map(([flag, alias]) => `${flag}: ${alias}`)
362377
if (invalidFlagAliases.length) {
@@ -368,6 +383,7 @@ export function createCli<R extends AnyRouter>({router, ...params}: TrpcCliParam
368383
program.__ran ||= []
369384
program.__ran.push(command)
370385
const options = command.opts()
386+
// console.dir({options, args}, {depth: null})
371387

372388
if (args.at(-2) !== options) {
373389
// This is a code bug and not recoverable. Will hopefully never happen but if commander totally changes their API this will break

src/parse-procedure.ts

+4
Original file line numberDiff line numberDiff line change
@@ -72,6 +72,10 @@ export function parseProcedureInputs(inputs: unknown[]): Result<ParsedProcedure>
7272
}
7373

7474
function handleMergedSchema(mergedSchema: JSONSchema7): Result<ParsedProcedure> {
75+
if (mergedSchema.additionalProperties) {
76+
return {success: false, error: `Inputs with additional properties are not currently supported`}
77+
}
78+
7579
if (mergedSchema.type === 'string') {
7680
return {
7781
success: true,

test/zod-inputs.test.ts

+26
Original file line numberDiff line numberDiff line change
@@ -437,6 +437,32 @@ test('mixed array input', async () => {
437437
expect(result).toMatchInlineSnapshot(`"list: [12,true,3.14,"null","undefined","hello"]"`)
438438
})
439439

440+
test('record input', async () => {
441+
const router = t.router({
442+
test: t.procedure
443+
.input(z.record(z.number()).optional()) //
444+
.query(({input}) => `input: ${JSON.stringify(input)}`),
445+
})
446+
447+
expect(await run(router, ['test', '--help'])).toMatchInlineSnapshot(`
448+
"Usage: program test [options]
449+
450+
Options:
451+
--input [json] Input formatted as JSON (procedure's schema couldn't be
452+
converted to CLI arguments: Inputs with additional properties
453+
are not currently supported)
454+
-h, --help display help for command
455+
"
456+
`)
457+
expect(await run(router, ['test'])).toMatchInlineSnapshot(`"input: undefined"`)
458+
expect(await run(router, ['test', '--input', '{"foo": 1}'])).toMatchInlineSnapshot(`"input: {"foo":1}"`)
459+
await expect(run(router, ['test', '--input', '{"foo": "x"}'])).rejects.toMatchInlineSnapshot(`
460+
CLI exited with code 1
461+
Caused by: CliValidationError: Validation error
462+
- Expected number, received string at "--foo"
463+
`)
464+
})
465+
440466
test("nullable array inputs aren't supported", async () => {
441467
const router = t.router({
442468
test1: t.procedure.input(z.array(z.string().nullable())).query(({input}) => `list: ${JSON.stringify(input)}`),

0 commit comments

Comments
 (0)