Async refinements
Predicates that await a server round-trip: uniqueness probes, slug availability, password-breach lookups. Same surface as sync refinements, just with
async.
- Category
- Schema pattern
- Triggers
- validateOn cadence + handleSubmit gate
- In-flight signal
fields.<path>.validating- Submit awaits
- Yes, every pending refinement before onSubmit
Type a username and blur the field to watch validating flip true for ~700ms while the simulated check runs. Try ada, champ, or athlete to see the "taken" error land. Try any unused name to see it accept. The submit handler awaits every in-flight refinement before dispatching; submitting mid-check holds until the check resolves.
Declare an async predicate
Zod's .refine accepts an async function:
z.string().refine(async (v) => isAvailable(v), {
message: 'That username is taken',
})
The predicate runs alongside sync refinements; the chain awaits its resolution before deciding pass / fail. Attaform forwards the awaited result to form.errors.<path> and fields.<path> like any other refinement.
In-flight signal
While an async refinement is pending at a path:
fields.<path>.validatingistrue.meta.validatingistruewhen ANY field has a pending async refinement.
Render a "Checking…" indicator next to the field:
<small v-if="fields.username.validating">Checking availability…</small>
This is per-field UX; for a form-level spinner reach for meta.validating instead.
handleSubmit awaits
The submit handler waits for every pending async refinement before deciding pass / fail:
- Sync validation runs across every active path.
- Async refinements await.
- If every refinement passes,
onSubmit(values)fires with the parsed Zod output. - If anything fails, focus pulls to the first invalid field and
onError(errors)fires.
Submitting mid-check is safe: the handler holds until the check resolves, then routes through onSubmit or onError. No flash-of-valid window where the user hits submit while a slow uniqueness probe hasn't finished.
Debouncing keystroke triggers
By default, sync refinements run on every committed write (with validateOn: 'change', which pairs with the directive's per-keystroke commit). For async refinements, you usually want blur: server probes shouldn't fire on every keystroke. Set validateOn: 'blur' per form:
useForm({
schema,
validateOn: 'blur', // async probes fire on blur, not keystroke
})
Blur fires the probe only when the value actually changed since the last pass, so refocusing a field and tabbing away without editing it won't re-hit the server.
Or stay on the per-keystroke trigger and coalesce bursts with debounceMs:
useForm({
schema,
validateOn: 'change',
debounceMs: 400, // wait 400ms of quiet before validating
})
See When validation runs for the full timing API.
Race-safety
Two rapid edits before the first probe returns: the directive cancels the stale in-flight request (where the underlying client supports AbortSignal) or ignores its result (where it doesn't). The newest probe's result is the one that lands in errors.<path>. No "earlier request resolves last, overwrites the correct error" race.
Where to next
- The validation lifecycle: the imperative
validateAsync()for non-submit code paths. - When validation runs: the
validateOncadence knob. handleSubmit: the dispatch surface that awaits async refinements.