Moving an Astro app from Webflow Cloud to Vercel mostly means swapping the adapter that targets the Cloudflare Workers runtime for one that targets Vercel's. On Webflow Cloud, your app runs as a Cloudflare Worker through the @astrojs/cloudflare adapter. On Vercel, the same app runs on Vercel Functions with Fluid compute enabled by default, in a full Node.js runtime, so it scales automatically and supports Astro's full on-demand rendering feature set.
This guide walks you through the full migration. You'll replace the @astrojs/cloudflare adapter with @astrojs/vercel, remove Wrangler, delete the Webflow Cloud configuration files, and drop the base path your app used to mount under a Webflow site. It also covers replacing Cloudflare storage bindings with their Vercel equivalents (for example, Object Storage with Vercel Blob), recreating environment variables, and deploying with Git or the vercel CLI.
Before you begin, make sure you have:
- An Astro app deployed on Webflow Cloud
- A Vercel account
- Vercel CLI installed (
npm i -g vercel) - Node.js 20 or later
If you use an AI coding agent like Claude Code or Cursor, you can have it handle most of the migration for you and provide expert guidance. Install the Vercel Plugin to provide your agent with Vercel-specific context, then add the companion skill for this guide.
Install the Vercel Plugin:
Add the Astro migration skill:
With both in place, ask your agent to migrate your Astro app from Webflow Cloud to Vercel. Your agent will follow the migration steps and apply Vercel's recommended patterns for the Vercel adapter, storage solutions, environment variables, and more.
On Webflow Cloud, your Astro app runs as a Cloudflare Worker. The @astrojs/cloudflare adapter compiles it for the workerd runtime, Webflow Cloud serves it from a mount path inside an environment (e.g., /app), and your server code reads storage through bindings on the Cloudflare runtime exposed via Astro's locals object.
On Vercel, the same application runs on Vercel Functions in a full Node.js runtime. Vercel auto-detects Astro on import, serves your app from the root, reads configuration from process.env, and connects to storage providers in the Vercel Marketplace through native integrations.
The table below maps each Webflow Cloud component to its Vercel counterpart.
| Webflow Cloud | Vercel |
|---|---|
Cloudflare Workers runtime (workerd) | Vercel Functions (Fluid compute) |
@astrojs/cloudflare adapter | @astrojs/vercel adapter |
platformProxy (adapter option for local dev) | Not needed |
react-dom/server.edge Vite alias | Not needed (full Node.js runtime) |
webflow.json | Not needed, Astro is auto-detected |
wrangler.json | vercel.json (optional) |
worker-configuration.d.ts | Not needed |
base and build.assetsPrefix (mount path) | Served from the root; remove unless you want a base path |
locals.runtime.env and Astro.locals.runtime.env | process.env |
dev.vars (local runtime variables) | .env via vercel env pull |
webflow cloud deploy or GitHub push | Git push or the vercel CLI |
npm only | npm, pnpm, Yarn, or Bun |
| Object Storage (R2 binding) | Vercel Blob |
| Key Value Store (Workers KV binding) | Redis from the Vercel Marketplace, or Edge Config for read-heavy config |
| SQLite (D1 binding) | Postgres from the Vercel Marketplace |
export const config = { runtime: 'edge' } on routes | Remove it to run on Node.js (recommended) |
| Edge runtime API routes | Astro Server Endpoints on Vercel Functions |
| Astro middleware on the Edge runtime | Astro middleware on Vercel Functions, or at the Edge with middlewareMode: 'edge' |
| No scheduled jobs, queues, or workflows | Cron Jobs, Queues, and Workflows |
Astro needs an adapter to server-render on each platform, so the core setup change is swapping one adapter for the other.
Uninstall the Cloudflare adapter and Wrangler, then install the Vercel adapter:
Update astro.config.mjs to import and use the Vercel adapter. Keep output: 'server' to render every route on demand as a Vercel function, which matches how your app ran on Webflow Cloud:
If most of your pages are static and only a few are dynamic, set output: "static" instead and add export const prerender = false to the components that need server rendering.
Two Cloudflare-specific options can go at the same time. Remove the platformProxy setting, which only configured the Workers runtime for local development, and remove the react-dom/server.edge Vite alias, a workaround for the Workers environment that Vercel's Node.js runtime doesn't need.
Delete the Cloudflare- and Webflow-specific files that no longer apply on Vercel:
webflow.json, which told Webflow Cloud your framework. Vercel detects Astro automatically.wrangler.json, including itscompatibility_date,nodejs_compatflag,assetsbinding, and storage bindings. Vercel Functions run on Node.js, so the compatibility flags have no equivalent.worker-configuration.d.ts, the generated binding types.
Your storage bindings live in wrangler.json. Before deleting the file, note which bindings your app uses (Object Storage, Key Value Store, or SQLite) so you can recreate them on Vercel in step five.
On Webflow Cloud, your app is served from a mount path such as /app, so you set base and build.assetsPrefix in astro.config.mjs to match.
On Vercel, your app is served from the root, so remove both options unless you intend to keep serving the app under a subpath:
Because the base path is gone, remove the manual prefixing Webflow Cloud required in client-side fetch calls and asset references. Change fetch(${import.meta.env.BASE_URL}/api/users) back to fetch("/api/users"), and drop import.meta.env.ASSETS_PREFIX from plain <img> tags and from the favicon link in your layout.
Webflow Cloud also runs API routes on the Edge runtime. On Vercel, remove the export const config = { runtime: 'edge' } directive from your Server Endpoints so they run on the default Node.js runtime. We recommend migrating from the Edge runtime to Node.js for improved performance and reliability, and both runtimes run on Fluid compute with Active CPU pricing. Running on Node.js also gives your routes access to the full Node.js API surface and to npm packages that depend on Node.js built-ins.
Replace the Webflow Cloud preview script in package.json with the standard Astro commands. Vercel runs the build for you, so you no longer need Wrangler to preview the Workers build:
Vercel auto-detects Astro on import and sets the build command and output directory, so these scripts mainly support local development.
This is the core code change.
On Webflow Cloud, you read storage through bindings on the Cloudflare runtime, which you access in a Server Endpoint through the locals object.
On Vercel, you read connection details from process.env and talk to each store through its SDK or client. Remove every locals.runtime.env and Astro.locals.runtime.env access, and replace the binding operations.
For example, an Object Storage upload on Webflow Cloud looks like this:
On Vercel, the same upload uses Vercel Blob:
Install the Blob SDK with npm i @vercel/blob, then create a Blob store from the Storage page in your Vercel dashboard and connect it to your project. Vercel adds the store's environment variables, and the SDK uses them automatically, so the put() call above needs no token in your code. This OIDC approach is recommended over the long-lived BLOB_READ_WRITE_TOKEN, which you'd use only for code that runs outside Vercel.
Map your other Webflow Cloud storage the same way:
- Key Value Store becomes a Redis integration (e.g., Upstash Redis) from the Vercel Marketplace for caching and session data, or Edge Config for small, read-heavy configuration.
- SQLite becomes a Postgres database (e.g., Neon) from the Vercel Marketplace. If you used Drizzle ORM with SQLite on Webflow Cloud, Drizzle supports Postgres too, so you keep your schema-first workflow.
When you provision storage from the Marketplace, Vercel adds the connection string and credentials as environment variables, which your code reads from process.env.
Recreate your Webflow Cloud environment variables as Vercel environment variables. Webflow Cloud stores these per environment in your app's settings and injects them at runtime only. Vercel stores them per environment (production, preview, and development) in project settings and makes them available at both build time and runtime.
Add each variable to your project's environment variables, or from the CLI:
To run your app locally with the same values, link the project and pull the variables into a local .env file. You can delete the dev.vars file Webflow Cloud used for local runtime variables, since Astro loads .env automatically:
The way you read variables changes, too.
On Webflow Cloud, custom server variables came from Astro.locals.runtime.env in components and locals.runtime.env in Server Endpoints.
On Vercel's Node.js runtime, read them from process.env (for example, process.env.DATABASE_URL). For variables you want exposed to client-side code, use Astro's PUBLIC_ prefix and import.meta.env.
Because Vercel exposes environment variables during the build, you can re-enable any build-time validation you had to disable on Webflow Cloud, where variables aren't available at build time.
You have two deployment options. Both run your app on Vercel Functions.
Deploy with Git (recommended):
- Push your project to GitHub, GitLab, or Bitbucket.
- In the Vercel dashboard, select Add New > Project, then import your repository.
- Vercel detects Astro and sets the build command and output directory. Confirm the framework preset, add your environment variables, and select Deploy.
After the first import, every push to your main branch creates a production deployment, and every pull request gets its own preview URL.
Deploy with the CLI:
Run vercel from your project root to create a preview deployment, or vercel --prod to deploy to production.
Once deployed, your app runs on Vercel Functions with Fluid compute, with preview deployments, observability, and the Vercel Firewall available.
Your config or server code still references the Cloudflare adapter. Search your project for @astrojs/cloudflare and locals.runtime, then remove the adapter import and platformProxy from astro.config.mjs and replace each binding call with its Vercel equivalent from step five. The adapter module is only resolved within the build that targets Cloudflare Workers.
Code still reads locals.runtime.env, which only exists when the Cloudflare adapter populates it. On Vercel, that object isn't present. Replace locals.runtime.env.VARIABLE with process.env.VARIABLE for secrets and connection strings, and use the relevant SDK for storage, as shown in step five.
A leftover base path is usually the cause. Confirm you removed base and build.assetsPrefix from astro.config.mjs, and remove any manual base-path prefixing from client-side fetch calls, <img> tags, and your layout's favicon link. On Vercel, your app is served from the root, so paths like /api/users and /logo.png resolve without a prefix.
Confirm each variable exists in the correct environment under your project's environment variables, then redeploy. Variables added to production aren't available in preview or development unless you add them there too. For local runs, re-run vercel env pull after changing variables.
- Tune function resources. If your routes need more time than the default, set
maxDurationin your Vercel adapter configuration (for example,adapter: vercel({ maxDuration: 60 })), or configure memory and regions invercel.json. - Place functions near your data. Set the function region close to your database region to reduce latency.
- Take advantage of features that were limited on Webflow Cloud. Incremental Static Regeneration (
adapter: vercel({ isr: true })), Image Optimization with Astro'sImagecomponent, Web Analytics, and Speed Insights all work on Vercel with minimal configuration.