vinext 0.0.4 → 0.0.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +1 -1
- package/README.md +509 -0
- package/package.json +2 -2
package/LICENSE
CHANGED
package/README.md
CHANGED
|
@@ -1 +1,510 @@
|
|
|
1
1
|
# vinext
|
|
2
|
+
|
|
3
|
+
The Next.js API surface, reimplemented on Vite.
|
|
4
|
+
|
|
5
|
+
> **Read the announcement:** [How we rebuilt Next.js with AI in one week](https://blog.cloudflare.com/vinext/)
|
|
6
|
+
|
|
7
|
+
> 🚧 **Experimental — under heavy development.** This project is an experiment in AI-driven software development. The vast majority of the code, tests, and documentation were written by AI (Claude Code). Humans direct architecture, priorities, and design decisions, but have not reviewed most of the code line-by-line. Treat this accordingly — there will be bugs, rough edges, and things that don't work. Use at your own risk.
|
|
8
|
+
|
|
9
|
+
## Quick start
|
|
10
|
+
|
|
11
|
+
vinext includes an [Agent Skill](https://agentskills.io/home) that handles migration for you. It works with Claude Code, OpenCode, Cursor, Codex, and dozens of other AI coding tools. Install it, open your Next.js project, and tell the AI to migrate:
|
|
12
|
+
|
|
13
|
+
```sh
|
|
14
|
+
npx skills add cloudflare/vinext
|
|
15
|
+
```
|
|
16
|
+
|
|
17
|
+
Then open your Next.js project in any supported tool and say:
|
|
18
|
+
|
|
19
|
+
```
|
|
20
|
+
migrate this project to vinext
|
|
21
|
+
```
|
|
22
|
+
|
|
23
|
+
The skill handles compatibility checking, dependency installation, config generation, and dev server startup. It knows what vinext supports and will flag anything that needs manual attention.
|
|
24
|
+
|
|
25
|
+
### Or do it manually
|
|
26
|
+
|
|
27
|
+
```bash
|
|
28
|
+
npm install vinext
|
|
29
|
+
```
|
|
30
|
+
|
|
31
|
+
Replace `next` with `vinext` in your scripts:
|
|
32
|
+
|
|
33
|
+
```json
|
|
34
|
+
{
|
|
35
|
+
"scripts": {
|
|
36
|
+
"dev": "vinext dev",
|
|
37
|
+
"build": "vinext build",
|
|
38
|
+
"start": "vinext start"
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
```
|
|
42
|
+
|
|
43
|
+
```bash
|
|
44
|
+
vinext dev # Development server with HMR
|
|
45
|
+
vinext build # Production build
|
|
46
|
+
vinext deploy # Build and deploy to Cloudflare Workers
|
|
47
|
+
```
|
|
48
|
+
|
|
49
|
+
vinext auto-detects your `app/` or `pages/` directory, loads `next.config.js`, and configures Vite automatically. No `vite.config.ts` required for basic usage.
|
|
50
|
+
|
|
51
|
+
Your existing `pages/`, `app/`, `next.config.js`, and `public/` directories work as-is. Run `vinext check` first to scan for known compatibility issues, or use `vinext init` to [automate the full migration](#migrating-an-existing-nextjs-project).
|
|
52
|
+
|
|
53
|
+
### CLI reference
|
|
54
|
+
|
|
55
|
+
| Command | Description |
|
|
56
|
+
|---------|-------------|
|
|
57
|
+
| `vinext dev` | Start dev server with HMR |
|
|
58
|
+
| `vinext build` | Production build (multi-environment for App Router: RSC + SSR + client) |
|
|
59
|
+
| `vinext start` | Start local production server for testing |
|
|
60
|
+
| `vinext deploy` | Build and deploy to Cloudflare Workers |
|
|
61
|
+
| `vinext init` | Migrate a Next.js project to run under vinext |
|
|
62
|
+
| `vinext check` | Scan your Next.js app for compatibility issues before migrating |
|
|
63
|
+
| `vinext lint` | Delegate to eslint or oxlint |
|
|
64
|
+
|
|
65
|
+
Options: `-p / --port <port>`, `-H / --hostname <host>`, `--turbopack` (accepted, no-op).
|
|
66
|
+
|
|
67
|
+
`vinext deploy` options: `--preview`, `--name <name>`, `--skip-build`, `--dry-run`, `--experimental-tpr`.
|
|
68
|
+
|
|
69
|
+
`vinext init` options: `--port <port>` (default: 3001), `--skip-check`, `--force`.
|
|
70
|
+
|
|
71
|
+
### Migrating an existing Next.js project
|
|
72
|
+
|
|
73
|
+
`vinext init` automates the migration in one command:
|
|
74
|
+
|
|
75
|
+
```bash
|
|
76
|
+
npx vinext init
|
|
77
|
+
```
|
|
78
|
+
|
|
79
|
+
This will:
|
|
80
|
+
|
|
81
|
+
1. Run `vinext check` to scan for compatibility issues
|
|
82
|
+
2. Install `vite` (and `@vitejs/plugin-rsc` for App Router projects) as devDependencies
|
|
83
|
+
3. Rename CJS config files (e.g. `postcss.config.js` -> `.cjs`) to avoid ESM conflicts
|
|
84
|
+
4. Add `"type": "module"` to `package.json`
|
|
85
|
+
5. Add `dev:vinext` and `build:vinext` scripts to `package.json`
|
|
86
|
+
6. Generate a minimal `vite.config.ts`
|
|
87
|
+
|
|
88
|
+
The migration is non-destructive -- your existing Next.js setup continues to work alongside vinext. It does not modify `next.config`, `tsconfig.json`, or any source files, and it does not remove Next.js dependencies.
|
|
89
|
+
|
|
90
|
+
```bash
|
|
91
|
+
npm run dev:vinext # Start the vinext dev server (port 3001)
|
|
92
|
+
npm run dev # Still runs Next.js as before
|
|
93
|
+
```
|
|
94
|
+
|
|
95
|
+
Use `--force` to overwrite an existing `vite.config.ts`, or `--skip-check` to skip the compatibility report.
|
|
96
|
+
|
|
97
|
+
## Why
|
|
98
|
+
|
|
99
|
+
Vite has become the default build tool for modern web frameworks — fast HMR, a clean plugin API, native ESM, and a growing ecosystem. With [`@vitejs/plugin-rsc`](https://github.com/vitejs/vite-plugin-rsc) adding React Server Components support, it's now possible to build a full RSC framework on Vite.
|
|
100
|
+
|
|
101
|
+
vinext is an experiment: can we reimplement the Next.js API surface on Vite, so that existing Next.js applications can run on a completely different toolchain? The answer, so far, is mostly yes — about 94% of the API surface works.
|
|
102
|
+
|
|
103
|
+
The current deployment target is Cloudflare Workers — zero cold starts, global by default, integrated platform (KV, R2, D1, AI). The `vinext deploy` command handles the full build-and-deploy pipeline. Expanding to other deployment targets is something we'd like to explore.
|
|
104
|
+
|
|
105
|
+
**Alternatives worth knowing about:**
|
|
106
|
+
- **[OpenNext](https://opennext.js.org/)** — adapts `next build` output for AWS, Cloudflare, and other platforms. More mature and battle-tested than vinext.
|
|
107
|
+
- **[Next.js self-hosting](https://nextjs.org/docs/app/building-your-application/deploying#self-hosting)** — Next.js can be deployed to any Node.js server, Docker container, or as a static export.
|
|
108
|
+
|
|
109
|
+
### Design principles
|
|
110
|
+
|
|
111
|
+
- **Start with Cloudflare, expand later.** Workers is the current deployment target. Every feature is built and tested for Workers. We're interested in supporting other platforms and welcome contributions.
|
|
112
|
+
- **Pragmatic compatibility, not bug-for-bug parity.** Targets 95%+ of real-world Next.js apps. Edge cases that depend on undocumented Vercel behavior are intentionally not supported.
|
|
113
|
+
- **Latest Next.js only.** Targets Next.js 16.x. No support for deprecated APIs from older versions.
|
|
114
|
+
- **Incremental adoption.** Drop in the plugin, fix what breaks, deploy.
|
|
115
|
+
|
|
116
|
+
## FAQ
|
|
117
|
+
|
|
118
|
+
**What is this?**
|
|
119
|
+
vinext is a Vite plugin that reimplements the public Next.js API — routing, server rendering, `next/*` module imports, the CLI — so you can run Next.js applications on Vite instead of the Next.js compiler toolchain. Cloudflare Workers is the current deployment target.
|
|
120
|
+
|
|
121
|
+
**Is this a fork of Next.js?**
|
|
122
|
+
No. vinext is an alternative implementation of the Next.js API surface built on Vite. It does import some Next.js types and utilities, but the core is written from scratch. The goal is not to create a competing framework or add features beyond what Next.js offers — it's an experiment in how far AI-driven development and Vite's toolchain can go in replicating an existing, well-defined API surface.
|
|
123
|
+
|
|
124
|
+
**How is this different from OpenNext?**
|
|
125
|
+
[OpenNext](https://opennext.js.org/) adapts the *output* of a standard `next build` to run on various platforms. vinext replaces the build entirely — it reimplements the Next.js APIs on Vite from scratch. OpenNext is a great choice if you need production-ready Next.js on non-Vercel platforms today.
|
|
126
|
+
|
|
127
|
+
**Can I use this in production?**
|
|
128
|
+
You can, with caution. This is experimental software with known bugs. It works well enough for demos and exploration, but it hasn't been battle-tested with real production traffic.
|
|
129
|
+
|
|
130
|
+
**Can I just self-host Next.js?**
|
|
131
|
+
Yes. Next.js supports [self-hosting](https://nextjs.org/docs/app/building-your-application/deploying#self-hosting) on Node.js servers, Docker containers, and static exports. If you're happy with the Next.js toolchain and just want to run it somewhere other than Vercel, self-hosting is the simplest path.
|
|
132
|
+
|
|
133
|
+
**How are you verifying this works?**
|
|
134
|
+
The test suite has over 1,700 Vitest tests and 380 Playwright E2E tests. This includes tests ported directly from the [Next.js test suite](https://github.com/vercel/next.js/tree/canary/test) and [OpenNext's Cloudflare conformance suite](https://github.com/opennextjs/opennextjs-cloudflare), covering routing, SSR, RSC, server actions, caching, metadata, middleware, streaming, and more. Vercel's [App Router Playground](https://github.com/vercel/next-app-router-playground) also runs on vinext as an integration test. See the [Tests](#tests) section and `tests/nextjs-compat/TRACKING.md` for details.
|
|
135
|
+
|
|
136
|
+
**Who is reviewing this code?**
|
|
137
|
+
Mostly nobody. This is an experiment in seeing how far AI-driven development can go. The test suite is the primary quality gate — not human code review. Contributions and code review are welcome.
|
|
138
|
+
|
|
139
|
+
**Why Vite?**
|
|
140
|
+
Vite is an excellent build tool with a rich plugin ecosystem, first-class ESM support, and fast HMR. The [`@vitejs/plugin-rsc`](https://github.com/vitejs/vite-plugin-rsc) plugin adds React Server Components support with multi-environment builds. This project is an experiment to see how much of the Next.js developer experience can be replicated on top of Vite's infrastructure.
|
|
141
|
+
|
|
142
|
+
**Does this support the Pages Router, App Router, or both?**
|
|
143
|
+
Both. File-system routing, SSR, client hydration, and deployment to Cloudflare Workers work for both routers.
|
|
144
|
+
|
|
145
|
+
**What version of Next.js does this target?**
|
|
146
|
+
Next.js 16.x. No support for deprecated APIs from older versions.
|
|
147
|
+
|
|
148
|
+
**Can I deploy to AWS/Netlify/other platforms?**
|
|
149
|
+
Currently only Cloudflare Workers is supported and tested. We're interested in exploring other deployment targets in the future and welcome contributions in that direction.
|
|
150
|
+
|
|
151
|
+
**What happens when Next.js releases a new feature?**
|
|
152
|
+
We track the public Next.js API surface and add support for new stable features. Experimental or unstable Next.js features are lower priority. The plan is to add commit-level tracking of the Next.js repo so we can stay current as new versions are released.
|
|
153
|
+
|
|
154
|
+
## Deploying to Cloudflare Workers
|
|
155
|
+
|
|
156
|
+
`vinext deploy` is the simplest path. It auto-generates the necessary configuration files (`vite.config.ts`, `wrangler.jsonc`, `worker/index.ts`) if they don't exist, builds the application, and deploys to Workers.
|
|
157
|
+
|
|
158
|
+
```bash
|
|
159
|
+
vinext deploy
|
|
160
|
+
```
|
|
161
|
+
|
|
162
|
+
The deploy command also auto-detects and fixes common migration issues:
|
|
163
|
+
- Adds `"type": "module"` to package.json if missing
|
|
164
|
+
- Resolves tsconfig.json path aliases automatically (via `vite-tsconfig-paths`)
|
|
165
|
+
- Detects MDX usage and configures `@mdx-js/rollup`
|
|
166
|
+
- Renames CJS config files (postcss.config.js, etc.) to `.cjs` when needed
|
|
167
|
+
- Detects native Node.js modules (sharp, resvg, satori, lightningcss, @napi-rs/canvas) and auto-stubs them for Workers. If you encounter others that need stubbing, PRs are welcome.
|
|
168
|
+
|
|
169
|
+
Both App Router and Pages Router work on Workers with full client-side hydration — interactive components, client-side navigation, and React state all work.
|
|
170
|
+
|
|
171
|
+
### Traffic-aware Pre-Rendering (experimental)
|
|
172
|
+
|
|
173
|
+
TPR queries Cloudflare zone analytics at deploy time to find which pages actually get traffic, pre-renders only those, and uploads them to KV cache. The result is SSG-level latency for popular pages without pre-rendering your entire site.
|
|
174
|
+
|
|
175
|
+
```bash
|
|
176
|
+
vinext deploy --experimental-tpr # Pre-render pages covering 90% of traffic
|
|
177
|
+
vinext deploy --experimental-tpr --tpr-coverage 95 # More aggressive coverage
|
|
178
|
+
vinext deploy --experimental-tpr --tpr-limit 500 # Cap at 500 pages
|
|
179
|
+
vinext deploy --experimental-tpr --tpr-window 48 # Use 48h of analytics
|
|
180
|
+
```
|
|
181
|
+
|
|
182
|
+
Requires a custom domain (zone analytics are unavailable on `*.workers.dev`) and `CLOUDFLARE_API_TOKEN` with Zone.Analytics read permission.
|
|
183
|
+
|
|
184
|
+
For production caching (ISR), use the built-in Cloudflare KV cache handler:
|
|
185
|
+
|
|
186
|
+
```ts
|
|
187
|
+
import { KVCacheHandler } from "vinext/cloudflare";
|
|
188
|
+
import { setCacheHandler } from "next/cache";
|
|
189
|
+
|
|
190
|
+
setCacheHandler(new KVCacheHandler(env.MY_KV_NAMESPACE));
|
|
191
|
+
```
|
|
192
|
+
|
|
193
|
+
### Custom Vite configuration
|
|
194
|
+
|
|
195
|
+
If you need to customize the Vite config, create a `vite.config.ts`. vinext will merge its config with yours. This is required for Cloudflare Workers deployment with the App Router (RSC needs explicit plugin configuration):
|
|
196
|
+
|
|
197
|
+
```ts
|
|
198
|
+
import { defineConfig } from "vite";
|
|
199
|
+
import vinext from "vinext";
|
|
200
|
+
import rsc from "@vitejs/plugin-rsc";
|
|
201
|
+
import { cloudflare } from "@cloudflare/vite-plugin";
|
|
202
|
+
|
|
203
|
+
export default defineConfig({
|
|
204
|
+
plugins: [
|
|
205
|
+
vinext(),
|
|
206
|
+
rsc({
|
|
207
|
+
entries: {
|
|
208
|
+
rsc: "virtual:vinext-rsc-entry",
|
|
209
|
+
ssr: "virtual:vinext-app-ssr-entry",
|
|
210
|
+
client: "virtual:vinext-app-browser-entry",
|
|
211
|
+
},
|
|
212
|
+
}),
|
|
213
|
+
cloudflare({
|
|
214
|
+
viteEnvironment: { name: "rsc", childEnvironments: ["ssr"] },
|
|
215
|
+
}),
|
|
216
|
+
],
|
|
217
|
+
});
|
|
218
|
+
```
|
|
219
|
+
|
|
220
|
+
See the [examples](#live-examples) for complete working configurations.
|
|
221
|
+
|
|
222
|
+
## Live examples
|
|
223
|
+
|
|
224
|
+
These are deployed to Cloudflare Workers and updated on every push to `main`:
|
|
225
|
+
|
|
226
|
+
| Example | Description | URL |
|
|
227
|
+
|---------|-------------|-----|
|
|
228
|
+
| App Router Playground | [Vercel's Next.js App Router Playground](https://github.com/vercel/next-app-router-playground) running on vinext | [app-router-playground.vinext.workers.dev](https://app-router-playground.vinext.workers.dev) |
|
|
229
|
+
| Hacker News | HN clone (App Router, RSC) | [hackernews.vinext.workers.dev](https://hackernews.vinext.workers.dev) |
|
|
230
|
+
| Nextra Docs | Nextra docs site (MDX, App Router) | [nextra-docs-template.vinext.workers.dev](https://nextra-docs-template.vinext.workers.dev) |
|
|
231
|
+
| App Router (minimal) | Minimal App Router on Workers | [app-router-cloudflare.vinext.workers.dev](https://app-router-cloudflare.vinext.workers.dev) |
|
|
232
|
+
| Pages Router (minimal) | Minimal Pages Router on Workers | [pages-router-cloudflare.vinext.workers.dev](https://pages-router-cloudflare.vinext.workers.dev) |
|
|
233
|
+
| RealWorld API | REST API routes example | [realworld-api-rest.vinext.workers.dev](https://realworld-api-rest.vinext.workers.dev) |
|
|
234
|
+
| Benchmarks Dashboard | Build performance tracking over time (D1-backed) | [benchmarks.vinext.workers.dev](https://benchmarks.vinext.workers.dev) |
|
|
235
|
+
|
|
236
|
+
## API coverage
|
|
237
|
+
|
|
238
|
+
~94% of the Next.js 16 API surface has full or partial support. The remaining gaps are intentional stubs for deprecated features and Partial Prerendering (which Next.js 16 reworked into `"use cache"` — that directive is fully supported).
|
|
239
|
+
|
|
240
|
+
> ✅ = full implementation | 🟡 = partial (runtime behavior correct, some build-time optimizations missing) | ⬜ = intentional stub/no-op
|
|
241
|
+
|
|
242
|
+
### Module shims
|
|
243
|
+
|
|
244
|
+
Every `next/*` import is shimmed to a Vite-compatible implementation.
|
|
245
|
+
|
|
246
|
+
| Module | | Notes |
|
|
247
|
+
|--------|---|-------|
|
|
248
|
+
| `next/link` | ✅ | All props including `prefetch` (IntersectionObserver), `onNavigate`, scroll restoration, `basePath`, `locale` |
|
|
249
|
+
| `next/image` | 🟡 | Remote images via [@unpic/react](https://unpic.pics) (28 CDNs). Local images via `<img>` + srcSet. No build-time optimization/resizing |
|
|
250
|
+
| `next/head` | ✅ | SSR collection + client-side DOM manipulation |
|
|
251
|
+
| `next/router` | ✅ | `useRouter`, `Router` singleton, events, client-side navigation, SSR context, i18n |
|
|
252
|
+
| `next/navigation` | ✅ | `usePathname`, `useSearchParams`, `useParams`, `useRouter`, `redirect`, `notFound`, `forbidden`, `unauthorized` |
|
|
253
|
+
| `next/server` | ✅ | `NextRequest`, `NextResponse`, `NextURL`, cookies, `userAgent`, `after`, `connection`, `URLPattern` |
|
|
254
|
+
| `next/headers` | ✅ | Async `headers()`, `cookies()`, `draftMode()` |
|
|
255
|
+
| `next/dynamic` | ✅ | `ssr: true`, `ssr: false`, `loading` component |
|
|
256
|
+
| `next/script` | ✅ | All 4 strategies (`beforeInteractive`, `afterInteractive`, `lazyOnload`, `worker`) |
|
|
257
|
+
| `next/font/google` | 🟡 | Runtime CDN loading. No self-hosting, font subsetting, or fallback metrics |
|
|
258
|
+
| `next/font/local` | 🟡 | Runtime `@font-face` injection. Not extracted at build time |
|
|
259
|
+
| `next/og` | ✅ | OG image generation via `@vercel/og` (Satori + resvg) |
|
|
260
|
+
| `next/cache` | ✅ | `revalidateTag`, `revalidatePath`, `unstable_cache`, pluggable `CacheHandler`, `"use cache"` with `cacheLife()` and `cacheTag()` |
|
|
261
|
+
| `next/form` | ✅ | GET form interception + POST server action delegation |
|
|
262
|
+
| `next/legacy/image` | ✅ | Translates legacy props to modern Image |
|
|
263
|
+
| `next/error` | ✅ | Default error page component |
|
|
264
|
+
| `next/config` | ✅ | `getConfig` / `setConfig` |
|
|
265
|
+
| `next/document` | ✅ | `Html`, `Head`, `Main`, `NextScript` |
|
|
266
|
+
| `next/constants` | ✅ | All phase constants |
|
|
267
|
+
| `next/amp` | ⬜ | No-op (AMP is deprecated) |
|
|
268
|
+
| `next/web-vitals` | ⬜ | No-op (use the `web-vitals` library directly) |
|
|
269
|
+
|
|
270
|
+
### Routing
|
|
271
|
+
|
|
272
|
+
| Feature | | Notes |
|
|
273
|
+
|---------|---|-------|
|
|
274
|
+
| File-system routing (`pages/`) | ✅ | Automatic scanning with hot-reload on file changes |
|
|
275
|
+
| File-system routing (`app/`) | ✅ | Pages, routes, layouts, templates, loading, error, not-found, forbidden, unauthorized |
|
|
276
|
+
| Dynamic routes `[param]` | ✅ | Both routers |
|
|
277
|
+
| Catch-all `[...slug]` | ✅ | Both routers |
|
|
278
|
+
| Optional catch-all `[[...slug]]` | ✅ | Both routers |
|
|
279
|
+
| Route groups `(group)` | ✅ | URL-transparent, layouts still apply |
|
|
280
|
+
| Parallel routes `@slot` | ✅ | Discovery, layout props, `default.tsx`, inherited slots |
|
|
281
|
+
| Intercepting routes | ✅ | `(.)`, `(..)`, `(..)(..)`, `(...)` conventions |
|
|
282
|
+
| Route handlers (`route.ts`) | ✅ | Named HTTP methods, auto OPTIONS/HEAD, cookie attachment |
|
|
283
|
+
| Middleware | ✅ | `middleware.ts` and `proxy.ts` (Next.js 16). Matcher patterns (string, array, regex, `:param`, `:path*`, `:path+`) |
|
|
284
|
+
| i18n routing | 🟡 | Pages Router locale prefix, Accept-Language detection, NEXT_LOCALE cookie. No domain-based routing |
|
|
285
|
+
| `basePath` | ✅ | Applied everywhere — URLs, Link, Router, navigation hooks |
|
|
286
|
+
| `trailingSlash` | ✅ | 308 redirects to canonical form |
|
|
287
|
+
|
|
288
|
+
### Server features
|
|
289
|
+
|
|
290
|
+
| Feature | | Notes |
|
|
291
|
+
|---------|---|-------|
|
|
292
|
+
| SSR (Pages Router) | ✅ | Streaming, `_app`/`_document`, `__NEXT_DATA__`, hydration |
|
|
293
|
+
| SSR (App Router) | ✅ | RSC pipeline, nested layouts, streaming, nav context for client components |
|
|
294
|
+
| `getStaticProps` | ✅ | Props, redirect, notFound, revalidate |
|
|
295
|
+
| `getStaticPaths` | ✅ | `fallback: false`, `true`, `"blocking"` |
|
|
296
|
+
| `getServerSideProps` | ✅ | Full context including locale |
|
|
297
|
+
| ISR | ✅ | Stale-while-revalidate, pluggable `CacheHandler`, background regeneration |
|
|
298
|
+
| Server Actions (`"use server"`) | ✅ | Action execution, FormData, re-render after mutation, `redirect()` in actions |
|
|
299
|
+
| React Server Components | ✅ | Via `@vitejs/plugin-rsc`. `"use client"` boundaries work correctly |
|
|
300
|
+
| Streaming SSR | ✅ | Both routers |
|
|
301
|
+
| Metadata API | ✅ | `metadata`, `generateMetadata`, `viewport`, `generateViewport`, title templates |
|
|
302
|
+
| `generateStaticParams` | ✅ | With `dynamicParams` enforcement |
|
|
303
|
+
| Metadata file routes | ✅ | sitemap.xml, robots.txt, manifest, favicon, OG images (static + dynamic) |
|
|
304
|
+
| Static export (`output: 'export'`) | ✅ | Generates static HTML/JSON for all routes |
|
|
305
|
+
| `connection()` | ✅ | Forces dynamic rendering |
|
|
306
|
+
| `"use cache"` directive | ✅ | File-level and function-level. `cacheLife()` profiles, `cacheTag()`, stale-while-revalidate |
|
|
307
|
+
| `instrumentation.ts` | ✅ | `register()` and `onRequestError()` callbacks |
|
|
308
|
+
| Route segment config | 🟡 | `revalidate`, `dynamic`, `dynamicParams`. `runtime` and `preferredRegion` are ignored |
|
|
309
|
+
|
|
310
|
+
### Configuration
|
|
311
|
+
|
|
312
|
+
| Feature | | Notes |
|
|
313
|
+
|---------|---|-------|
|
|
314
|
+
| `next.config.js` / `.ts` / `.mjs` | ✅ | Function configs, phase argument |
|
|
315
|
+
| `rewrites` / `redirects` / `headers` | ✅ | All phases, param interpolation |
|
|
316
|
+
| Environment variables (`NEXT_PUBLIC_*`) | ✅ | Inlined at build time via Vite |
|
|
317
|
+
| `images` config | 🟡 | Parsed but not used for optimization |
|
|
318
|
+
|
|
319
|
+
### Caching
|
|
320
|
+
|
|
321
|
+
The cache is pluggable. The default `MemoryCacheHandler` works out of the box. Swap in your own backend for production:
|
|
322
|
+
|
|
323
|
+
```ts
|
|
324
|
+
import { setCacheHandler } from "next/cache";
|
|
325
|
+
setCacheHandler(new MyCacheHandler()); // Redis, DynamoDB, etc.
|
|
326
|
+
```
|
|
327
|
+
|
|
328
|
+
The `CacheHandler` interface matches Next.js 16's shape, so community adapters should be compatible.
|
|
329
|
+
|
|
330
|
+
## What's NOT supported (and won't be)
|
|
331
|
+
|
|
332
|
+
These are intentional exclusions:
|
|
333
|
+
|
|
334
|
+
- **Vercel-specific features** — `@vercel/og` edge runtime, Vercel Analytics integration, Vercel KV/Blob/Postgres bindings. Use platform equivalents.
|
|
335
|
+
- **AMP** — Deprecated since Next.js 13. `useAmp()` returns `false`.
|
|
336
|
+
- **`next export` (legacy)** — Use `output: 'export'` in config instead.
|
|
337
|
+
- **Turbopack/webpack configuration** — This runs on Vite. Use Vite plugins instead of webpack loaders/plugins.
|
|
338
|
+
- **`next/jest`** — Use Vitest.
|
|
339
|
+
- **`create-next-app` scaffolding** — Not a goal.
|
|
340
|
+
- **Bug-for-bug parity with undocumented behavior** — If it's not in the Next.js docs, we probably don't replicate it.
|
|
341
|
+
|
|
342
|
+
## Known limitations
|
|
343
|
+
|
|
344
|
+
- **Image optimization doesn't happen at build time.** Remote images work via `@unpic/react` (auto-detects 28 CDN providers). Local images are routed through a `/_vinext/image` endpoint that can resize and transcode on Cloudflare Workers (via the Images binding) in production, but no build-time optimization or static resizing occurs.
|
|
345
|
+
- **Google Fonts are loaded from the CDN, not self-hosted.** No `size-adjust` fallback font metrics. Local fonts work but `@font-face` CSS is injected at runtime, not extracted at build time.
|
|
346
|
+
- **`useSelectedLayoutSegment(s)`** derives segments from the pathname rather than being truly layout-aware. May differ from Next.js in edge cases with parallel routes.
|
|
347
|
+
- **Route segment config** — `runtime` and `preferredRegion` are ignored (everything runs in the same environment).
|
|
348
|
+
- **Node.js production server (`vinext start`)** works for testing but is less complete than Workers deployment. Cloudflare Workers is the primary target.
|
|
349
|
+
- **Native Node modules (sharp, resvg, satori, lightningcss, @napi-rs/canvas)** crash Vite's RSC dev environment. Dynamic OG image/icon routes using these work in production builds but not in dev mode. These are auto-stubbed during `vinext deploy`.
|
|
350
|
+
|
|
351
|
+
## Benchmarks
|
|
352
|
+
|
|
353
|
+
> **Caveat:** Benchmarks are hard to get right and these are early results. Take them as directional, not definitive.
|
|
354
|
+
|
|
355
|
+
These benchmarks measure **compilation and bundling speed**, not production serving performance. Next.js and vinext have fundamentally different default approaches: Next.js statically pre-renders pages at build time (making builds slower but production serving faster for static content), while vinext server-renders all pages on each request. To make the comparison apples-to-apples, the benchmark app uses `export const dynamic = "force-dynamic"` to disable Next.js static pre-rendering — both frameworks are doing the same work: compiling, bundling, and preparing server-rendered routes.
|
|
356
|
+
|
|
357
|
+
The benchmark app is a shared 33-route App Router application (server components, client components, dynamic routes, nested layouts, API routes) built identically by both tools. We compare Next.js 16 (Turbopack) against vinext on both Vite 7 (Rollup) and Vite 8 (Rolldown). Turbopack and Rolldown both parallelize across cores, so results on machines with more cores may differ significantly.
|
|
358
|
+
|
|
359
|
+
We measure three things:
|
|
360
|
+
|
|
361
|
+
- **Production build time** — 5 runs, timed with `hyperfine`.
|
|
362
|
+
- **Client bundle size** — gzipped output of each build.
|
|
363
|
+
- **Dev server cold start** — 10 runs, randomized execution order. Vite's dependency optimizer cache is cleared before each run.
|
|
364
|
+
|
|
365
|
+
Benchmarks run on GitHub CI runners (2-core Ubuntu) on every merge to `main`. See the launch numbers in the [announcement blog post](https://blog.cloudflare.com/vinext/) and the latest results at **[benchmarks.vinext.workers.dev](https://benchmarks.vinext.workers.dev)**.
|
|
366
|
+
|
|
367
|
+
<details>
|
|
368
|
+
<summary>Why the bundle size difference?</summary>
|
|
369
|
+
|
|
370
|
+
Analysis of the build output shows two main factors:
|
|
371
|
+
|
|
372
|
+
1. **Tree-shaking**: Vite/Rollup produces a smaller React+ReactDOM bundle than Next.js/Turbopack. Rollup's more aggressive dead-code elimination accounts for roughly half the overall difference.
|
|
373
|
+
2. **Framework overhead**: Next.js ships more client-side infrastructure (router, Turbopack runtime loader, prefetching, error handling) than vinext's lighter client runtime.
|
|
374
|
+
|
|
375
|
+
Both frameworks ship the same app code and the same RSC client runtime (`react-server-dom-webpack`). The difference is in how much of React's internals survive tree-shaking and how much framework plumbing each tool adds.
|
|
376
|
+
|
|
377
|
+
</details>
|
|
378
|
+
|
|
379
|
+
Reproduce with `node benchmarks/run.mjs --runs=5 --dev-runs=10`. Exact framework versions are recorded in each result.
|
|
380
|
+
|
|
381
|
+
## Architecture
|
|
382
|
+
|
|
383
|
+
vinext is a Vite plugin that:
|
|
384
|
+
|
|
385
|
+
1. **Resolves all `next/*` imports** to local shim modules that reimplement the Next.js API using standard Web APIs and React primitives.
|
|
386
|
+
2. **Scans your `pages/` and `app/` directories** to build a file-system router matching Next.js conventions.
|
|
387
|
+
3. **Generates virtual entry modules** for the RSC, SSR, and browser environments that handle request routing, component rendering, and client hydration.
|
|
388
|
+
4. **Integrates with `@vitejs/plugin-rsc`** for React Server Components — handling `"use client"` / `"use server"` directives, RSC stream serialization, and multi-environment builds.
|
|
389
|
+
|
|
390
|
+
The result is a standard Vite application that happens to be API-compatible with Next.js.
|
|
391
|
+
|
|
392
|
+
### Pages Router flow
|
|
393
|
+
|
|
394
|
+
```
|
|
395
|
+
Request → Vite dev server middleware → Route match → getServerSideProps/getStaticProps
|
|
396
|
+
→ renderToReadableStream(App + Page) → HTML with __NEXT_DATA__ → Client hydration
|
|
397
|
+
```
|
|
398
|
+
|
|
399
|
+
### App Router flow
|
|
400
|
+
|
|
401
|
+
```
|
|
402
|
+
Request → RSC entry (Vite rsc environment) → Route match → Build layout/page tree
|
|
403
|
+
→ renderToReadableStream (RSC payload) → SSR entry (Vite ssr environment)
|
|
404
|
+
→ renderToReadableStream (HTML) → Client hydration from RSC stream
|
|
405
|
+
```
|
|
406
|
+
|
|
407
|
+
## Project structure
|
|
408
|
+
|
|
409
|
+
```
|
|
410
|
+
packages/vinext/
|
|
411
|
+
src/
|
|
412
|
+
index.ts # Main plugin — resolve aliases, config, virtual modules
|
|
413
|
+
cli.ts # vinext CLI (dev/build/start/deploy/init/check/lint)
|
|
414
|
+
check.ts # Compatibility scanner
|
|
415
|
+
deploy.ts # Cloudflare Workers deployment
|
|
416
|
+
init.ts # vinext init — one-command migration for Next.js apps
|
|
417
|
+
client/
|
|
418
|
+
entry.ts # Client-side hydration entry
|
|
419
|
+
routing/
|
|
420
|
+
pages-router.ts # Pages Router file-system scanner
|
|
421
|
+
app-router.ts # App Router file-system scanner
|
|
422
|
+
server/
|
|
423
|
+
dev-server.ts # Pages Router SSR request handler
|
|
424
|
+
app-dev-server.ts # App Router RSC entry generator
|
|
425
|
+
prod-server.ts # Production server with compression
|
|
426
|
+
api-handler.ts # Pages Router API routes
|
|
427
|
+
isr-cache.ts # ISR cache layer
|
|
428
|
+
middleware.ts # middleware.ts / proxy.ts runner
|
|
429
|
+
metadata-routes.ts # File-based metadata route scanner
|
|
430
|
+
instrumentation.ts # instrumentation.ts support
|
|
431
|
+
cloudflare/
|
|
432
|
+
kv-cache-handler.ts # Cloudflare KV-backed CacheHandler for ISR
|
|
433
|
+
shims/ # One file per next/* module (33 shims + 6 internal)
|
|
434
|
+
build/
|
|
435
|
+
static-export.ts # output: 'export' support
|
|
436
|
+
utils/
|
|
437
|
+
project.ts # Shared project utilities (ESM, CJS, package manager detection)
|
|
438
|
+
config/
|
|
439
|
+
next-config.ts # next.config.js loader
|
|
440
|
+
config-matchers.ts # Config matching utilities
|
|
441
|
+
|
|
442
|
+
tests/
|
|
443
|
+
*.test.ts # Vitest unit + integration tests
|
|
444
|
+
nextjs-compat/ # Tests ported from Next.js test suite
|
|
445
|
+
fixtures/ # Test apps (pages-basic, app-basic, ecosystem libs)
|
|
446
|
+
e2e/ # Playwright E2E tests (5 projects)
|
|
447
|
+
|
|
448
|
+
examples/ # Deployed demo apps (see Live Examples above)
|
|
449
|
+
```
|
|
450
|
+
|
|
451
|
+
## Tests
|
|
452
|
+
|
|
453
|
+
```bash
|
|
454
|
+
pnpm test # Vitest unit + integration tests
|
|
455
|
+
pnpm run test:e2e # Playwright E2E tests (5 projects)
|
|
456
|
+
pnpm run typecheck # TypeScript checking (tsgo)
|
|
457
|
+
pnpm run lint # Linting (oxlint)
|
|
458
|
+
```
|
|
459
|
+
|
|
460
|
+
E2E tests cover Pages Router (dev + production), App Router (dev), and both routers on Cloudflare Workers via `wrangler dev`.
|
|
461
|
+
|
|
462
|
+
The [Vercel App Router Playground](https://github.com/vercel/next-app-router-playground) runs on vinext as an integration test — see it live at [app-router-playground.vinext.workers.dev](https://app-router-playground.vinext.workers.dev).
|
|
463
|
+
|
|
464
|
+
## Local setup (from source)
|
|
465
|
+
|
|
466
|
+
If you're working from the repo instead of installing from npm:
|
|
467
|
+
|
|
468
|
+
```bash
|
|
469
|
+
git clone https://github.com/cloudflare/vinext.git
|
|
470
|
+
cd vinext
|
|
471
|
+
pnpm install
|
|
472
|
+
pnpm run build
|
|
473
|
+
```
|
|
474
|
+
|
|
475
|
+
This compiles the vinext package to `packages/vinext/dist/`. For active development, use `pnpm --filter vinext run dev` to run `tsc --watch`.
|
|
476
|
+
|
|
477
|
+
To use it against an external Next.js app, link the built package:
|
|
478
|
+
|
|
479
|
+
```bash
|
|
480
|
+
# From your Next.js project directory:
|
|
481
|
+
pnpm link /path/to/vinext/packages/vinext
|
|
482
|
+
```
|
|
483
|
+
|
|
484
|
+
Or add it to your `package.json` as a file dependency:
|
|
485
|
+
|
|
486
|
+
```json
|
|
487
|
+
{
|
|
488
|
+
"dependencies": {
|
|
489
|
+
"vinext": "file:/path/to/vinext/packages/vinext"
|
|
490
|
+
}
|
|
491
|
+
}
|
|
492
|
+
```
|
|
493
|
+
|
|
494
|
+
vinext has peer dependencies on `react ^19.2.4`, `react-dom ^19.2.4`, and `vite ^7.0.0`. Then replace `next` with `vinext` in your scripts and run as normal.
|
|
495
|
+
|
|
496
|
+
## Contributing
|
|
497
|
+
|
|
498
|
+
This project is experimental and under active development. Issues and PRs are welcome.
|
|
499
|
+
|
|
500
|
+
### Reporting bugs
|
|
501
|
+
|
|
502
|
+
If something doesn't work with your Next.js app, please file an issue — we want to hear about it.
|
|
503
|
+
|
|
504
|
+
Before you do, try pointing an AI agent at the problem. Open your project with Claude Code, Cursor, OpenCode, or whatever you use, and ask it to figure out why your app isn't working with vinext. In our experience, agents are very good at tracing through the vinext source, identifying the gap or bug, and often producing a fix or at least a clear diagnosis. An issue that includes "here's what the agent found" is significantly more actionable than "it doesn't work."
|
|
505
|
+
|
|
506
|
+
Even a partial diagnosis helps — stack traces, which `next/*` import is involved, whether it's a dev or production build issue, App Router vs Pages Router. The more context, the faster we can fix it.
|
|
507
|
+
|
|
508
|
+
## License
|
|
509
|
+
|
|
510
|
+
MIT
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "vinext",
|
|
3
|
-
"version": "0.0.
|
|
3
|
+
"version": "0.0.5",
|
|
4
4
|
"description": "Run Next.js apps on Vite. Drop-in replacement for the next CLI.",
|
|
5
5
|
"license": "MIT",
|
|
6
6
|
"repository": {
|
|
@@ -44,7 +44,7 @@
|
|
|
44
44
|
],
|
|
45
45
|
"scripts": {
|
|
46
46
|
"build": "tsc",
|
|
47
|
-
"prepack": "pnpm run build",
|
|
47
|
+
"prepack": "cp ../../README.md README.md && pnpm run build",
|
|
48
48
|
"dev": "tsc --watch"
|
|
49
49
|
},
|
|
50
50
|
"peerDependencies": {
|