--- name: migrate-to-vinext description: Migrates Next.js projects to vinext (Vite-based Next.js reimplementation). Load when asked to migrate, convert, or switch from Next.js to vinext. Handles compatibility scanning, package replacement, Vite config generation, ESM conversion, and deployment setup (Cloudflare Workers natively, other platforms via Nitro). --- # Migrate Next.js to vinext vinext reimplements the Next.js API surface on Vite. Existing `app/`, `pages/`, and `next.config.js` work as-is — migration is a package swap, config generation, and ESM conversion. No changes to application code required. ## FIRST: Verify Next.js Project Confirm `next` is in `dependencies` or `devDependencies` in `package.json`. If not found, STOP — this skill does not apply. Detect the package manager from the lockfile: | Lockfile | Manager | Install | Uninstall | | --------------------------- | ------- | ------------- | --------------- | | `pnpm-lock.yaml` | pnpm | `pnpm add` | `pnpm remove` | | `yarn.lock` | yarn | `yarn add` | `yarn remove` | | `bun.lockb` / `bun.lock` | bun | `bun add` | `bun remove` | | `package-lock.json` or none | npm | `npm install` | `npm uninstall` | Detect the router: if an `app/` directory exists at root or under `src/`, it's App Router. If only `pages/` exists, it's Pages Router. Both can coexist. ## Quick Reference | Command | Purpose | | --------------- | ---------------------------------------------------------------------- | | `vinext check` | Scan project for compatibility issues, produce scored report | | `vinext init` | Automated migration — installs deps, generates config, converts to ESM | | `vinext dev` | Development server with HMR | | `vinext build` | Production build (multi-environment for App Router) | | `vinext start` | Local production server | | `vinext deploy` | Build and deploy to Cloudflare Workers | ## Phase 1: Check Compatibility Run `vinext check` (install vinext first if needed via `npx vinext check`). Review the scored report. If critical incompatibilities exist, inform the user before proceeding. See [references/compatibility.md](references/compatibility.md) for supported/unsupported features and ecosystem library status. ## Phase 2: Automated Migration (Recommended) Run `vinext init`. This command: 1. Runs `vinext check` for a compatibility report 2. Installs `vite` as a devDependency (and `@vitejs/plugin-rsc` for App Router) 3. Adds `"type": "module"` to package.json 4. Renames CJS config files (e.g., `postcss.config.js` → `.cjs`) to avoid ESM conflicts 5. Adds `dev:vinext` and `build:vinext` scripts to package.json 6. Generates a minimal `vite.config.ts` This is non-destructive — the existing Next.js setup continues to work alongside vinext. Use the `dev:vinext` script to test before fully switching over. If `vinext init` succeeds, skip to Phase 4 (Verify). If it fails or the user prefers manual control, continue to Phase 3. ## Phase 3: Manual Migration Use this as a fallback when `vinext init` doesn't work or the user wants full control. ### 3a. Replace packages ```bash # Example with npm: npm uninstall next npm install vinext npm install -D vite # App Router only: npm install -D @vitejs/plugin-rsc ``` ### 3b. Update scripts Replace all `next` commands in `package.json` scripts: | Before | After | Notes | | ------------ | -------------- | -------------------------- | | `next dev` | `vinext dev` | Dev server with HMR | | `next build` | `vinext build` | Production build | | `next start` | `vinext start` | Local production server | | `next lint` | `vinext lint` | Delegates to eslint/oxlint | Preserve flags: `next dev --port 3001` → `vinext dev --port 3001`. ### 3c. Convert to ESM Add `"type": "module"` to package.json. Rename any CJS config files: - `postcss.config.js` → `postcss.config.cjs` - `tailwind.config.js` → `tailwind.config.cjs` - Any other `.js` config that uses `module.exports` ### 3d. Generate vite.config.ts See [references/config-examples.md](references/config-examples.md) for config variants per router and deployment target. If the project already has custom Vite config, prefer Vite 8-native keys when editing it: `oxc`, `optimizeDeps.rolldownOptions`, and `build.rolldownOptions`. Older `esbuild` and `build.rollupOptions` settings still work for now but are migration targets. **Pages Router (minimal):** ```ts import vinext from "vinext"; import { defineConfig } from "vite"; export default defineConfig({ plugins: [vinext()] }); ``` **App Router (minimal):** ```ts import vinext from "vinext"; import { defineConfig } from "vite"; export default defineConfig({ plugins: [vinext()] }); ``` vinext auto-registers `@vitejs/plugin-rsc` for App Router when the `rsc` option is not explicitly `false`. No manual RSC plugin config needed for local development. ## Phase 4: Deployment (Optional) ### Option A: Cloudflare Workers (recommended for Cloudflare) If the user wants to deploy to Cloudflare Workers, use `vinext deploy`. It auto-generates `wrangler.jsonc`, worker entry, and Vite config if missing, installs `@cloudflare/vite-plugin` and `wrangler`, then builds and deploys. For manual setup or custom worker entries, see [references/config-examples.md](references/config-examples.md). #### Cloudflare Bindings (D1, R2, KV, AI, etc.) To access Cloudflare bindings (D1, R2, KV, AI, Queues, Durable Objects, etc.), use `import { env } from "cloudflare:workers"` in any server component, route handler, or server action: ```tsx import { env } from "cloudflare:workers"; export default async function Page() { const result = await env.DB.prepare("SELECT * FROM posts").all(); return