Skip to content

Routing

Functions utilize file-based routing. Your /functions directory structure determines the designated routes that your Functions will run on. You can create a /functions directory with as many levels as needed for your project’s use case. Review the following directory:

  • Directoryfunctions
    • index.js
    • helloworld.js
    • howdyworld.js
    • Directoryfruits
      • index.js
      • apple.js
      • banana.js

The following routes will be generated based on the above file structure. These routes map the URL pattern to the /functions file that will be invoked when a visitor goes to the URL:

File pathRoute
/functions/index.jsexample.com
/functions/helloworld.jsexample.com/helloworld
/functions/howdyworld.jsexample.com/howdyworld
/functions/fruits/index.jsexample.com/fruits
/functions/fruits/apple.jsexample.com/fruits/apple
/functions/fruits/banana.jsexample.com/fruits/banana

If no Function is matched, it will fall back to a static asset if there is one. Otherwise, the Function will fall back to the default routing behavior for Pages’ static assets.

Dynamic routes

Dynamic routes allow you to match URLs with parameterized segments. This can be useful if you are building dynamic applications. You can accept dynamic values which map to a single path by changing your filename.

Single path segments

To create a dynamic route, place one set of brackets around your filename – for example, /users/[user].js. By doing this, you are creating a placeholder for a single path segment:

PathMatches?
/users/neviYes
/users/danielYes
/profile/neviNo
/users/nevi/foobarNo
/neviNo

Multipath segments

By placing two sets of brackets around your filename – for example, /users/[[user]].js – you are matching any depth of route after /users/:

PathMatches?
/users/neviYes
/users/danielYes
/profile/neviNo
/users/nevi/foobarYes
/users/daniel/xyz/123Yes
/neviNo

Dynamic route examples

Review the following /functions/ directory structure:

  • Directoryfunctions
    • date.js
    • Directoryusers
      • special.js
      • [user].js
      • [[catchall]].js

The following requests will match the following files:

RequestFile
/fooWill route to a static asset if one is available.
/date/date.js
/users/daniel/users/[user].js
/users/nevi/users/[user].js
/users/special/users/special.js
/users/daniel/xyz/123/users/[[catchall]].js

The URL segment(s) that match the placeholder ([user]) will be available in the request context object. The context.params object can be used to find the matched value for a given filename placeholder.

For files which match a single URL segment (use a single set of brackets), the values are returned as a string:

export function onRequest(context) {
return new Response(context.params.user)
}

The above logic will return daniel for requests to /users/daniel.

For files which match against multiple URL segments (use a double set of brackets), the values are returned as an array:

export function onRequest(context) {
return new Response(JSON.stringify(context.params.catchall))
}

The above logic will return ["daniel", "xyz", "123"] for requests to /users/daniel/xyz/123.

Functions invocation routes

On a purely static project, Pages offers unlimited free requests. However, once you add Functions on a Pages project, all requests by default will invoke your Function. To continue receiving unlimited free static requests, exclude your project’s static routes by creating a _routes.json file. This file will be automatically generated if a functions directory is detected in your project when you publish your project with Pages CI or Wrangler.

Create a _routes.json file

Create a _routes.json file to control when your Function is invoked. It should be placed in the output directory of your project.

This file will include three different properties:

  • version: Defines the version of the schema. Currently there is only one version of the schema (version 1), however, we may add more in the future and aim to be backwards compatible.
  • include: Defines routes that will be invoked by Functions. Accepts wildcard behavior.
  • exclude: Defines routes that will not be invoked by Functions. Accepts wildcard behavior. exclude always take priority over include.

Example configuration

Below is an example of a _routes.json.

{
"version": 1,
"include": ["/*"],
"exclude": []
}

This _routes.json will invoke your Functions on all routes.

Below is another example of a _routes.json file. Any route inside the /build directory will not invoke the Function and will not incur a Functions invocation charge.

{
"version": 1,
"include": ["/*"],
"exclude": ["/build/*"]
}

Limits

Functions invocation routes have the following limits:

  • You must have at least one include rule.
  • You may have no more than 100 include/exclude rules combined.
  • Each rule may have no more than 100 characters.