Cloudflare Docs
Pages
Visit Pages on GitHub
Set theme to dark (⇧+D)

Functions (beta)

With Pages, you can now build full-stack applications by executing code on the Cloudflare network with help from Cloudflare Workers. Functions enable you to run server-side code to enable dynamic functionality without running a dedicated server. With Functions, you can introduce application aspects such as authenticating, querying databases, handling form submissions, or working with middleware.

​​ Built with Cloudflare Workers

Cloudflare Workers provides a serverless execution environment that allows you to create entirely new applications or augment existing ones without configuring or maintaining infrastructure.

Previously, you could only add dynamic functionality to your Pages site by manually deploying a Worker using Wrangler, which meant that your application is written across both Pages and Workers. Functions allow you to leverage the Workers platform directly from within a Pages project by utilizing a project’s filesystem convention. This enables you to deploy your entire site – both its static and dynamic content – when you git push.

​​ Setup

To get started, create a /functions directory at the root of your project. Writing your Functions files in this directory will automatically generate a Worker with custom functionality at the predesignated routes.

​​ Functions routing

Using a /functions directory will generate a routing table based on the files present in the directory. You may use JavaScript (*.js) or TypeScript (*.ts) to write your Functions.

For example, assume this directory structure:

├── ...
├── functions
|   └── api
│       ├── [[path]].ts
│       ├── [username]
│       │   └── profile.ts
│       ├── time.ts
│       └── todos
│           ├── [[path]].ts
│           ├── [id].ts
│           └── index.ts
└── ...

The following routes will be generated based on the file structure, mapping the URL pattern to the /functions file that will be invoked:

/api/time => ./functions/api/time.ts
/api/todos => ./functions/api/todos/index.ts
/api/todos/* => ./functions/api/todos/[id].ts
/api/todos/*/** => ./functions/api/todos/[[path]].ts
/api/*/profile => ./functions/api/[username]/profile.ts
/api/** => ./functions/api/[[path]].ts

​​ Path segments

In the example above:

  • A * denotes a placeholder for a single path segment (for example, /todos/123).
  • A ** matches one or more path segments (for example, /todos/123/dates/confirm).

When naming your files:

  • [name] is a placeholder for a single path segment.
  • [[name]] matches any depth of route below this point.

When a filename includes a placeholder, the name must be alphanumeric and cannot contain spaces. In turn, the URL segment(s) that match the placeholder will be available under the context.params object using the filename placeholder as the key.

​​ Using TypeScript

If you prefer to write TypeScript, we declare a PagesFunction type in the @cloudflare/workers-types library which you can use to type-check your Functions. To use this, create a tsconfig.json file in your /functions folder with the following contents:

functions/tsconfig.json
{
"compilerOptions": {
"target": "ES2020",
"module": "CommonJS",
"lib": ["ES2020"],
"types": ["@cloudflare/workers-types"]
}
}

If you already have a tsconfig.json at the root of your project, you may wish to explicitly exclude the functions directory to avoid conflicts:

tsconfig.json
{
"include": ["src/**/*"],
"exclude": ["functions/**/*"],
"compilerOptions": {
}
}

​​ Writing your first function

When writing request handlers within your Pages application, each /functions file must export a function to handle the incoming request. Each function will receive a singular context object, which contains all the information for the request:

export async function onRequest(context) {
// Contents of context object
const {
request, // same as existing Worker API
env, // same as existing Worker API
params, // if filename includes [id] or [[path]]
waitUntil, // same as ctx.waitUntil in existing Worker API
next, // used for middleware or to fetch assets
data, // arbitrary space for passing data between middlewares
} = context;
return new Response("Hello, world!");
}

When migrating from a Module Worker, this signature combines the traditional fetch handler’s arguments into a single object along with additional, Pages-specific keys.

In the previous example, an onRequest function was exported. This is a generic name because it generically handles all HTTP requests. However, to react to specific HTTP request methods, you may use the method name as a suffix to the exported function. For example, a handler that should only receive GET requests should be named onRequestGet. The following other handlers are supported:

  • onRequestPost
  • onRequestPut
  • onRequestPatch
  • onRequestDelete
  • onRequestHead
  • onRequestOptions

These are the requests you export to write your first function. For example, you can write a function to output "Hello World" when it hits a /functions/hello-world.js file:

functions/hello-world.js
// Reacts to POST /hello-world
export async function onRequestPost(request) {
// ...
return new Response(`Hello world`);
}

Another helpful example for handling single path segments can be querying an API for data, for example, Rick and Morty API for information on the show characters. You can write a function to show each character on request using the ID to identify them:

functions/character/[id].js
export async function onRequestGet({ params }) {
const res = await fetch(`https://rickandmortyapi.com/api/character/${params.id}`);
const data = await res.json();
const info = JSON.stringify(data, null, 2);
return new Response(info);
}

The above will return each character at /character/{id} ID being associated with the character.

​​ Handling multiple requests in a single function

You can define multiple HTTP handlers in a single file by defining multiple exports within the same file. For example, this file will handle POST and PUT requests with the same handler code:

export async function onRequestPost(context) {
// ...
}
export const onRequestPut = onRequestPost;

Additionally, an exported handler may be an array of function handlers. This allows you to easily compose Functions as a group, which may include a mix of shared and/or one-off behaviors:

import { extraLogging } from "middlewares.ts";
export const onRequest = [
extraLogging,
async ({ request }) => {
// ...
},
];

​​ Adding middleware

Middleware are reusable chunks of logic that can be executed before and/or after route handlers. These are typically utility Functions that should be applied for chunks of an application’s routes. For example, error handling, user authentication, and logging are typical candidates for middleware within an application.

​​ Exporting middleware

Middleware files are similar to standard Function files. You may export an onRequest handler or any of its method-specific variants. Additionally, like Functions files, you may export an array of Functions as your middleware handler.

In your _middleware.{js|ts} files, you can define a middleware function that handles errors for all your Functions and export it to all the onRequest methods. This means the Functions defined within this file will be called on each function request declared in the directory that the middleware lives in. For example:

functions/_middleware.js
const errorHandler = async ({ next }) => {
try {
return await next();
} catch (err) {
return new Response(`${err.message}\n${err.stack}`, { status: 500 });
}
};
const hello = async ({ next }) => {
const response = await next();
response.headers.set('X-Hello', 'Hello from functions Middleware!');
return response;
};
export const onRequest = [errorHandler, hello];

In the function above, you can see that the errorHandler and hello Functions are exported to all requests so that if this middleware is in the base of the /functions directory it will run on all Functions defined in that directory. And if the middleware is defined in a subdirectory such as /functions/todos/_middleware.ts it will only run on all requests in that directory.

​​ Method-specific middleware

Much like Function handlers, you may export a method-specific handler instead of the generic onRequest name. For example, you may want to define an errorHandler for all requests and then an additional hello function for GET requests:

functions/hello/_middleware.js
import { errorHandler } from '../shared';
import { hello } from '../custom';
export const onRequest = [errorHandler];
export const onRequestGet = [errorHandler, hello];

​​ Middleware routing

A middleware will handle some checks, changes, or add information to a request before it reaches an endpoint. For logic that needs to be shared across multiple endpoints, define a _middleware.{js|ts} file in the /functions directory or subdirectory at whatever point you want it to apply. These _middleware files still apply in accordance with the URL pattern. For example:

├── functions
│ ├── _middleware.ts # Applies to all routes
│ ├── ...
│ └── todos
│ ├── _middleware.ts # Adds extra middleware to /todos/**
│ ├── ...
└── ...

This directory structure has two types of middleware: one that acts on every file in the folder (functions/_middleware.ts) and the other that is invoked only for a matching URL prefix (functions/todos/_middleware.ts). These middleware files will be applied to the the following URL paths:

/todos/** => [
./functions/_middleware.ts,
./functions/todos/_middleware.ts
]
<all other routes> => [
./functions/_middleware.ts
]

​​ Middleware chaining

Within Pages, middleware functions have access to a context.next function which, when invoked, will await the next function’s execution before the current middleware resumes. The ability to wait for other middleware and/or the final route handler(s) to finish is what allows for use cases like error handling, for example.

async function errorHandler(context) {
try {
// wait for the next function to finish
return await context.next();
} catch (err) {
// catch and report and errors when running the next function
return new Response(`${err.message}\n${err.stack}`, { status: 500 });
}
}
// Attach `errorHandler` to all HTTP requests
export const onRequest = errorHandler;

Another use case for the next function is passing the request cycle from the current middleware function to the next function in the stack if the current function does not end the request-response cycle. Using the next() function will pass control to the next middleware function, depending on the order of execution. For example:

// Attach multiple handlers
export const onRequest = [
async ({ request, next }) => {
try {
// Call the next handler in the stack
const response = await next();
const responseText = await response.text();
//~> "Hello from next base middleware"
return new Response(responseText + " from middleware");
} catch (thrown) {
return new Response(`Error ${thrown}`, {
status: 500,
statusText: "Internal Server Error",
});
}
},
({ request, next }) => {
return new Response("Hello from next base middleware");
},
];

​​ Middleware data

Handler functions have the ability to pass data between one another. This is done through the context.data property, which is accessible and mutable by all handlers throughout a request’s execution.

More often than not, context.data is only relevant from a middleware’s perspective, but it is available to all functions regardless.

functions/_middleware.js
export async function onRequest(context) {
let res;
try {
context.data.timestamp = Date.now();
res = await context.next();
} catch (err) {
res = new Response('Oops!', { status: 500 });
} finally {
let delta = Date.now() - context.data.timestamp;
res.headers.set('x-response-timing', delta);
return res;
}
}

​​ Adding bindings

A binding is how your Function (Worker) interacts with external resources. You can add KV, Durable Object, and plain-text bindings to your project. A binding is a runtime variable that the Workers runtime provides to your code. You can also use these bindings in development with Wrangler.

​​ KV namespace

Workers KV is Cloudflare’s globally replicated key-value storage solution. Within Pages, you can choose from the list of KV namespaces that you created from the dashboard by going to Account Home > Pages > your Pages project > Settings > Functions > KV namespace bindings. Select Add binding and input a Variable name and select a KV namespace from the list of your existing Workers KV namespaces. You will need to repeat this for both the Production and Preview environments.

Editing a KV namespace Binding and adding a Variable name

​​ KV namespace locally

While developing locally, you can interact with your KV namespace by add -k, --kv [Namespace binding] to your run command. For example, if your namespace is bound to TodoList, you can access the KV namespace in your local dev by running npx wrangler pages dev dist --kv TodoList. The data from this namespace can be accessed using context.env.

export async function onRequest({ env }) {
const task = await env.TodoList.get("Task:123");
return new Response(task);
}

​​ Durable Object namespace

Durable Objects are Cloudflare’s strongly consistent coordination primitive that power capabilities such as connecting WebSockets, handling state, and building applications. As with Workers KV, you first have to create the Durable Object. You can then configure it as a binding to your Pages project.

Go to Account Home > Pages > your Pages project > Settings > Functions > Durable Object bindings. Select Add binding and input a Variable name and select a Durable Object namespace from the list of your existing Durable Objects. You will need to repeat this for both the Production and Preview environments.

Editing a Durable Object namespace Binding and adding a Variable name

​​ Durable Objects locally

Currently, Durable Objects are not supported in local development mode. To use Durable Objects in your Pages application, deploy a Worker containing a Durable Object. Then add it as a binding to your Pages project as shown in the section above.

Support for using Durable Objects in local development is actively being worked on and will be available soon.

​​ R2 bucket

Cloudflare R2 is Cloudflare’s blob storage solution that allows developers to store large amounts of unstructured data without the costly egress bandwidth fees associated with typical cloud storage services. Within Pages, you can choose from a list of R2 buckets that you created from the dashboard by going to Account Home > Pages > your Pages project > Settings > Functions > R2 buckets. Select an R2 bucket from the list of your existing R2 buckets. You will need to repeat this step for both the Production and Preview environments.

Editing an R2 bucket Binding and adding a Variable name

​​ Using R2 buckets locally

While developing locally, you can interact with an R2 bucket by adding --r2=<BINDING> to your run command. For example, if your bucket is bound to BUCKET, you can access this bucket in local dev by running npx wrangler pages dev dist --r2=BUCKET. The can interact with this binding by using context.env (e.g. context.env.BUCKET).

export async function onRequestGet({ env }) {
const obj = await env.BUCKET.get('some-key');
if (obj === null) {
return new Response('Not found', { status: 404 });
}
return new Response(obj.body);
}

​​ D1 database

Cloudflare D1 is Cloudflare’s first SQL database built on SQLite. If you have access to D1, within Pages, you can choose from a list of D1 databases that you created from the dashboard by going to Account Home > Pages > your Pages project > Settings > Functions > D1 Databases. Select a D1 database from the list of your existing D1 databases. You must repeat this step for both the Production and Preview environments.

Editing a D1 database Binding and adding a Variable name

​​ Using D1 database locally

While developing locally, you can interact with a D1 database by adding --d1=<BINDING> to your run command. For example, if your database is bound to NORTHWIND_DB, you can access this database in local dev by running npx wrangler pages dev dist --d1=NORTHWIND_DB. You can interact with this binding by using context.env (e.g. context.env.NORTHWIND_DB).

export async function onRequestGet({ env }) {
const ps = env.NORTHWIND_DB.prepare('SELECT * from users');
const data = await ps.first();
return Response.json(data);
}

​​ Environment variable

An environment variable is an injected value that can be accessed by your Functions. It is stored as plaintext. You can set your environment variables directly within the Pages interface for both your production and preview environments at run-time and build-time.

To add environment variables, go to Account Home > Pages > your Pages project > Settings > Environment variables.

Editing an environment variable by adding a variable name and value

​​ Adding environment variables locally

When developing locally, you can access environment variables by adding a .dev.vars file to the root directory of your project. Next, define your environment variables and then access them in your component by using env.ENV_NAME.

For example :

/.dev.vars
ENV_NAME = "SUPER_SECRET_KEY"
functions/index.js
export async function onRequest({ env }) {
return new Response(env.ENV_NAME);
}

Here is a real-world example of using environment variables inside a middleware function. To connect Sentry to a Cloudflare Worker, you can use Toucan js and access your Sentry Data Source Name (DSN) in your function.

const SentryMiddleware = async ({ request, next, env, waitUntil }) => {
const sentry = new Toucan({
dsn: env.SENTRY_DSN,
context: { waitUntil, request },
});
try {
return await next();
} catch (thrown) {
sentry.captureException(thrown);
return new Response(`Error ${thrown}`, {
status: 500,
});
}
};
export const onRequest = [SentryMiddleware];

​​ Advanced mode

In some cases, the built-in routing and middleware system is not desirable for existing applications. You may already have a Worker that is fairly complex and/or would be tedious to splice it up into Pages' file-based routing system. For these cases, Pages offers developers the ability to define a _worker.js file in the output directory of your Pages project.

When using a _worker.js file, the entire /functions directory is ignored – this includes its routing and middleware characteristics. Instead, the _worker.js file is deployed as is and must be written using the Module Worker syntax.

If you have never used Module syntax, refer to the JavaScript modules blog post to learn more. Using Module Workers enables JavaScript frameworks to generate a Worker as part of the Pages output directory contents.

Your custom Module Worker will assume full control of all incoming HTTP requests to your domain. Because of this, your custom Worker is required to make and/or forward requests to your project’s static assets.

_worker.js
export default {
async fetch(request, env) {
const url = new URL(request.url);
if (url.pathname.startsWith('/api/')) {
// TODO: Add your custom /api/* logic here.
return new Response('Ok');
}
// Otherwise, serve the static assets.
// Without this, the Worker will error and no assets will be served.
return env.ASSETS.fetch(request);
},
};

The env.ASSETS.fetch() function will allow you to send the user to a modified path which is defined through the url parameter. env is the object that contains your environment variables and bindings. ASSETS is a default Function binding that allows communication between your Function and Pages' asset serving resource. fetch() calls to Pages' asset-serving resource and serves the requested asset.

Then after placing your _worker.js file in your output directory, deploy your project normally through your git integration.

​​ Migrating from Workers

When migrating a Worker into the Pages platform, the simplest path is to target the advanced mode of Functions. To do this, ensure your Worker is in the Module Worker format. Then call env.ASSETS when you want to serve static assets. Failure to do so will result in broken and/or unwanted behavior.

​​ Develop and preview locally

You can run your entire application locally with Wrangler, which supports secrets, environment variables, KV and Durable Objects. Point Wrangler at a directory of static assets, or seamlessly connect to your existing tools:

# Install wrangler
$ npm install wrangler
# Show help message
$ npx wrangler pages dev --help
# Serve a folder of static assets
$ npx wrangler pages dev ./dist
# Bind to a KV store
$ npx wrangler pages dev ./dist --kv KV_NAMESPACE
# Bind to a Durable Object
$ npx wrangler pages dev ./dist --do ENV_NAME=CLASS_NAME
# Bind to an R2 Bucket
$ npx wrangler pages dev ./dist --r2 MY_BUCKET
# Bind to a D1 Database
$ npx wrangler pages dev ./dist --d1 MY_DATABASE
# Or automatically proxy your existing tools
$ npx wrangler pages dev -- npx react-scripts start
# Bind variable/secret (KEY=VALUE)
$ npx wrangler pages dev dist --binding ENV_NAME=\"ENV_VALUE\"

Developing locally does not deploy your changes. It is only a means to preview and test. To deploy your changes to your Pages site, you will need to git commit and git push as normal.

​​ Pricing and limits

While still in open beta, there is no additional cost to deploy Functions within a Pages project. However, during beta, there is a total limit of 100,000 invocation requests per day. If you reach your daily limit, Pages will stop executing your Functions and fallback to serving only static assets.

If you reach this limit or need it increased in order to migrate your existing Workers applications into a Pages project, complete this form to request an increase. You may track this limit by going to Account Home > Pages > and reviewing Functions requests today.

In the future, you can expect billing to reflect that of the Workers Bundled plan.

​​ Demo

To get started with your first Pages project with Functions, refer to the demo blog post on how to build an image sharing application. In this demo, you will build a JSON API with Functions (storing data on KV and Durable Objects), integrate with Cloudflare Images and Cloudflare Access, and use React for your front end.