Benchmarks

How Attaform holds up across the Vue form-library field on the same demanding forms, measured in a real browser. Bundle size, supply-chain health, and per-scenario runtime, with every number traced to the run that produced it.

This page is an honest scoreboard. The numbers come from apps/bench-arena, a self-contained harness that drives every library through identical scenarios in real Chromium via Playwright, then writes one provenance-stamped results.json that this page renders directly. Nothing is hand-entered. Where Attaform leads, the run says so; where it pays a cost, the run says that too. Both ship from the same green run.

How to read this

A fair cross-library comparison has to account for the fact that these libraries do different amounts of work. The harness handles that with a few rules:

  • Layers are the fairness axis. Form-state libraries (Attaform among them) own reactive values, validation, and input binding. Validation-only libraries own validation against state you wire yourself. A batteries-included library renders its own inputs. Each row is labeled with its layer, so a validation-only engine mounting faster than a full form-state library is read as owning less, not as winning.
  • The DOM is held constant. Every headless library drives the same bare <input> markup over the same field count and the same schema, so a runtime number reflects the library's own machinery, not its choice of components.
  • Every library runs in its fastest idiomatic configuration. Debounces are neutralized, validation triggers are normalized, and array and union primitives use each library's native fast path. Attaform is measured on its shipping default, strict mode, never a relaxed setting.
  • Real builds, pinned validators. Attaform is consumed as its published dist, the same artifact an installer gets, minified in the same build as every other library. Zod v3 is pinned across the Zod-capable cohort.
  • Numbers normalize two ways. A ratio compares each library to Attaform at the same size. A slope compares a library to itself at the scenario's smallest size, so the shape of growth survives a change of machine.

The harness, every adapter, and the scenario generators live in apps/bench-arena. Found a fairer configuration for a library? The adapters are small and the README invites a pull request.

What it costs to adopt

Before any runtime number, three figures decide whether a library is worth reaching for: what it can express, what it adds to your bundle, and how its supply chain scores.

Capability coverage

What each library expresses as a first-class primitive versus composes by hand. A gap here becomes honest expressiveness data, never a rigged timing loss: a shape a library cannot express idiomatically is left out of its runtime rows rather than forced into a slow number.

LibraryLayerFlatDeeply nestedDynamic arraysGridDiscriminated unionMassiveWizard
Attaform Zod 3Form stateNativeNativeNativeNativeNativeNativeNative
vee-validate Zod 3Form stateNativeNativeNativeNativeHand-rolledNativeHand-rolled
@tanstack/vue-form Zod 3Form stateNativeNativeNativeNativeHand-rolledNativeHand-rolled
@formisch/vue ValibotForm stateNativeNativeNativeNativeHand-rolledNativeHand-rolled
Regle (schema) Zod 3Validation onlyNativeNativeNativeNativeNativeNativeHand-rolled
Regle (rules) native rulesValidation onlyNativeNativeNativeNativeHand-rolledNativeHand-rolled
FormKit Zod 3Batteries includedNativeNativeNativeNativeHand-rolledNativeHand-rolled
Vuelidate native rulesValidation onlyNativeNativeHand-rolledHand-rolledHand-rolledNativeHand-rolled

Native: a first-class primitive. Hand-rolled: composed from lower-level pieces. Dash: not expressed, which the runtime tables read as no number, never a slow one.

Source: CI run #27380201472mixed: AMD EPYC 7763 64-Core Processor + AMD EPYC 9V74 80-Core ProcessorNode v24.16.02026-06-11

Bundle size

Attaform is the heaviest in the cohort. That is the cost of shipping reactive form state, schema validation binding, persistence, undo and redo, and a multi-step wizard in one zero-dependency package, and it is the honest price of admission. The figure is the full bundle; an app that route-splits its forms pulls less on a first paint.

LibraryGzippedvs AttaformValidator
@formisch/vue
3.6 kB
0.05×valibot 1.4.1
Vuelidate
5.0 kB
0.07×native validators
vee-validate
24.6 kB
0.36×zod 3.25.76
@tanstack/vue-form
30.0 kB
0.44×zod 3.25.76
Regle
31.5 kB
0.46×zod 3.25.76
FormKit
43.2 kB
0.63×zod 3.25.76
Attaform
68.3 kB
zod 3.25.76

Each row is the same minimal real form (one text field, one email field, schema-validated, a submit handler) in that library's idiomatic API, with its validator weighed in. Vue is external, since every app ships it once. Attaform's figure is its full bundle; with route-level code-splitting a first paint pulls less.

Source: CI run #27380201472mixed: AMD EPYC 7763 64-Core Processor + AMD EPYC 9V74 80-Core ProcessorNode v24.16.02026-06-11

Supply-chain health

The OpenSSF Scorecard rates a project's adoption of supply-chain practices: branch protection, signed releases, pinned dependencies, CI hardening, and more. For a form library headed into an audited setting, that posture is part of the cost of adoption, so the benchmark stamps each project's current score alongside the size and runtime figures.

LibraryOpenSSF ScorecardAs ofLink
Attaform7.9 / 102026-06-11 Scorecard
vee-validate3.6 / 102026-06-08 Scorecard
@tanstack/vue-formNot published Repository
@formisch/vueNot published Repository
Regle (schema)Not published Repository
Regle (rules)Not published Repository
FormKitNot published Repository
Vuelidate4.4 / 102026-06-08 Scorecard

The OpenSSF Scorecard rates a project's adoption of supply-chain practices out of 10. An absent score has two meanings, kept distinct here. Not published means the project has not opted into a Scorecard, which is a choice, not a deficiency. Unavailable means the lookup did not complete on this run, a network gap on our side and never a statement about the project. Scores are point-in-time; the linked viewer shows the live result.

Source: CI run #27380201472mixed: AMD EPYC 7763 64-Core Processor + AMD EPYC 9V74 80-Core ProcessorNode v24.16.02026-06-11

Typing into a form

Keystroke latency

The headline interaction. A keystroke runs the value write, validation, and the re-render it triggers. On a flat form Attaform clears a 60 fps frame budget with room to spare.

Flat: Keystroke latency

In the leading group of form-state libraries on this run.

LibraryF10F50
Attaform
0.60 ms
1.10 ms
vee-validate
0.60 ms
0.90 ms 0.82×
@tanstack/vue-form
0.70 ms 1.17×
1.20 ms 1.09×
@formisch/vue
0.60 ms
1.10 ms
Regle (schema)
1.00 ms 1.67×
1.20 ms 1.09×
Regle (rules)
0.60 ms
0.80 ms 0.73×
FormKit
0.70 ms 1.17×
1.10 ms
Vuelidate
0.50 ms 0.83×
0.90 ms 0.82×
Source: CI run #27380201472mixed: AMD EPYC 7763 64-Core Processor + AMD EPYC 9V74 80-Core ProcessorNode v24.16.02026-06-11

At five thousand fields the picture tightens. The harness reports where Attaform lands plainly, and it is a scenario worth a future look.

Massive: Keystroke latency

Near the front of the form-state pack here.

LibraryL2000L5000
Attaform
14.2 ms
26.3 ms
vee-validate
12.4 ms 0.87×
20.3 ms 0.77×
@tanstack/vue-form
31.3 ms 2.2×
68.5 ms 2.6×
@formisch/vue
12.1 ms 0.85×
21.2 ms 0.8×
Regle (schema)
18.7 ms 1.32×
46.0 ms 1.74×
Regle (rules)
10.0 ms 0.7×
20.1 ms 0.76×
FormKit
4.95 ms 0.35×
did not finish> 5 min
Vuelidate
12.5 ms 0.88×
20.7 ms 0.79×
Source: CI run #27380201472mixed: AMD EPYC 7763 64-Core Processor + AMD EPYC 9V74 80-Core ProcessorNode v24.16.02026-06-11

Re-render scope

Editing one cell of a large grid should re-render one field, not the form. Configured optimally, the modern headless cohort all reaches that bound, and there is no lower number to beat.

Grid: Re-render scope per keystroke

One render per keystroke, whatever the form size. That is the design target, and the run holds it.

LibraryN20M8N100M8
Attaform
1 renders
1 renders
vee-validate
1 renders
1 renders
@tanstack/vue-form
1 renders
1 renders
@formisch/vue
1 renders
1 renders
Regle (schema)
1 renders
1 renders
Regle (rules)
1 renders
1 renders
FormKit
3 DOM mutations
3 DOM mutations
Vuelidate
1 renders
1 renders

† Reported as DOM mutations, a proxy for a library that owns its inputs rather than binding the shared bare field. Not directly comparable to a Vue render count.

Source: CI run #27380201472mixed: AMD EPYC 7763 64-Core Processor + AMD EPYC 9V74 80-Core ProcessorNode v24.16.02026-06-11

Standing up a form

Mounting a large form

Building a two-thousand-field form from scratch is where the form-state libraries separate. Attaform mounts the whole reactive tree, validation wiring included, every value and validation path live before the first paint.

Massive: Mount

Squarely in the form-state pack, neither out front nor at the back.

LibraryL2000
Attaform
178 ms
vee-validate
890 ms 4.99×
@tanstack/vue-form
7813 ms 43.81×
@formisch/vue
99.3 ms 0.56×
Regle (schema)
268 ms 1.5×
Regle (rules)
296 ms 1.66×
FormKit
12705 ms 71.23×
Vuelidate
135 ms 0.75×
Source: CI run #27380201472mixed: AMD EPYC 7763 64-Core Processor + AMD EPYC 9V74 80-Core ProcessorNode v24.16.02026-06-11

Memory

Retained heap after mount, the library's reactive and validation state at scenario size. Churn is the per-cycle allocation pressure, and leak is the residual across mount and teardown cycles. The sparkline traces retained heap across the measured cycles.

Massive: Retained heap

Among the faster form-state libraries on this shape.

LibraryL2000
Attaform
11814 kB
churn 16333 kB · leak 42 kB
vee-validate
22638 kB1.92×
churn 1562 kB · leak 1479 kB
@tanstack/vue-form
22550 kB1.91×
churn 5131 kB · leak 1650 kB
@formisch/vue
11516 kB0.97×
churn 1556 kB · leak 4 kB
Regle (schema)
33530 kB2.84×
churn 5231 kB · leak 24 kB
Regle (rules)
41671 kB3.53×
churn 570 kB · leak 34 kB
FormKit
377189 kB31.93×
churn 1036 kB · leak 27 kB
Vuelidate
14820 kB1.25×
churn 3511 kB · leak 8 kB
Source: CI run #27380201472mixed: AMD EPYC 7763 64-Core Processor + AMD EPYC 9V74 80-Core ProcessorNode v24.16.02026-06-11

Working the harder shapes

Validation throughput

A full-form validation pass over a massive form, the cost of a submit on the largest shape in the suite.

Massive: Full-form validation

Near the front of the form-state pack here.

LibraryL2000L5000
Attaform
2.00 ms
2.90 ms
vee-validate
18.8 ms 9.38×
38.4 ms 13.24×
@tanstack/vue-form
938 ms 468.8×
did not finish> 5 min
@formisch/vue
1.40 ms 0.7×
2.70 ms 0.93×
Regle (schema)
30.2 ms 15.1×
88.8 ms 30.62×
Regle (rules)
95.1 ms 47.55×
238 ms 82.21×
FormKit
385 ms 192.3×
did not finish> 5 min
Vuelidate
0.30 ms 0.15×
0.80 ms 0.28×
Source: CI run #27380201472mixed: AMD EPYC 7763 64-Core Processor + AMD EPYC 9V74 80-Core ProcessorNode v24.16.02026-06-11

Discriminated unions

Writing into and flipping between variants of a discriminated union. Attaform walks only the active branch, so a variant flip touches the branch in play rather than every alternative.

Discriminated union: Variant flip

Near the front of the form-state pack here.

LibraryDU
Attaform
1.00 ms
vee-validate
1.10 ms 1.1×
@tanstack/vue-form
1.00 ms
@formisch/vue
0.80 ms 0.8×
Regle (schema)
1.40 ms 1.4×
Regle (rules)
1.70 ms 1.7×
FormKit
6.90 ms 6.9×
Vuelidate
0.80 ms 0.8×
Source: CI run #27380201472mixed: AMD EPYC 7763 64-Core Processor + AMD EPYC 9V74 80-Core ProcessorNode v24.16.02026-06-11

Dynamic arrays

Appending and reordering rows in a growing list. Attaform's array helpers copy the target array before mutating, which keeps reads fast and identity stable but shows up as real cost on a reorder at a hundred rows. It is an honest line on the board.

Dynamic arrays: Array append

Behind the form-state pack on this shape, and the run shows it plainly.

LibraryN10N100
Attaform
2.60 ms
11.6 ms
vee-validate
2.10 ms 0.81×
10.0 ms 0.86×
@tanstack/vue-form
1.20 ms 0.46×
3.50 ms 0.3×
@formisch/vue
0.70 ms 0.27×
1.40 ms 0.12×
Regle (schema)
2.00 ms 0.77×
11.6 ms
Regle (rules)
1.90 ms 0.73×
10.0 ms 0.86×
FormKit
4.60 ms 1.77×
11.0 ms 0.95×
Vuelidate
1.20 ms 0.46×
3.20 ms 0.28×
Source: CI run #27380201472mixed: AMD EPYC 7763 64-Core Processor + AMD EPYC 9V74 80-Core ProcessorNode v24.16.02026-06-11
Dynamic arrays: Array reorder

Off the lead here among the form-state libraries, a line we are actively sharpening.

LibraryN10N100
Attaform
1.80 ms
6.85 ms
vee-validate
1.00 ms 0.56×
4.65 ms 0.68×
@tanstack/vue-form
0.70 ms 0.39×
2.60 ms 0.38×
@formisch/vue
0.60 ms 0.33×
1.30 ms 0.19×
Regle (schema)
1.00 ms 0.56×
6.10 ms 0.89×
Regle (rules)
0.90 ms 0.5×
5.45 ms 0.8×
FormKit
1.50 ms 0.83×
4.60 ms 0.67×
Vuelidate
0.70 ms 0.39×
2.00 ms 0.29×
Source: CI run #27380201472mixed: AMD EPYC 7763 64-Core Processor + AMD EPYC 9V74 80-Core ProcessorNode v24.16.02026-06-11

Multi-step wizard

Most of the cohort has no wizard primitive and composes a multi-step flow by hand, so this is an expressiveness comparison as much as a timing one. Where a comparable step transition exists, here is its cost; Attaform's useWizard ships the flow as a first-class shape.

Wizard: Step transition

Holds the middle of the form-state field on this shape.

LibraryS4
Attaform
0.40 ms
vee-validate
5.40 ms 13.5×
@tanstack/vue-form
0.40 ms
@formisch/vue
0.10 ms 0.25×
Regle (schema)
0.90 ms 2.25×
Regle (rules)
0.10 ms 0.25×
FormKit
2.00 ms
Vuelidate
0.10 ms 0.25×
Source: CI run #27380201472mixed: AMD EPYC 7763 64-Core Processor + AMD EPYC 9V74 80-Core ProcessorNode v24.16.02026-06-11

Caveats

The methodology is only as good as what it admits.

  • FormKit owns its inputs. It cannot drive the shared bare field, so its re-render figure is a DOM-mutation proxy, marked in the tables, and its mount and memory figures include its own component tree. It is labeled batteries-included throughout and never placed silently beside bare-input libraries.
  • Heap is Chromium-quantized. usedJSHeapSize reports rounded magnitudes, not byte-exact values, so memory figures show whole kilobytes and the slope across sizes carries more signal than any single number.
  • Every cell shares one time budget. Each measured cell gets the same per-cell ceiling on the CI runner, identical for every library. A cell that cannot settle to a stable median inside it (a single mount of thousands of fields, or a full-form validation at the largest sizes on the heaviest libraries) is recorded as "did not finish" rather than dropped or estimated. The ceiling is uniform across the cohort, so it marks where a shape outgrows one measurement window, never a verdict on a library.
  • Bundle is total, not first-paint. Every figure is the full minified and gzipped cost with the validator weighed in. Vue is external, since every app ships it once. Code-splitting changes what a first paint actually pulls.
  • An absent score has two distinct meanings. "Not published" means a project has not opted into a Scorecard, which is a choice and not a deficiency. "Unavailable" means the lookup did not complete on that run, a network gap on our side and never a statement about the project. The viewer linked on each row shows the live result either way, and scores are point-in-time.
  • Local versus CI. The committed numbers come from CI on a fixed runner. A figure stamped "local run" is illustrative shape data from a developer machine, superseded the next time CI refreshes the page.

Reproduce it yourself

The harness is meant to be run by hand. Clone the repo, build Attaform's real dist with pnpm prepack, then from apps/bench-arena install the browser and run the arena. The full instructions live in the bench-arena README. Every adapter mounts by query parameter, so you can also open a single library and scenario in a browser and watch it work.

Where to next