Back to Rivet

Deploying to Vercel

website/src/content/docs/deploy/vercel.mdx

2.3.24.7 KB
Original Source

This guide assumes a Next.js app.

Prerequisites

Steps

<Steps> <Step title="Set up your project">

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 handler
  • src/rivet/registry.ts: Actor definitions and registry

The route handler sets maxDuration to extend the serverless function timeout so long-lived actor requests are not cut short:

ts
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);
</Step> <Step title="Set Environment Variables">

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.

</Step> <Step title="Deploy to Vercel">
  1. Connect your GitHub repository to Vercel
  2. Vercel will deploy your app
</Step> <Step title="Configure Preview Deployments (Recommended)">

Add a GitHub action to automatically create isolated Rivet namespaces for each PR:

  1. Add these secrets to your GitHub repository:

  2. Create .github/workflows/rivet-preview.yml:

yaml
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 }}
</Step> </Steps>

Troubleshooting

<AccordionGroup> <Accordion title="ENOENT: no such file or directory (rivetkit state)">
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:

  1. Create a bypass secret in your Vercel project settings
  2. In Rivet, go to Settings > Providers
  3. Click the three dots on your provider and select Edit
  4. Click Add Header and add x-vercel-protection-bypass with your bypass secret

If using the preview-namespace-action, this is configured automatically.

</Accordion> </AccordionGroup>