Skip to main content

Flagsmith React SDK

The JavaScript Library contains a bundled isomorphic library, allowing you to fetch flags in the server and hydrate your application with the resulting state.

Example applications for a variety of Next.js and SSR can be found here.

Example application using the Next.js app router can be found here.

Example applications for Svelte be found here.

An example application for Next.js middleware can be found here.

Installation

NPM

npm i flagsmith --save

Basic Usage

The SDK is initialised against a single environment. You can find your Client-side Environment Key in the Environment settings page.

Comparing SSR and client-side Flagsmith usage

The SDK is initialised and used in the same way as the JavaScript and React SDK. The main difference is that Flagsmith should be imported from flagsmith/isomorphic.

The main flow with Next.js and any JavaScript-based SSR can be as follows:

  1. Fetch the flags on the server, optionally passing an identity to flagsmith.init({})
  2. Pass the resulting state to the client with flagsmith.getState()
  3. Initialise flagsmith on the client with flagsmith.setState(state)

Example: Initialising the SDK with Next.js

Taking the above into account, the following examples fetch flags on the server and initialises Flagsmith with the state. Below is an example for the app router as well as the pages router.

App Router Example

// src/app/components/FeatureFlagProvider.tsx
"use client";

import { ReactNode, useRef } from "react";

import { FlagsmithProvider } from "flagsmith/react";
import { IState } from "flagsmith/types";
import { createFlagsmithInstance } from "flagsmith/isomorphic";

export const FeatureFlagProvider = ({
serverState,
children,
}: {
serverState: IState;
children: ReactNode;
}) => {
const flagsmithInstance = useRef(createFlagsmithInstance());
return (
<FlagsmithProvider flagsmith={flagsmithInstance.current} serverState={serverState}>
<>{children}</>
</FlagsmithProvider>
);
};


// src/app/layout.jsx
import { ReactNode } from "react";
import { FeatureFlagProvider } from './components/FeatureFlagProvider';
import flagsmith from "flagsmith/isomorphic";

export default async function RootLayout({
children,
}: Readonly<{
children: ReactNode;
}>) {
await flagsmith.init({
environmentID: "<YOUR_SERVERSIDE_ENVIRONMENT_KEY>",
// Add optional identity, etc.
});
const serverState = flagsmith.getState();

return (
<html lang="en">
<head>
<meta name="viewport" content="initial-scale=1, width=device-width" />
</head>
<body>
<FeatureFlagProvider serverState={serverState}>
{children}
</FeatureFlagProvider>
</body>
</html>
);
}

Pages Router Example

// src/pages/_app.jsx
import { FlagsmithProvider } from 'flagsmith/react';
import { createFlagsmithInstance } from 'flagsmith/isomorphic';
function MyApp({ Component, pageProps, flagsmithState }) {
const flagsmithRef = useRef(createFlagsmithInstance());
return (
<FlagsmithProvider flagsmith={flagsmithRef.current} serverState={flagsmithState}>
<Component {...pageProps} />
</FlagsmithProvider>
);
}

MyApp.getInitialProps = async () => {
const flagsmithSSR = createFlagsmithInstance();
await flagsmithSSR.init({
// fetches flags on the server
environmentID: '<YOUR_SERVERSIDE_ENVIRONMENT_KEY>',
identity: 'my_user_id', // optionaly specify the identity of the user to get their specific flags
});
return { flagsmithState: flagsmithSSR.getState() };
};

export default MyApp;

Client Component

'use client'; // Only required by the app router version.

import { useFlags } from 'flagsmith/react';

export function MyComponent() {
const flags = useFlags(['font_size'], ['example_trait']); // only causes re-render if specified flag values / traits change
return (
<div className="App">
font_size: {flags.font_size.value}
example_trait: {flags.example_trait}
</div>
);
}

From this point on, the SDK usage is the same as the React SDK Guide

Example: Flagsmith with Next.js middleware

The Flagsmith JS client includes flagsmith/next-middleware, it can be used just like the regular library within Next.js middleware.

// middleware.ts
import { NextResponse } from 'next/server';
import type { NextRequest } from 'next/server';
import flagsmith from 'flagsmith/next-middleware';

export async function middleware(request: NextRequest) {
const identity = request.cookies.get('user');

if (!identity) {
// redirect to homepage
return NextResponse.redirect(new URL('/', request.url));
}

await flagsmith.init({
environmentID: '<YOUR_SERVERSIDE_ENVIRONMENT_KEY>',
identity,
});

// Return a different URL based on a feature flag
if (flagsmith.hasFeature('beta')) {
return NextResponse.redirect(new URL(`/account-v2/`, request.url));
}

// Return a different URL based on a remote config
const theme = flagsmith.getValue('colour');
return NextResponse.redirect(new URL(`/account/${theme}`, request.url));
}

export const config = {
matcher: '/login',
};

Example: SSR without Next.js

The same can be accomplished without using Next.js.

Step 1: Initialising the SDK and passing the resulting state to the client.

await flagsmith.init({
// fetches flags on the server
environmentID: '<YOUR_SERVERSIDE_ENVIRONMENT_KEY>',
identity: 'my_user_id', // optionaly specify the identity of the user to get their specific flags
});
const state = flagsmith.getState(); // Pass this data to your client

Step 2: Initialising the SDK on the client.

flagsmith.setState(state);

Step 3: Optionally force the client to fetch a fresh set of flags

flagsmith.getFlags();

From that point the SDK usage is the same as the JavaScript SDK Guide