website/src/content/docs/deploy/vercel.mdx
This guide assumes a Next.js app.
Follow the Next.js Quickstart to set up your project.
</Step> <Step title="Verify Your Project Structure">Your Next.js project should have the following structure:
src/app/api/rivet/[...all]/route.ts: RivetKit route handlersrc/rivet/registry.ts: Actor definitions and registryThe route handler sets maxDuration to extend the serverless function timeout so long-lived actor requests are not cut short:
import { toNextHandler } from "@rivetkit/next-js";
import { registry } from "@/rivet/registry";
export const maxDuration = 300;
export const { GET, POST, PUT, DELETE, PATCH, HEAD, OPTIONS } = toNextHandler(registry);
Set RIVET_ENDPOINT and RIVET_PUBLIC_ENDPOINT in your Vercel project settings using the URL auth format:
RIVET_ENDPOINT=https://my-namespace:sk_****@api.rivet.dev
RIVET_PUBLIC_ENDPOINT=https://my-namespace:pk_****@api.rivet.dev
RIVET_ENDPOINT uses the secret token for server-side access. RIVET_PUBLIC_ENDPOINT uses the publishable token and tells the metadata endpoint what connection info to provide to clients.
Add a GitHub action to automatically create isolated Rivet namespaces for each PR:
Add these secrets to your GitHub repository:
RIVET_CLOUD_TOKEN: Get from Rivet Dashboard → Settings → Advanced → Cloud API TokensVERCEL_TOKEN: Get from Vercel Account SettingsCreate .github/workflows/rivet-preview.yml:
name: Rivet Preview
on:
pull_request:
types: [opened, synchronize, reopened]
push:
branches: [main]
concurrency:
group: rivet-preview-${{ github.event.pull_request.number || github.ref }}
cancel-in-progress: true
jobs:
rivet-preview:
runs-on: ubuntu-latest
permissions:
pull-requests: write
steps:
- uses: rivet-dev/preview-namespace-action@v1
with:
platform: vercel
rivet-token: ${{ secrets.RIVET_CLOUD_TOKEN }}
vercel-token: ${{ secrets.VERCEL_TOKEN }}
Error: ENOENT: no such file or directory, mkdir '.../rivetkit/.../state'
Cause: The RIVET_ENDPOINT environment variable is not configured. RivetKit falls back to the file system driver, which fails in Vercel's read-only serverless environment.
Solution: Ensure RIVET_ENDPOINT is set with your Rivet endpoint using the URL auth format:
RIVET_ENDPOINT=https://my-namespace:sk_****@api.rivet.dev
If using the preview-namespace-action, this is configured automatically.
</Accordion> <Accordion title="Metadata endpoint missing clientEndpoint">The /api/rivet/metadata endpoint returns data but clientEndpoint, clientNamespace, and clientToken are missing.
Cause: The RIVET_PUBLIC_ENDPOINT environment variable is not configured. This tells the metadata endpoint what connection info to provide to clients.
Solution: Set RIVET_PUBLIC_ENDPOINT with the publishable token (for client access):
RIVET_PUBLIC_ENDPOINT=https://my-namespace:pk_****@api.rivet.dev
If using the preview-namespace-action, this is configured automatically.
</Accordion> <Accordion title="401 Authentication Required from Rivet Actor">Rivet fails to connect to your Vercel deployment with a 401 error mentioning "Authentication Required".
Cause: Vercel Deployment Protection is blocking requests from Rivet.
Solution:
x-vercel-protection-bypass with your bypass secretIf using the preview-namespace-action, this is configured automatically.
</Accordion> </AccordionGroup>