Start from existing
Use Workers Static Assets Instead
You should use Workers Static Assets to host full-stack applications instead of Workers Sites. It has been deprecated in Wrangler v4, and the Cloudflare Vite plugin does not support Workers Sites. Do not use Workers Sites for new projects.
Workers Sites require Wrangler ↗ — make sure to use the latest version.
To deploy a pre-existing static site project, start with a pre-generated site. Workers Sites works with all static site generators, for example:
- Hugo ↗
- Gatsby ↗, requires Node
- Jekyll ↗, requires Ruby
- Eleventy ↗, requires Node
- WordPress ↗ (refer to the tutorial on deploying static WordPress sites with Pages)
-
Run the
wrangler init
command in the root of your project's directory to generate a basic Worker:Terminal window wrangler init -yThis command adds/update the following files:
wrangler.jsonc
: The file containing project configuration.package.json
: WranglerdevDependencies
are added.tsconfig.json
: Added if not already there to support writing the Worker in TypeScript.src/index.ts
: A basic Cloudflare Worker, written in TypeScript.
-
Add your site's build/output directory to the Wrangler file:
{"site": {"bucket": "./public"}}[site]bucket = "./public" # <-- Add your build directory name here.The default directories for the most popular static site generators are listed below:
- Hugo:
public
- Gatsby:
public
- Jekyll:
_site
- Eleventy:
_site
- Hugo:
-
Install the
@cloudflare/kv-asset-handler
package in your project:Terminal window npm i -D @cloudflare/kv-asset-handler -
Replace the contents of
src/index.ts
with the following code snippet:
import { getAssetFromKV } from "@cloudflare/kv-asset-handler";import manifestJSON from "__STATIC_CONTENT_MANIFEST";const assetManifest = JSON.parse(manifestJSON);
export default { async fetch(request, env, ctx) { try { // Add logic to decide whether to serve an asset or run your original Worker code return await getAssetFromKV( { request, waitUntil: ctx.waitUntil.bind(ctx), }, { ASSET_NAMESPACE: env.__STATIC_CONTENT, ASSET_MANIFEST: assetManifest, }, ); } catch (e) { let pathname = new URL(request.url).pathname; return new Response(`"${pathname}" not found`, { status: 404, statusText: "not found", }); } },};
import { getAssetFromKV } from "@cloudflare/kv-asset-handler";
addEventListener("fetch", (event) => { event.respondWith(handleEvent(event));});
async function handleEvent(event) { try { // Add logic to decide whether to serve an asset or run your original Worker code return await getAssetFromKV(event); } catch (e) { let pathname = new URL(event.request.url).pathname; return new Response(`"${pathname}" not found`, { status: 404, statusText: "not found", }); }}
-
Run
wrangler dev
ornpx wrangler deploy
to preview or deploy your site on Cloudflare. Wrangler will automatically upload the assets found in the configured directory.Terminal window npx wrangler deploy -
Deploy your site to a custom domain that you own and have already attached as a Cloudflare zone. Add a
route
property to the Wrangler file.{"route": "https://example.com/*"}route = "https://example.com/*"
Learn more about configuring your project.
Was this helpful?
- Resources
- API
- New to Cloudflare?
- Products
- Sponsorships
- Open Source
- Support
- Help Center
- System Status
- Compliance
- GDPR
- Company
- cloudflare.com
- Our team
- Careers
- © 2025 Cloudflare, Inc.
- Privacy Policy
- Terms of Use
- Report Security Issues
- Trademark