Static Site Generation (SSG) and custom 404 pages
Static Site Generation (SSG) applications are web applications which are predominantely built or "prerendered" ahead-of-time. They are often built with a framework such as Gatsby or Docusaurus. The build process of these frameworks will produce many HTML files and accompanying client-side resources (e.g. JavaScript bundles, CSS stylesheets, images, fonts, etc.). Data is either static, fetched and compiled into the HTML at build-time, or fetched by the client from an API with client-side requests.
Often, an SSG framework will allow you to create a custom 404 page.
In order to deploy a Static Site Generation application to Workers, you must configure the assets.directory
, and optionally, the assets.not_found_handling
and assets.html_handling
options in your Wrangler configuration file:
{ "name": "my-worker", "compatibility_date": "2025-06-06", "assets": { "directory": "./dist/", "not_found_handling": "404-page", "html_handling": "auto-trailing-slash" }}
name = "my-worker"compatibility_date = "2025-06-06"
[assets]directory = "./dist/"not_found_handling = "404-page"html_handling = "auto-trailing-slash"
assets.html_handling
defaults to auto-trailing-slash
and this will usually give you the desired behavior automatically: individual files (e.g. foo.html
) will be served without a trailing slash and folder index files (e.g. foo/index.html
) will be served with a trailing slash. Alternatively, you can force trailing slashes (force-trailing-slash
) or drop trailing slashes (drop-trailing-slash
) on requests for HTML pages.
Configuring assets.not_found_handling
to 404-page
overrides the default serving behavior of Workers for static assets. When an incoming request does not match a file in the assets.directory
, Workers will serve the contents of the nearest 404.html
file with a 404 Not Found
status.
If you have a Worker script (main
), have configured assets.not_found_handling
, and use the assets_navigation_prefers_asset_serving
compatibility flag (or set a compatibility date of 2025-04-01
or greater), navigation requests will not invoke the Worker script. A navigation request is a request made with the Sec-Fetch-Mode: navigate
header, which browsers automatically attach when navigating to a page. This reduces billable invocations of your Worker script, and is particularly useful for client-heavy applications which would otherwise invoke your Worker script very frequently and unnecessarily.
In some cases, you might need to pass a value from a navigation request to your Worker script. For example, if you are acting as an OAuth callback, you might expect to see requests made to some route such as /oauth/callback?code=...
. With the assets_navigation_prefers_asset_serving
flag, your HTML assets will be server, rather than your Worker script. In this case, we recommend, either as part of your client application for this appropriate route, or with a slimmed-down endpoint-specific HTML file, passing the value to the server with client-side JavaScript.
<!DOCTYPE html><html> <head> <title>OAuth callback</title> </head> <body> <p>Loading...</p> <script> (async () => { const response = await fetch("/api/oauth/callback" + window.location.search); if (response.ok) { window.location.href = '/'; } else { document.querySelector('p').textContent = 'Error: ' + (await response.json()).error; } })(); </script> </body></html>