1
I recently added sync/async mode support to Optique, a type-safe CLI parserfor TypeScript. It turned out to be one of the trickier features I'veimplemented—the object() combinator alone needed to compute a combined modefrom all its child parsers, and TypeScript's inference kept hitting edge cases.
What is Optique?
Optique is a type-safe, combinatorial CLI parser for TypeScript, inspired byHaskell's optparse-applicative. Instead of decorators or builder patterns,you compose small parsers into larger ones using combinators, and TypeScriptinfers the result types.
Here's a quick taste:
import { object } from "@optique/core/constructs";import { argument, option } from "@optique/core/primitives";import { string, integer } from "@optique/core/valueparser";import { run } from "@optique/run";const cli = object({ name: argument(string()), count: option("-n", "--count", integer()),});// TypeScript infers: { name: string; count: number | undefined }const result = run(cli); // sync by default
The type inference works through arbitrarily deep compositions—in most cases,you don't need explicit type annotations.
How it started
Lucas Garron (@lgarron) opened an issue requestingasync support for shell completions. He wanted to provideTab-completion suggestions by running shell commands likegit for-each-ref to list branches and tags.
// Lucas's example: fetching Git branches and tags in parallelconst [branches, tags] = await Promise.all([ $`git for-each-ref --format='%(refname:short)' refs/heads/`.text(), $`git for-each-ref --format='%(refname:short)' refs/tags/`.text(),]);
At first, I didn't like the idea. Optique's entire API was synchronous, whichmade it simpler to reason about and avoided the “async infection” problem whereone async function forces everything upstream to become async. I argued thatshell completion should be near-instantaneous, and if you need async data, youshould cache it at startup.
But Lucas pushed back. The filesystem is a database, and many usefulcompletions inherently require async work—Git refs change constantly, andpre-caching everything at startup doesn't scale for large repos. Fair point.
What I needed to solve
So, how do you support both sync and async execution modes in a composableparser library while maintaining type safety?
The key requirements were:
parse() returns T or Promise
complete() returns T or Promise
suggest() returns Iterable or AsyncIterable
When combining parsers, if any parser is async, the combined resultmust be async
Existing sync code should continue to work unchanged
The fourth requirement is the tricky one. Consider this:
const syncParser = flag("--verbose");const asyncParser = option("--branch", asyncValueParser);// What's the type of this?const combined = object({ verbose: syncParser, branch: asyncParser });
The combined parser should be async because one of its fields is async.This means we need type-level logic to compute the combined mode.
Five design options
I explored five different approaches, each with its own trade-offs.
Option A: conditional types with mode parameter
Add a mode type parameter to Parser and use conditional types:
type Mode = "sync" | "async";type ModeValue = M extends "async" ? Promise : T;interface Parser { parse(context: ParserContext): ModeValue>; // ...}
The challenge is computing combined modes:
type CombineModes>> = T[keyof T] extends Parser ? M extends "async" ? "async" : "sync" : never;
Option B: mode parameter with default value
A variant of Option A, but place the mode parameter first with a defaultof "sync":
interface Parser { readonly $mode: M; // ...}
The default value maintains backward compatibility—existing user code keepsworking without changes.
Option C: separate interfaces
Define completely separate Parser and AsyncParser interfaces withexplicit conversion:
interface Parser { /* sync methods */ }interface AsyncParser { /* async methods */ }function toAsync(parser: Parser): AsyncParser;
Simpler to understand, but requires code duplication and explicit conversions.
Option D: union return types for suggest() only
The minimal approach. Only allow suggest() to be async:
interface Parser { parse(context: ParserContext): ParserResult; // always sync suggest(context: ParserContext, prefix: string): Iterable | AsyncIterable; // can be either}
This addresses the original use case but doesn't help if async parse() isever needed.
Option E: fp-ts style HKT simulation
Use the technique from fp-ts to simulate Higher-Kinded Types:
interface URItoKind { Identity: A; Promise: Promise;}type Kind, A> = URItoKind[F];interface Parser, TValue, TState> { parse(context: ParserContext): Kind>;}
The most flexible approach, but with a steep learning curve.
Testing the idea
Rather than commit to an approach based on theoretical analysis, I createda prototype to test how well TypeScript handles the type inference in practice.I published my findings in the GitHub issue:
Both approaches correctly handle the “any async → all async” rule at thetype level. (…) Complex conditional types likeModeValue, ParserResult> sometimes requireexplicit type casting in the implementation. This only affects libraryinternals. The user-facing API remains clean.
The prototype validated that Option B (explicit mode parameter with default)would work. I chose it for these reasons:
Backward compatible: The default "sync" keeps existing code working
Explicit: The mode is visible in both types and runtime (via a $modeproperty)
Debuggable: Easy to inspect the current mode at runtime
Better IDE support: Type information is more predictable
How CombineModes works
The CombineModes type computes whether a combined parser should be sync orasync:
type CombineModes = "async" extends T[number] ? "async" : "sync";
This type checks if "async" is present anywhere in the tuple of modes.If so, the result is "async"; otherwise, it's "sync".
For combinators like object(), I needed to extract modes from parserobjects and combine them:
// Extract the mode from a single parsertype ParserMode = T extends Parser ? M : never;// Combine modes from all values in a record of parserstype CombineObjectModes>> = CombineModes<{ [K in keyof T]: ParserMode }[keyof T][]>;
Runtime implementation
The type system handles compile-time safety, but the implementation also needsruntime logic. Each parser has a $mode property that indicates its executionmode:
const syncParser = option("-n", "--name", string());console.log(syncParser.$mode); // "sync"const asyncParser = option("-b", "--branch", asyncValueParser);console.log(asyncParser.$mode); // "async"
Combinators compute their mode at construction time:
function object>>( parsers: T): Parser, ObjectValue, ObjectState> { const parserKeys = Reflect.ownKeys(parsers); const combinedMode: Mode = parserKeys.some( (k) => parsers[k as keyof T].$mode === "async" ) ? "async" : "sync"; // ... implementation}
Refining the API
Lucas suggested an important refinement during ourdiscussion. Instead of having run() automatically choose between sync andasync based on the parser mode, he proposed separate functions:
Perhaps run(…) could be automatic, and runSync(…) and runAsync(…) couldenforce that the inferred type matches what is expected.
So we ended up with:
run(): automatic based on parser mode
runSync(): enforces sync mode at compile time
runAsync(): enforces async mode at compile time
// Automatic: returns T for sync parsers, Promise for asyncconst result1 = run(syncParser); // stringconst result2 = run(asyncParser); // Promise// Explicit: compile-time enforcementconst result3 = runSync(syncParser); // stringconst result4 = runAsync(asyncParser); // Promise// Compile error: can't use runSync with async parserconst result5 = runSync(asyncParser); // Type error!
I applied the same pattern to parse()/parseSync()/parseAsync() andsuggest()/suggestSync()/suggestAsync() in the facade functions.
Creating async value parsers
With the new API, creating an async value parser for Git branches lookslike this:
import type { Suggestion } from "@optique/core/parser";import type { ValueParser, ValueParserResult } from "@optique/core/valueparser";function gitRef(): ValueParser<"async", string> { return { $mode: "async", metavar: "REF", parse(input: string): Promise> { return Promise.resolve({ success: true, value: input }); }, format(value: string): string { return value; }, async *suggest(prefix: string): AsyncIterable { const { $ } = await import("bun"); const [branches, tags] = await Promise.all([ $`git for-each-ref --format='%(refname:short)' refs/heads/`.text(), $`git for-each-ref --format='%(refname:short)' refs/tags/`.text(), ]); for (const ref of [...branches.split("\n"), ...tags.split("\n")]) { const trimmed = ref.trim(); if (trimmed && trimmed.startsWith(prefix)) { yield { kind: "literal", text: trimmed }; } } }, };}
Notice that parse() returns Promise.resolve() even though it's synchronous.This is because the ValueParser<"async", T> type requires all methods to useasync signatures. Lucas pointed out this is a minor ergonomic issue. If onlysuggest() needs to be async, you still have to wrap parse() in a Promise.
I considered per-method mode granularity (e.g., ValueParser), but the implementation complexity would multiplysubstantially. For now, the workaround is simple enough:
// Option 1: Use Promise.resolve()parse(input) { return Promise.resolve({ success: true, value: input });}// Option 2: Mark as async and suppress the linter// biome-ignore lint/suspicious/useAwait: sync implementation in async ValueParserasync parse(input) { return { success: true, value: input };}
What it cost
Supporting dual modes added significant complexity to Optique's internals.Every combinator needed updates:
Type signatures grew more complex with mode parameters
Mode propagation logic had to be added to every combinator
Dual implementations were needed for sync and async code paths
Type casts were sometimes necessary in the implementation to satisfyTypeScript
For example, the object() combinator went from around 100 lines to around250 lines. The internal implementation uses conditional logic based on thecombined mode:
if (combinedMode === "async") { return { $mode: "async" as M, // ... async implementation with Promise chains async parse(context) { // ... await each field's parse result }, };} else { return { $mode: "sync" as M, // ... sync implementation parse(context) { // ... directly call each field's parse }, };}
This duplication is the cost of supporting both modes without runtime overheadfor sync-only use cases.
Lessons learned
Listen to users, but validate with prototypes
My initial instinct was to resist async support. Lucas's persistence andconcrete examples changed my mind, but I validated the approach with aprototype before committing. The prototype revealed practical issues (likeTypeScript inference limits) that pure design analysis would have missed.
Backward compatibility is worth the complexity
Making "sync" the default mode meant existing code continued to workunchanged. This was a deliberate choice. Breaking changes should requireuser action, not break silently.
Unified mode vs per-method granularity
I chose unified mode (all methods share the same sync/async mode) overper-method granularity. This means users occasionally writePromise.resolve() for methods that don't actually need async, but thealternative was multiplicative complexity in the type system.
Designing in public
The entire design process happened in a public GitHub issue. Lucas, Giuseppe,and others contributed ideas that shaped the final API. TherunSync()/runAsync() distinction came directly from Lucas's feedback.
Conclusion
This was one of the more challenging features I've implemented in Optique.TypeScript's type system is powerful enough to encode the “any async means allasync” rule at compile time, but getting there required careful design work andprototyping.
What made it work: conditional types like ModeValue can bridge the gapbetween sync and async worlds. You pay for it with implementation complexity,but the user-facing API stays clean and type-safe.
Optique 0.9.0 with async support is currently in pre-release testing. Ifyou'd like to try it, check out PR #70 or install the pre-release:
npm add @optique/core@0.9.0-dev.212 @optique/run@0.9.0-dev.212deno add --jsr @optique/core@0.9.0-dev.212 @optique/run@0.9.0-dev.212
Feedback is welcome!
You must log in or register to comment.

