Information about how to setup and run AB Testing in a multitude of ways in this project.
The base framework used is the Guardian's AB Testing Library.
More framework/API specific documentation is available there.
- Define the AB test: Each AB test and their variants are defined in code with configuration such as audience size & offset and impression & success listeners etc
- Initialise the library: The AB Test library is initialised with configuration values such as a user's MVT ID, an array of the above defined A/B tests etc
- Use the AB Test API: The intialisation returns an API that can be used to check if the current user is in a variant of a test along with a variety of other API methods
There are 2 ways to use AB Tests on Gateway.
- Client Side: Runs on the client/browser, e.g. for showing different components or taking different actions depending on the test in question. Works with SSR too. This is the easiest way to setup and run tests.
- Per Request: The AB test information and API are also available on the RequestState (res.locals) to be untilised within routes/middleware. This is more complex to setup and run, but for example it gives the option of directing users to different routes/flows based on the test in question.
The src/shared/model/experiments/tests folder contains test definitions for tests that may need to be run.
To create a new test, create a new file, and the test definition using the information/template from the library.
In the abTests.ts file, import and add the new test to the tests
array. This will make it available to the ab testing library and api.
Finally you have to add a switch for the test in the abSwitches.ts file, in the abSwitches
object. The key
should be ab
+ the id
from the test definition. For example, if the id
in the test definition is ExampleTest
, then the switch key should be abExampleTest
. The value
should be a boolean
, with true
if the test is enabled, and false
if the test is disabled.
The AB Testing Library has more information available to setup tests with.
See the ABTestDemo
component for possible ways to run tests on the client.
You can view this demo by adding this component to the Main
component.
// other imports
...
import { ABTestDemo } from './components/ABTestDemo';
...
export const Main = (props: ClientState) => {
...
return (
<>
...
<ClientStateProvider clientState={props}>
{/* This will show the demo above the rest of the app*/}
<ABTestDemo />
<GatewayRoutes />
</ClientStateProvider>
</>
);
}
Running per request is a bit more complicated, as an example, see the middleware code example, on a possible way of running an AB test on that particular users request.
This can me demoed by adding this middleware to in the middleware index file, after the requestStateMiddleware
has been declared.
It is important to load it after requestStateMiddleware
other wise the AB tests will not have the state available to them to work.
Assuming the middleware is called exampleABMiddleware
, then:
export const applyMiddleware = (server: Express): void => {
...
server.use(requestStateMiddleware);
server.use(exampleABMiddleware);
server.use(routes);
...
}
In both demos be sure not to commit the demo changes to production.
There are 2 ways of forcing yourself into a test:
- Recommended: Use
GU_mvt_id
(orGU_mvt_id_local
) cookie - Use URL parameters
This method requires manually setting the GU_mvt_id
cookie, or the GU_mvt_id_local
cookie (if you're on the DEV
stage).
Use this simple calculator to work out what value you need for a particular test. Simply add the AB Test Config information (audience + offset), and the variants in that test. Then modify the MVT ID value until Is user in test?
is Yes
and the variant you want is highlighted. Then copy this MVT ID value as the cookie value for the GU_mvt_id
or GU_mvt_id_local
cookie.
The library has useful documentation about the calculator.
At first glance this may seem more difficult that using the URL parameters, but the main advantages to this method are:
- Less chance of overlapping tests
- URL params set the
forcedTestVariants
parameter in the framework, which may mean that it may overlap with any other tests in that bucket.
- URL params set the
- You'll always be in that AB Test throughout the lifetime of the cookie
- URL parameters are only valid for that single request, which means that tests that rely on multiple requests/routes would be required to add the parameters on every request manually
- More granular control for audience/offset testing
- As the values are individual buckets, you can fine tune the audience and the offset as required.
You can also force yourself into a test and variant using URL parameters, either in the query parameters (after the ?
) or as a search parameter (after the #
). This requires knowing the ab test id and the variant name. Also to note, you have to prefix the test id with ab-
. For example, to force yourself into the ExampleTest
and the variant
variant. You could add the parameter onto the URL like this https://profile.theguardian.com/signin#ab-ExampleTest=variant
.
The advantages to this are that it's simple to do and test, however the parameters may not persist between requests, so might not be able to test a full flow relying on the AB test.
import { ResponseWithRequestStateLocals } from '@/server/models/Express';
import { tests } from '@/shared/model/experiments/abTests';
import { exampleTest } from '@/shared/model/experiments/tests/example-test';
import { Request, NextFunction } from 'express';
export const abTestDemoMiddleware = (
_: Request,
res: ResponseWithRequestStateLocals,
next: NextFunction,
) => {
// get the AB Test API
const ABTestAPI = res.locals.abTestAPI;
// WAYS TO RUN AB TESTS
// 1) Using the API (recommended)
// More documentation at https://github.com/guardian/ab-testing#the-api
// A) Example for getting and running the first possible runnable test:
// Get the first possible runnable test
const firstRunnableTest = ABTestAPI.firstRunnableTest(tests);
// Get the variant information to run
const variantFromRunnable = firstRunnableTest?.variantToRun;
// Get the test method which should run
const testToRun = variantFromRunnable?.test;
// Run the test method
console.log(
'A) API - First Runnable Test - Outcome:',
testToRun && testToRun({}),
);
// B) Example for checking if a user is in a particular test
// then get the variant, and run the test method
const runnableTest = ABTestAPI.runnableTest(exampleTest);
console.log(
'B) API - Check for Test - Outcome:',
runnableTest?.variantToRun.test({}),
);
// C) Example for checking if a user is in a specific test and variant
const isUserInVariant = ABTestAPI.isUserInVariant(
exampleTest.id,
exampleTest.variants[0].id,
);
console.log(
'C) API - Check for test and variant boolean - Outcome:',
isUserInVariant,
);
// you can use any of the above for conditional logic too
// example using C)
if (isUserInVariant) {
console.log('C) User in a variant');
} else {
console.log('C) User not in a variant');
}
// 2) Using the RequestState res.locals (not recommended)
// In the RequestState res.locals, we pass the testId and variant of any tests
// the user is in, you can check this too.
// This isn't recommended as the RequestState res.locals only has the test id and variant
// so may be more complex to run a test using this
next();
};