Use Pages Functions for A/B testing
In this guide, you will learn how to use Pages Functions for A/B testing in your Pages projects. A/B testing is a user experience research methodology applied when comparing two or more versions of a web page or application. With A/B testing, you can serve two or more versions of a webpage to users and divide traffic to your site.
Configuring different versions of your application for A/B testing will be unique to your specific use case. For all developers, A/B testing setup can be simplified into a few helpful principles.
Depending on the number of application versions you have (this guide uses two), you can assign your users into experimental groups. The experimental groups in this guide are the base route / and the test route /test.
To ensure that a user remains in the group you have given, you will set and store a cookie in the browser and depending on the cookie value you have set, the corresponding route will be served.
In your project, you can handle the logic for A/B testing using Pages Functions. Pages Functions allows you to handle server logic from within your Pages project.
To begin:
- Go to your Pages project directory on your local machine.
- Create a /functionsdirectory. Your application server logic will live in the/functionsdirectory.
Pages Functions have utility functions that can reuse chunks of logic which are executed before and/or after route handlers. These are called middleware. Following this guide, middleware will allow you to intercept requests to your Pages project before they reach your site.
In your /functions directory, create a _middleware.js file.
Following the Functions naming convention, the _middleware.js file exports a single async onRequest function that accepts a request, env and next as an argument.
const abTest = async ({request, next, env}) => {  /*  Todo:  1. Conditional statements to check for the cookie  2. Assign cookies based on percentage, then serve  */}
export const onRequest = [abTest]To set the cookie, create the cookieName variable and assign any value. Then create the newHomepagePathName variable and assign it /test:
const cookieName = "ab-test-cookie"const newHomepagePathName = "/test"
const abTest = async ({request, next, env}) => {  /*  Todo:  1. Conditional statements to check for the cookie  2. Assign cookie based on percentage then serve  */}
export const onRequest = [abTest]Based on the URL pathname, check that the cookie value is equal to new. If the value is new, then newHomepagePathName will be served.
const cookieName = "ab-test-cookie"const newHomepagePathName = "/test"
const abTest = async ({request, next, env}) => {  /*  Todo:  1. Assign cookies based on randomly generated percentage, then serve  */
  const url = new URL(request.url)  if (url.pathname === "/") {    // if cookie ab-test-cookie=new then change the request to go to /test    // if no cookie set, pass x% of traffic and set a cookie value to "current" or "new"
    let cookie = request.headers.get("cookie")    // is cookie set?    if (cookie && cookie.includes(`${cookieName}=new`)) {      // Change the request to go to /test (as set in the newHomepagePathName variable)      url.pathname = newHomepagePathName      return env.ASSETS.fetch(url)    }  }}
export const onRequest = [abTest]If the cookie value is not present, you will have to assign one. Generate a percentage (from 0-99) by using: Math.floor(Math.random() * 100). Your default cookie version is given a value of current.
If the percentage of the number generated is lower than 50, you will assign the cookie version to new. Based on the percentage randomly generated, you will set the cookie and serve the assets. After the conditional block, pass the request to next(). This will pass the request to Pages. This will result in 50% of users getting the /test homepage.
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 the Pages asset-serving resource and returns the asset (/test homepage) to your website's visitor.
const cookieName = "ab-test-cookie"const newHomepagePathName = "/test"
const abTest = async (context) => {  const url = new URL(context.request.url)  // if homepage  if (url.pathname === "/") {    // if cookie ab-test-cookie=new then change the request to go to /test    // if no cookie set, pass x% of traffic and set a cookie value to "current" or "new"
    let cookie = request.headers.get("cookie")    // is cookie set?    if (cookie && cookie.includes(`${cookieName}=new`)) {      // pass the request to /test      url.pathname = newHomepagePathName      return context.env.ASSETS.fetch(url)    } else {      const percentage = Math.floor(Math.random() * 100)      let version = "current" // default version      // change pathname and version name for 50% of traffic      if (percentage < 50) {        url.pathname = newHomepagePathName        version = "new"      }      // get the static file from ASSETS, and attach a cookie      const asset = await context.env.ASSETS.fetch(url)      let response = new Response(asset.body, asset)      response.headers.append("Set-Cookie", `${cookieName}=${version}; path=/`)      return response    }  }  return context.next()};
export const onRequest = [abTest];After you have set up your functions/_middleware.js file in your project you are ready to deploy with Pages. Push your project changes to GitHub/GitLab.
After you have deployed your application, review your middleware Function:
- Log in to the Cloudflare dashboard ↗ and select your account.
- In Account Home, select Workers & Pages.
- In Overview, select your Pages project > Settings > Functions > Configuration.