Test each framework for it's performance, particularly common Lighthouse and CWV metrics as applications scale
Important: This is not a measure of "is framework x better than y". There are many tradeoffs to weigh when choosing the best framework for you - such as DX, features, familiarity, ecosystem, documentation, etc. These benchmarks only show a part of the picture.
The goal for this project are to understand the performance tradeoffs of popular frameworks in real world-ish scenarios. We want to assume non trivial codebases and imperfect code and conditions, so to see how each framework holds up to real world scenarios and scale.
We are intentionally not focused on client side rendering performance, for that please use Stefan Krause's great js-framework-benchmark
This project is in initial development. Do not put too much weight on these current results, there is still much more to do to ensure accuracy, consistency, and fairness.
Contributions are welcome!
We created a basic starting point for each framework in the frameworks/
folder using each framework's suggested starter/cli.
We then create basic example components and use Mitosis to compile them to best-effort idiomatic code for each framework. This will never be perfectly optimized code
We then build and serve each project, and run Lighthouse on each project with puppeteer, including with emulation of slow devices and networks (aka includes CPU and network throttling), and measure:
<script>
tags for the initial page load. This is the actual size transferred over the network, including compression (lower is better)We take the median of multiple runs, and sort the results by TTI, ascending
We are also experimenting with looking at other metrics, such as build times
Alphabetically:
Important: This project is still in initial development. Do not put too much weight on these current results, there is still much more to do to ensure accuracy, consistency, and fairness.
Jump to:
A more feature rich app for displaying table data, sorting, filtering, etc. Uses more JS, more like a median website. Source
┌─────────┬────────────┬──────────┬──────────┬──────────┬──────────┬───────┬──────────────┬───────────┐
│ (index) │ name │ TTI │ FCP │ LCP │ TBT │ Score │ Eager JS KiB │ Total KiB │
├─────────┼────────────┼──────────┼──────────┼──────────┼──────────┼───────┼──────────────┼───────────┤
│ 0 │ 'qwik' │ '0.6 s' │ '0.6 s' │ '1.5 s' │ '0 ms' │ 100 │ 2 │ 38 │
│ 1 │ 'react' │ '0.8 s' │ '0.8 s' │ '2.4 s' │ '0 ms' │ 98 │ 187 │ 199 │
│ 2 │ 'gatsby' │ '0.8 s' │ '0.8 s' │ '1.4 s' │ '0 ms' │ 100 │ 82 │ 87 │
│ 3 │ 'lit' │ '0.8 s' │ '0.8 s' │ '1.1 s' │ '0 ms' │ 100 │ 23 │ 25 │
│ 4 │ 'solid' │ '0.9 s' │ '0.6 s' │ '1.3 s' │ '0 ms' │ 85 │ 24 │ 28 │
│ 5 │ 'astro' │ '0.9 s' │ '0.9 s' │ '1.1 s' │ '0 ms' │ 100 │ 15 │ 35 │
│ 6 │ 'marko' │ '1.0 s' │ '0.8 s' │ '0.9 s' │ '10 ms' │ 100 │ 24 │ 33 │
│ 7 │ 'fresh' │ '1.3 s' │ '1.3 s' │ '1.5 s' │ '0 ms' │ 100 │ 17 │ 46 │
│ 8 │ 'next' │ '1.6 s' │ '0.6 s' │ '1.2 s' │ '10 ms' │ 100 │ 91 │ 103 │
│ 9 │ 'svelte' │ '1.6 s' │ '1.6 s' │ '1.7 s' │ '0 ms' │ 99 │ 29 │ 35 │
│ 10 │ 'angular' │ '1.7 s' │ '1.5 s' │ '1.5 s' │ '150 ms' │ 98 │ 86 │ 88 │
│ 11 │ 'nuxt3' │ '1.7 s' │ '1.7 s' │ '1.7 s' │ '0 ms' │ 99 │ 59 │ 65 │
│ 12 │ 'vue3' │ '1.8 s' │ '1.2 s' │ '2.1 s' │ '0 ms' │ 94 │ 41 │ 50 │
│ 13 │ 'nuxt2' │ '2.1 s' │ '1.2 s' │ '2.1 s' │ '70 ms' │ 98 │ 106 │ 118 │
└─────────┴────────────┴──────────┴──────────┴──────────┴──────────┴───────┴──────────────┴───────────┘
A very simple/trivial interactive Todo app. Source
Ordered by TTI, ascending:
┌─────────┬────────────┬─────────┬─────────┬─────────┬──────────┬───────┬──────────────┬───────────┐
│ (index) │ name │ TTI │ FCP │ LCP │ TBT │ Score │ Eager JS KiB │ Total KiB │
├─────────┼────────────┼─────────┼─────────┼─────────┼──────────┼───────┼──────────────┼───────────┤
│ 0 │ 'astro' │ '0.6 s' │ '0.6 s' │ '0.6 s' │ '0 ms' │ 100 │ 20 │ 32 │
│ 1 │ 'qwik' │ '0.7 s' │ '0.7 s' │ '1.2 s' │ '0 ms' │ 100 │ 2 │ 25 │
│ 2 │ 'react' │ '0.8 s' │ '0.8 s' │ '2.2 s' │ '0 ms' │ 99 │ 159 │ 171 │
│ 3 │ 'fresh' │ '0.8 s' │ '0.8 s' │ '0.9 s' │ '0 ms' │ 100 │ 9 │ 37 │
│ 4 │ 'lit' │ '0.8 s' │ '0.8 s' │ '1.1 s' │ '0 ms' │ 100 │ 16 │ 18 │
│ 5 │ 'solid' │ '1.0 s' │ '0.7 s' │ '1.3 s' │ '40 ms' │ 86 │ 17 │ 19 │
│ 6 │ 'marko' │ '1.1 s' │ '0.8 s' │ '0.9 s' │ '10 ms' │ 100 │ 17 │ 23 │
│ 7 │ 'vue3' │ '1.2 s' │ '1.2 s' │ '1.8 s' │ '10 ms' │ 99 │ 33 │ 41 │
│ 8 │ 'svelte' │ '1.5 s' │ '1.5 s' │ '1.5 s' │ '0 ms' │ 100 │ 19 │ 24 │
│ 9 │ 'nuxt3' │ '1.5 s' │ '1.5 s' │ '1.7 s' │ '0 ms' │ 99 │ 50 │ 55 │
│ 10 │ 'gatsby' │ '1.6 s' │ '0.8 s' │ '1.1 s' │ '0 ms' │ 100 │ 70 │ 75 │
│ 11 │ 'nuxt2' │ '1.6 s' │ '1.0 s' │ '1.0 s' │ '40 ms' │ 100 │ 95 │ 106 │
│ 12 │ 'angular' │ '1.6 s' │ '1.5 s' │ '1.6 s' │ '30 ms' │ 99 │ 72 │ 74 │
│ 13 │ 'next' │ '2.2 s' │ '0.7 s' │ '0.8 s' │ '110 ms' │ 99 │ 83 │ 94 │
└─────────┴────────────┴─────────┴─────────┴─────────┴──────────┴───────┴──────────────┴───────────┘
Just a few links and <h1>Hello World</h1>
. Source
Ordered by TTI, ascending:
┌─────────┬────────────┬─────────┬─────────┬─────────┬──────────┬───────┬──────────────┬───────────┐
│ (index) │ name │ TTI │ FCP │ LCP │ TBT │ Score │ Eager JS KiB │ Total KiB │
├─────────┼────────────┼─────────┼─────────┼─────────┼──────────┼───────┼──────────────┼───────────┤
│ 0 │ 'qwik' │ '0.7 s' │ '0.7 s' │ '0.7 s' │ '0 ms' │ 100 │ 0 │ 4 │
│ 1 │ 'astro' │ '0.7 s' │ '0.7 s' │ '0.7 s' │ '0 ms' │ 100 │ 0 │ 9 │
│ 2 │ 'react' │ '0.8 s' │ '0.8 s' │ '2.2 s' │ '0 ms' │ 99 │ 154 │ 166 │
│ 3 │ 'fresh' │ '0.8 s' │ '0.8 s' │ '0.8 s' │ '0 ms' │ 100 │ 0 │ 27 │
│ 4 │ 'marko' │ '0.8 s' │ '0.8 s' │ '1.1 s' │ '0 ms' │ 100 │ 15 │ 21 │
│ 5 │ 'lit' │ '0.8 s' │ '0.6 s' │ '0.9 s' │ '10 ms' │ 100 │ 15 │ 16 │
│ 6 │ 'solid' │ '0.9 s' │ '0.9 s' │ '1.1 s' │ '0 ms' │ 100 │ 16 │ 18 │
│ 7 │ 'vue3' │ '1.2 s' │ '1.2 s' │ '1.5 s' │ '0 ms' │ 100 │ 31 │ 38 │
│ 8 │ 'gatsby' │ '1.4 s' │ '0.6 s' │ '1.0 s' │ '0 ms' │ 100 │ 69 │ 73 │
│ 9 │ 'svelte' │ '1.5 s' │ '1.5 s' │ '1.5 s' │ '0 ms' │ 100 │ 18 │ 22 │
│ 10 │ 'nuxt2' │ '1.5 s' │ '0.9 s' │ '0.9 s' │ '120 ms' │ 99 │ 93 │ 103 │
│ 11 │ 'nuxt3' │ '1.5 s' │ '1.5 s' │ '1.7 s' │ '0 ms' │ 99 │ 50 │ 57 │
│ 12 │ 'angular' │ '1.7 s' │ '1.5 s' │ '1.5 s' │ '150 ms' │ 98 │ 72 │ 74 │
│ 13 │ 'hydrogen' │ '1.8 s' │ '0.6 s' │ '1.6 s' │ '30 ms' │ 91 │ 160 │ 172 │
│ 14 │ 'next' │ '2.1 s' │ '0.7 s' │ '0.8 s' │ '40 ms' │ 100 │ 82 │ 93 │
└─────────┴────────────┴─────────┴─────────┴─────────┴──────────┴───────┴──────────────┴───────────┘
Time it took to server-side render the /dashboard page in milliseconds. Smaller numbers are better.
┌─────────┬────────────┬─────┬─────┬─────┬────────┬─────────┐
│ (index) │ name │ 1% │ 50% │ 99% │ Avg │ Std Dev │
├─────────┼────────────┼─────┼─────┼─────┼────────┼─────────┤
│ 0 │ 'marko' │ 1 │ 1 │ 4 │ 1.29 │ 0.8 │
│ 1 │ 'fresh' │ 4 │ 4 │ 6 │ 4.19 │ 0.88 │
│ 2 │ 'hydrogen' │ 4 │ 5 │ 14 │ 5.63 │ 4.39 │
│ 3 │ 'svelte' │ 6 │ 7 │ 18 │ 8.17 │ 3.12 │
│ 4 │ 'solid' │ 6 │ 8 │ 22 │ 8.77 │ 3.8 │
│ 5 │ 'nuxt3' │ 19 │ 25 │ 68 │ 29.49 │ 18.09 │
│ 6 │ 'nuxt2' │ 11 │ 14 │ 32 │ 15.04 │ 4.89 │
│ 7 │ 'astro' │ 11 │ 14 │ 38 │ 16.56 │ 5.98 │
│ 8 │ 'remix' │ 12 │ 18 │ 62 │ 20.3 │ 8.87 │
│ 9 │ 'gatsby' │ 27 │ 33 │ 103 │ 36.9 │ 12.86 │
│ 10 │ 'next' │ 35 │ 40 │ 113 │ 46.96 │ 19.06 │
│ 11 │ 'angular' │ 113 │ 127 │ 400 │ 141.73 │ 47.64 │
└─────────┴────────────┴─────┴─────┴─────┴────────┴─────────┘
SSR throughput of the dashboard page, measured by autocannon, sorted by 99% percentile descending. Larger numbers are better.
┌─────────┬───────────────────┬──────┬──────┬──────┬─────────┬─────────┐
│ (index) │ name │ 1% │ 50% │ 99% │ Avg │ Std Dev │
├─────────┼───────────────────┼──────┼──────┼──────┼─────────┼─────────┤
│ 0 │ 'marko' │ 3677 │ 5411 │ 5951 │ 5263.1 │ 702.18 │
│ 1 │ 'fresh' │ 1748 │ 2057 │ 2123 │ 1997.73 │ 117.92 │
│ 2 │ 'preact-ssr-node' │ 1755 │ 2065 │ 2185 │ 2052.6 │ 115.11 │
│ 3 │ 'hydrogen' │ 843 │ 1653 │ 1807 │ 1528.1 │ 290.27 │
│ 4 │ 'svelte' │ 536 │ 820 │ 1107 │ 795.3 │ 182.24 │
│ 5 │ 'solid' │ 534 │ 842 │ 1019 │ 830.64 │ 138.46 │
│ 6 │ 'astro' │ 444 │ 573 │ 641 │ 579 │ 56.1 │
│ 7 │ 'nuxt2' │ 384 │ 550 │ 588 │ 539.8 │ 54.2 │
│ 8 │ 'nuxt3' │ 229 │ 344 │ 411 │ 335.3 │ 45.62 │
│ 9 │ 'remix' │ 264 │ 371 │ 485 │ 387.9 │ 69.16 │
│ 10 │ 'gatsby' │ 158 │ 233 │ 277 │ 239.6 │ 34.64 │
│ 11 │ 'next' │ 142 │ 206 │ 233 │ 203.7 │ 24.71 │
│ 12 │ 'angular' │ 45 │ 72 │ 76 │ 69.91 │ 8.61 │
└─────────┴───────────────────┴──────┴──────┴──────┴─────────┴─────────┘
Measure Node vs Bun vs Deno at SSR speed of a non trivial (the dashboard) React app. The below is requests per second. Larger numbers are better.
┌─────────┬───────────────────┬──────┬──────┬──────┬────── ─┬─────────┐
│ (index) │ name │ 1% │ 50% │ 99% │ Avg │ Std Dev │
├─────────┼───────────────────┼──────┼──────┼─ ────┼────────┼─────────┤
│ 0 │ 'preact-ssr-node' │ 1755 │ 2065 │ 2185 │ 2052.6 │ 115.11 │
│ 1 │ 'react-ssr-bun' │ 500 │ 669 │ 718 │ 650.8 │ 71.7 │
│ 2 │ 'react-ssr-deno' │ 550 │ 600 │ 630 │ 601 │ 20.89 │
│ 3 │ 'react-ssr-node' │ 267 │ 375 │ 394 │ 366.5 │ 35.04 │
└─────────┴───────────────────┴──────┴──────┴──────┴────────┴─────────┘
The above uses Bun version 0.1.10
, Deno version 1.25.0
, Node.js version 16.14.0
. Run on a 2.6 GHz 6-Core Intel Core i7
.
┌─────────┬────────────┬──────────────────────┐
│ (index) │ Name │ Build Time (Seconds) │
├─────────┼────────────┼──────────────────────┤
│ 0 │ 'fresh' │ 0 │
│ 1 │ 'vue3' │ 3.5 │
│ 2 │ 'react' │ 7.4 │
│ 3 │ 'svelte' │ 7.7 │
│ 4 │ 'qwik' │ 7.7 │
│ 5 │ 'next' │ 7.8 │
│ 6 │ 'astro' │ 7.9 │
│ 7 │ 'hydrogen' │ 8.4 │
│ 8 │ 'lit' │ 8.4 │
│ 9 │ 'gatsby' │ 9.9 │
│ 10 │ 'angular' │ 10.7 │
│ 11 │ 'nuxt2' │ 12.8 │
│ 12 │ 'solid' │ 12.9 │
│ 13 │ 'nuxt3' │ 16 │
│ 14 │ 'marko' │ 16.2 │
└─────────┴────────────┴──────────────────────┘
Next things we want to add:
Contributions welcome!
You will need Node.js >= 16.14.0k Deno installed locally, and Bun installed locally
After cloning this repo, install dependencies:
npm install
Now you can start running the below scripts:
Use the install script to install dependencies of each framework:
npm run fw-install
First, we must generate the component code for each framework via Mitosis.
cd apps/components
npm install
npm run build
Use the build script to build all frameworks (be sure to install
first as described above):
npm run build
To measure the weight of each framework (after you ran install
and build
):
npm run measure