Official documentation for Amplitude Experiment's Client-side JavaScript SDK implementation.
Install the Experiment JavaScript Client SDK with one of the three following methods:
# Install Experiment SDK only
npm install --save @amplitude/experiment-js-client
# Or install Unified SDK to get access to all Amplitude products
npm install @amplitude/unified
# Install Experiment SDK only
yarn add @amplitude/experiment-js-client
# Or install Unified SDK to get access to all Amplitude products
yarn add @amplitude/unified
<script src="https://unpkg.com/@amplitude/experiment-js-client@1.15.5/dist/experiment.umd.js"></script>
<script>
    // TODO: Replace DEPLOYMENT_KEY with your own deployment key.
    // If you're using a 3rd party for analytics, configure an exposure
    // tracking provider.
    window.experiment = Experiment.initializeWithAmplitudeAnalytics('DEPLOYMENT_KEY');
</script>
import { Experiment } from '@amplitude/experiment-js-client';
// (1) Initialize the experiment client with Amplitude Analytics.
const experiment = Experiment.initializeWithAmplitudeAnalytics(
    'DEPLOYMENT_KEY'
);
// (2) Fetch variants and await the promise result.
await experiment.fetch();
// (3) Lookup a flag's variant.
const variant = experiment.variant('FLAG_KEY');
if (variant.value === 'on') {
    // Flag is on
} else {
    // Flag is off
}
import { initAll, experiment } from '@amplitude/unified';
// (1) Initialize the Unified SDK with your API key
// Note: if deploymentKey is not provided, it will fall back to the api key
initAll('YOUR_API_KEY', {
    experiment: {
        deploymentKey: 'DEPLOYMENT_KEY'
    }
});
// (2) Fetch variants and await the promise result.
await experiment()?.fetch();
// (3) Lookup a flag's variant.
const variant = experiment()?.variant('FLAG_KEY');
if (variant.value === 'on') {
    // Flag is on
} else {
    // Flag is off
}
import { Experiment } from '@amplitude/experiment-js-client';
// (1) Initialize the experiment client and implement a
//     custom exposure tracking provider.
const experiment = Experiment.initialize(
    'DEPLOYMENT_KEY',
    {
        exposureTrackingProvider: {
            track: (exposure) => {
                // TODO: Implement exposure tracking
                // analytics.track('$exposure', exposure)
            }
        }
    }
);
// (2) Fetch variants with the user and await the promise result.
const user = {
    user_id: 'user@company.com',
    device_id: 'abcdefg',
    user_properties: {
        premium: true,
    },
}
await experiment.fetch(user);
// (3) Lookup a flag's variant.
const variant = experiment.variant('FLAG_KEY');
if (variant.value === 'on') {
    // Flag is on
} else {
    // Flag is off
}
Initialize the SDK in your application on startup. The deployment key argument you pass into the apiKey parameter must live in the same Amplitude project to which you send events.
initializeWithAmplitudeAnalytics(apiKey: string, config?: ExperimentConfig): ExperimentClient
initAll(apiKey: string, config?: UnifiedConfig): void
initialize(apiKey: string, config?: ExperimentConfig): ExperimentClient
| Paramater | Description | 
|---|---|
| apikey | Required. The deployment key which authorizes fetch requests and determines which flags to evaluate for the user | 
| config | The client configuration to customize SDK client behavior. | 
The initializer returns a singleton instance, so subsequent initializations for the same instance name always return the initial instance. To create multiple instances, use the instanceName configuration.
import { Experiment } from '@amplitude/experiment-js-client';
const experiment = initializeWithAmplitudeAnalytics('DEPLOYMENT_KEY');
import { initAll, experiment } from '@amplitude/unified';
// Note: if deploymentKey is not provided, it will fall back to the api key
initAll('YOUR_API_KEY', {
    experiment: {
        deploymentKey: 'DEPLOYMENT_KEY'
    }
});
import { Experiment } from '@amplitude/experiment-js-client';
const experiment = Experiment.initialize(
    'DEPLOYMENT_KEY',
    {
        exposureTrackingProvider: {
            track: (exposure) => {
                // TODO: Implement exposure tracking
                // analytics.track('$exposure', exposure)
            }
        }
    }
);
Configure the SDK client once during initialization.
Configuration
            
    
 
Name 
Description 
Default Value 
 
debugEnable additional debug logging within the SDK. Should be set to false in production builds. 
false
 
fallbackVariantThe default variant to fall back if a variant for the provided key doesn't exist. 
{}
 
initialVariantsAn initial set of variants to access. This field is valuable for bootstrapping the client SDK with values rendered by the server using server-side rendering (SSR). 
{}
 
sourceThe primary source of variants. Set the value to  
Source.InitialVariants and configured initialVariants to bootstrap the SDK for SSR or testing purposes.Source.LocalStorage
 
serverZoneSelect the Amplitude data center to get flags and variants from,  
us or eu.us
 
serverUrlThe host to fetch remote evaluation variants from. For hitting the EU data center, use  
serverZone.https://api.lab.amplitude.com
 
flagsServerUrlThe host to fetch local evaluation flags from. For hitting the EU data center, use  
serverZone.https://flag.lab.amplitude.com
 
fetchTimeoutMillisThe timeout for fetching variants in milliseconds. 
10000
 
retryFetchOnFailureWhether to retry variant fetches in the background if the request doesn't succeed. 
true
 
automaticExposureTrackingIf true, calling  
variant() will track an exposure event through the configured exposureTrackingProvider. If no exposure tracking provider is set, this configuration option does nothing.true
 
fetchOnStartIf true or undefined, always fetch remote evaluation variants on start. If false, never fetch on start. 
true
 
pollOnStartPoll for local evaluation flag configuration updates once per minute on start. 
true
 
automaticFetchOnAmplitudeIdentityChangeOnly matters if you use the  
initializeWithAmplitudeAnalytics initialization function to seamlessly integrate with the Amplitude Analytics SDK. If true any change to the user ID, device ID or user properties from analytics will trigger the experiment SDK to fetch variants and update it's cache.false
 
userProviderAn interface used to provide the user object to  
fetch() when called.null
 
exposureTrackingProviderImplement and configure this interface to track exposure events through the experiment SDK, either automatically or explicitly. 
null
 
instanceNameCustom instance name for experiment SDK instance. The value of this field is case-sensitive. 
null
 
initialFlagsA JSON string representing an initial array of flag configurations to use for local evaluation. 
undefined
 
httpClient(Advanced) Use your own HTTP client implementation to handle network requests made by the SDK. 
Default HTTP client 
If you use either Amplitude or Segment Analytics SDKs to track events into Amplitude, set up an integration on initialization. Integrations automatically implement provider interfaces to enable a more streamlined developer experience by making it easier to manage user identity and track exposures events.
Using the integration initializer configures implementations of the user provider and exposure tracking provider interfaces to pull user data from the Amplitude Analytics SDK and track exposure events. Supported Versions All versions of the next-generation Amplitude analytics Browser SDK support this integration.Amplitude integration
            
    import * as amplitude from '@amplitude/analytics-browser';
import { Experiment } from '@amplitude/experiment-js-client';
amplitude.init('API_KEY');
const experiment = Experiment.initializeWithAmplitudeAnalytics('DEPLOYMENT_KEY');
 
Legacy Analytics SDK Version 
Experiment SDK Version 
 
8.18.1+1.4.1+
When fetching variants, pass the segment anonymous ID and user ID for the device ID and user ID, respectively.Segment integration
            
    analytics.ready(() => {
    const experiment =  Experiment.initialize('DEPLOYMENT_KEY', {
        exposureTrackingProvider: {
            track: (exposure) => {
                analytics.track('$exposure', exposure)
            }
        }
    });
});
await experiment.fetch({
    user_id: analytics.user().id(),
    device_id: analytics.user().analyticsId(),
});
In accordance with your event forwarding settings to Amplitude, the event type mParticle integration
            
    user_id and device_id depend on your specific configuration.
Other may not be the right classification. Make sure that the event type you use is forwarded to Amplitude in destination settings.const identityRequest = {
    userIdentities: {
        email: "joe_slow@gmail.com",
        customerid: "abcdxyz"
    }
};
mParticle.Identity.login(identityRequest, () => {
    console.log("Identity callback");
});
const experiment = Experiment.initialize("DEPLOYMENT_KEY", {
    exposureTrackingProvider: {
        track: (exposure) => {
            window.mParticle.logEvent('$exposure', window.mParticle.EventType.Other, exposure);
        }
    },
    userProvider: {
        getUser: () => {
            const user_id = window.mParticle.Identity.getCurrentUser().getUserIdentities().userIdentities.customerid;
            const device_id = window.mParticle.getDeviceId();
            if (user_id != null) {
                return {
                    user_id: user_id,
                    device_id: device_id
                };
                else
                    return {
                        device_id: device_id
                    };
            }
        }
    }
});
Fetches variants for a user and store the results in the client for fast access. This function remote evaluates the user for flags associated with the deployment used to initialize the SDK client.
fetch() whenever the user state changes in a meaningful way. For example, if the user logs in and receives a user ID, or has a user property set which may effect flag or experiment targeting rules.
Pass new user properties explicitly to fetch() instead of relying on user enrichment prior to remote evaluation. This is because user properties that are synced remotely through a separate system have no timing guarantees with respect to fetch()--for example, a race.
fetch(user?: ExperimentUser, options?: FetchOptions): Promise<Client>
| Parameter | Requirement | Description | 
|---|---|---|
| user | optional | Explicit user information to pass with the request to evaluate. This user information is merged with user information provided from integrations via the user provider, preferring properties passed explicitly to fetch()over provided properties. | 
| options | optional | Explicit flag keys to fetch. | 
Groups must either be included in the user sent with the fetch request (recommended), or identified with the user via a group identify call from the Group Identify API or via  To pass freeform group properties, see this example:Account-level bucketing and analysis (v1.5.6+)
            
    setGroup() from an analytics SDK.await fetch({
    user_id: 'user@company.com',
    device_id: 'abcdefg',
    user_properties: {
        'premium': true,
    },
    groups: {'org name': ['Amplitude']}
});
await fetch({
    user_id: 'user@company.com',
    device_id: 'abcdefg',
    user_properties: {
        'premium': true,
    },
    group_properties: {'org name': ['Amplitude']}
});
const user = {
    user_id: 'user@company.com',
    device_id: 'abcdefg',
    user_properties: {
        'premium': true,
    },
};
await experiment.fetch(user);
If you're using an integration or a custom user provider then you can fetch without inputting the user.
await experiment.fetch();
If fetch() times out (default 10 seconds) or fails for any reason, the SDK client will return and retry in the background with back-off. You may configure the timeout or disable retries in the configuration options when the SDK client is initialized.
start if you're using client-side local evaluation. If you're only using remote evaluation, call fetch instead of start.Start the SDK by getting flag configurations from the server and fetching remote evaluation variants for the user. The SDK is ready once the returned promise resolves.
start(user?: ExperimentUser): Promise<void>
| Parameter | Requirement | Description | 
|---|---|---|
| user | optional | Explicit user information to pass with the request to fetch variants. This user information is merged with user information provided from integrations via the user provider, preferring properties passed explicitly to fetch()over provided properties. Also sets the user in the SDK for reuse. | 
Call start() when your application is initializing, after user information is available to use to evaluate or fetch variants. The returned promise resolves after loading local evaluation flag configurations and fetching remote evaluation variants.
Configure the behavior of start() by setting fetchOnStart in the SDK configuration on initialization to improve performance based on the needs of your application.
fetchOnStart to false to avoid increased startup latency caused by remote evaluation.fetchOnStart to false and call fetch() and await the promise separately.
await experiment.start();
const user = {
    user_id: 'user@company.com',
    device_id: 'abcdefg',
    user_properties: {
        premium: true
    }
};
await experiment.start(user);
Access a variant for a flag or experiment from the SDK client's local store.
variant() will automatically track an exposure event through the tracking provider. To disable this functionality, configure automaticExposureTracking to be false, and track exposures manually using exposure().variant(key: string, fallback?: string | Variant): Variant
| Parameter | Requirement | Description | 
|---|---|---|
| key | required | The flag key to identify the flag or experiment to access the variant for. | 
| fallback | optional | The value to return if no variant was found for the given flagKey. | 
When determining which variant a user has been bucketed into, you'll want to compare the variant value to a well-known string.
const variant = experiment.variant('<FLAG_KEY>');
if (variant.value === 'on') {
    // Flag is on
} else {
    // Flag is off
}
payload field from the variant object after checking the variant's value.
const variant = experiment.variant('<FLAG_KEY>');
if (variant.value === 'on') {
    const payload = variant.payload;
}
A null variant value means that the user hasn't been bucketed into a variant. You may use the built in fallback parameter to provide a variant to return if the store doesn't contain a variant for the given flag key.
const variant = experiment.variant('<FLAG_KEY>', { value: 'control' });
if (variant.value === 'control') {
    // Control
} else if (variant.value === 'treatment') {
    // Treatment
}
Access all variants stored by the SDK client.
all(): Variants
Clear all variants in the cache and storage.
clear(): void
You can call clear after user logout to clear the variants in cache and storage.
experiment.clear();
Manually track an exposure event for the current variant of the given flag key through configured integration or custom exposure tracking provider. Generally used in conjunction with setting the automaticExposureTracking configuration optional to false.
exposure(key: string): void
| Parameter | Requirement | Description | 
|---|---|---|
| key | required | The flag key to identify the flag or experiment variant to track an exposure event for. | 
const variant = experiment.variant('<FLAG_KEY>');
// Do other things...
experiment.exposure('<FLAG_KEY>');
if (variant.value === 'control') {
    // Control
} else if (variant.value === 'treatment') {
    // Treatment
}
Provider implementations enable a more streamlined developer experience by making it easier to manage user identity and track exposures events.
The user provider is used by the SDK client to access the most up-to-date user information only when it's needed (for example, when fetch() is called). This provider is optional, but helps if you have a user information store already set up in your application. This way, you don't need to manage two separate user info stores in parallel, which may result in a divergent user state if the application user store is updated and experiment isn't (or via versa).
interface ExperimentUserProvider {
  getUser(): ExperimentUser;
}
To use your custom user provider, set the userProvider configuration option with an instance of your custom implementation on SDK initialization.
const experiment = Experiment.initialize('<DEPLOYMENT_KEY>', {
    userProvider: new CustomUserProvider(),
});
Implementing an exposure tracking provider is highly recommended. Exposure tracking increases the accuracy and reliability of experiment results and improves visibility into which flags and experiments a user is exposed to.
export interface ExposureTrackingProvider {
  track(exposure: Exposure): void;
}
The implementation of track() should track an event of type $exposure (a.k.a name) with two event properties, flag_key and variant, corresponding to the two fields on the Exposure object argument. Finally, the event tracked must eventually end up in Amplitude Analytics for the same project that the [deployment] used to initialize the SDK client lives within, and for the same user that variants were fetched for.
To use your custom user provider, set the exposureTrackingProvider configuration option with an instance of your custom implementation on SDK initialization.
const experiment = Experiment.initialize('<DEPLOYMENT_KEY>', {
    exposureTrackingProvider: new CustomExposureTrackingProvider(),
});
You may want to bootstrap the experiment client with an initial set of flags or variants when variants are obtained from an external source (for example, not from calling fetch() on the SDK client). Use cases include local evaluation, server-side rendering, or integration testing on specific variants.
To bootstrap the client with a predefined set of variants, set the flags and variants in the initialVariants configuration object, then set the source to Source.InitialVariants so that the SDK client prefers the bootstrapped variants over any previously fetched & stored variants for the same flags.
const experiment = Experiment.initialize('<DEPLOYMENT_KEY>', {
    // Map flag keys to variant objects. The variant object may either be
    // pre-evaluation (SSR) or input manually in for testing.
    initialVariants: {
        "<FLAG_KEY>": {
            "value": "<VARIANT>"
         }
    },
    source: Source.InitialVariants,
});
You may choose to bootstrap the SDK with an initial set of local evaluation flag configurations using the initialFlags configuration. These will be evaluated  when variant is called, unless an updated flag config or variant is loaded with start or fetch.
To download initial flags, use the evaluation flags API
const experiment = Experiment.initialize('<DEPLOYMENT_KEY>', {
    initialFlags: "<FLAGS_JSON>",
});
You can provide a custom HTTP client implementation to handle network requests made by the SDK. This is useful for environments with specific networking requirements or when you need to customize request handling.
export interface SimpleResponse {
  status: number;
  body: string;
}
export interface HttpClient {
  request(
    requestUrl: string,
    method: string,
    headers: Record<string, string>,
    data: string,
    timeoutMillis?: number,
  ): Promise<SimpleResponse>;
}
To use your custom HTTP client, set the httpClient configuration option with an instance of your implementation on SDK initialization.
const experiment = Experiment.initialize('<DEPLOYMENT_KEY>', {
    httpClient: new CustomHttpClient(),
});
June 4th, 2024
Need help? Contact Support
Visit Amplitude.com
Have a look at the Amplitude Blog
Learn more at Amplitude Academy
© 2025 Amplitude, Inc. All rights reserved. Amplitude is a registered trademark of Amplitude, Inc.