How to use Next.js middleware to bootstrap feature flags
Jul 04, 2023
Next.js middleware enables you to run functions between requests and responses. We can use this to bootstrap feature flags on page load and make them available immediately without making an additional requests. This is useful for redirecting, showing components without flickering, avoiding layout changes, and more.
In this tutorial, we create a basic Next.js app, set up feature flags with PostHog, write middleware to get those feature flags, and bootstrap them in the client.
Create Next.js app and add PostHog
First, once Node is installed, create a Next.js app. Select No for TypeScript, Yes for use app router, and the defaults for every other option.
Next, go into your app and install PostHog.
Once done, set up a PostHog provider by creating a new file in the app folder named providers.js. In this file, import PostHogProvider, initialize PostHog with your project API key and instance address (from your project settings), and set up the provider component.
After this, import your provider component into layout.js and wrap your main app with it.
Create and set up a feature flag
After setting up PostHog, we need a feature flag. To create one, go to the feature flag tab in your PostHog instance, and click "New feature flag." Add a key (like bootstrap-test), set rollout to 100% of users, and press save.
Back in your app, set up the base page.js file to check the flag:
Once done, you can run the site by running the npm run dev command. Go to your site running on your localhost and you might see "Not yet." Refresh and, if your feature flag is set up correctly, it changes to "PostHog is awesome." This means the flag works, but the app isn’t loading the feature flag value right away. To solve this, we can bootstrap the feature flags. 
Getting feature flags with Next.js middleware
To load flags right away and prevent "flickering", we can bootstrap them. This means getting the flag data from PostHog in between a request and page load. We can do this with Next.js middleware.
To set this up, create a middleware.js file. Create it in your base directory (not in the app directory). In this file, start by handling the request and setting up the matcher config with a negative lookahead to remove non-page requests.
Next, use your project API key (found in your project settings) to get your cookie. With the cookie, check for the distinct ID or create one if it isn’t found.
With the distinct ID, we can make an API request to evaluate flags for the user. This requires making a request to https://us.i.posthog.com/decide?v=3 (or https://eu.i.posthog.com/decide?v=3) with the api_key and distinct_id. 
Note: We use the API because Vercel edge middleware (which optimizes the speed of this request in your app) has a limited number of packages and
posthog-nodeisn’t one of them.
With the flag data, we can create an object to use the client-side bootstrapping and then pass this value as a cookie. Altogether, this looks like this:
Next, we will handle this data in the client and use it to bootstrap flags.
Bootstrapping feature flags
With the feature flag data passed to the frontend, we need the code to handle it there. First, install the cookie-cutter package to handle cookies.
Once installed, go back to providers.js to get the cookie data and bootstrap it into our PostHog initialization.
Finally, rework our page.js file to check that the flag isn’t undefined before using its value.
Once you restart your app to reinitialize PostHog, your flags load instantly.
When to bootstrap vs use cookies? You might think to use cookies directly, but they don’t update when properties change. You should use bootstrapping and the
posthog-jslibrary when you want flags to update over time in the user session. For example, when they do something that updates properties like opt into a beta.
Note: When using Next.js v12, there's a bug in 12.3.0 that can cause session recording not to work when using this method. To fix this, use a version of Next.js v12 at least at v12.3.4.
Further reading
- How to bootstrap feature flags in React and Express
- Testing frontend feature flags with React, Jest, and PostHog
- How to evaluate and update feature flags with the PostHog API
