JavaScript
This category contains topics explaining how to configure Bucketeer's JavaScript SDK.
Bucketeer JavaScript SDK has reached the production stage, offering you a stable and reliable experience.
Getting started
Before starting, ensure that you follow the Getting Started guide.
Installing dependency
Install the dependency in your application.
- npm
- Yarn
npm install @bucketeer/js-client-sdk
yarn add @bucketeer/js-client-sdk
Importing client
Import the Bucketeer client into your application code.
- JavaScript
import { BKTClient, getBKTClient, defineBKTConfig, defineBKTUser, initializeBKTClient } from '@bucketeer/js-client-sdk';
Configuring client
Configure the SDK config and user configuration.
The featureTag setting is the tag you configure when creating a Feature Flag. It will evaluate all the Feature Flags in the environment when it is not configured.
We strongly recommend using tags to speed up the evaluation process and reduce the cache size in the client.
All the settings in the example below are required.
- JavaScript
const config = defineBKTConfig({
apiKey: 'YOUR_API_KEY',
apiEndpoint: 'YOUR_API_URL',
featureTag: 'YOUR_FEATURE_TAG', // Optional
appVersion: 'YOUR_APP_VERSION',
});
const user = defineBKTUser({
id: 'USER_ID'
});
Depending on your use, you may want to change the optional configurations available.
- pollingInterval - Minimum 60 seconds. Default is 10 minutes (In Milliseconds)
- eventsFlushInterval - Default is 30 seconds (In Milliseconds)
- eventsMaxQueueSize - Default is 50 events
- storageKeyPrefix - Default is empty
- userAgent - Default is
window.navigator.userAgent
- fetch - Default is
window.fetch
The Bucketeer SDK doesn't save the user data. The Application must save and set it when initializing the client SDK.
Initializing client
Initialize the client by passing the configurations in the previous step.
- JavaScript
await initializeBKTClient(config, user);
const client = getBKTClient();
The initialize process default timeout is 5 seconds.
Once initialization is finished, all the requests in the SDK use a timeout of 30 seconds.
If you want to use the feature flag on Splash or Main views, the SDK cache may be old or not exist and may not have enough time to fetch the variations from the Bucketeer server. In this case, we recommend using the Promise
returned from the initialize method. The Promise rejects with BKTException
when something goes wrong.
During the initialization process, errors are not related to the initialization itself. Instead, they arise from a timeout request, indicating the variations data from the server weren't received. Therefore, the SDK will work as usual and update the variations in the next polling request.
- JavaScript
const timeout = 2000 // Default is 5 seconds (In milliseconds)
const initialFetchPromise = initializeBKTClient(config, user, timeout);
initialFetchPromise
.then(() => {
const client = getBKTClient();
const showNewFeature = client?.booleanVariation('YOUR_FEATURE_FLAG_ID', false);
if (showNewFeature) {
// The Application code to show the new feature
} else {
// The code to run when the feature is off
}
})
.catch((error) => {
// Handle the error when there is no cache or the cache is not updated
});
Polling
The initialize process starts polling right away the latest evaluations from the Bucketeer server in the background using the interval pollingInterval
configuration. JavaScript SDK does not support Background fetch.
Polling retry behavior
The Bucketeer SDK regularly polls the latest evaluations from the server based on the pollingInterval parameter. By default, the pollingInterval
is set to 10 minutes, but you can adjust it to suit your needs.
If a polling request fails, the SDK initiates a retry procedure. The SDK attempts a new polling request every minute up to 5 times. If all five retry attempts fail, the SDK sends a new polling request once the pollingInterval
time elapses. The table below shows this scenario:
Polling Time | Retry Time | Request Status |
---|---|---|
10:00 | - | Fail |
- | 10:01 | Fail |
- | 10:02 | Fail |
- | 10:03 | Fail |
- | 10:04 | Fail |
- | 10:05 | Fail |
10:10 | - | Successful |
The polling counter, which uses the pollingInterval
information, resets in case of a successful retry. The table below shows the described scenario.
Polling Time | Retry Time | Request status |
---|---|---|
10:00 | - | Fail |
- | 10:01 | Successful |
10:11 | - | Successful |
Handling exceptions
While most of the time error is handled internally, some methods throw BKTException
when something goes wrong.
Those methods are:
initializeBKTClient()
BKTClient#fetchEvaluations()
BKTClient#flush()
These methods return Promise
and might reject with `BKTException``, so you should make sure to catch the error.
Supported features
Evaluating user
The variation method determines whether or not a feature flag is enabled for a specific user.
To check which variation a specific user will receive, you can use the client like below.
- JavaScript
const showNewFeature = client?.booleanVariation('YOUR_FEATURE_FLAG_ID', false);
if (showNewFeature) {
// The Application code to show the new feature
} else {
// The code to run when the feature is off
}
The variation method will return the default value if the feature flag is missing in the SDK.
Variation types
The Bucketeer SDK supports the following variation types.
The jsonVariation
interface is deprecated. Please use the objectVariation
instead.
- JavaScript
booleanVariation(featureId: string, defaultValue: boolean): Promise<boolean>;
stringVariation(featureId: string, defaultValue: string): Promise<string>;
numberVariation(featureId: string, defaultValue: number): Promise<number>;
// The returned value will be either a BKTJsonObject or a BKTJsonArray. If no result is found, it will return the provided `defaultValue`, which can be of any type within `BKTValue`.
objectVariation(featureId: string, defaultValue: BKTValue):Promise<BKTValue>;
Getting evaluation details
The following methods will return the evaluation details for a specific feature flag. If the feature flag is missing in the SDK's cache, the variable reason
value will be CLIENT
, which means the default value was returned.
This is useful if you use another A/B Test solution with Bucketeer and need to know the variation name, reason, and other information.
The evaluationDetails
interface is deprecated. Please use the following interfaces.
Interfaces
- JavaScript
interface BKTClient {
booleanVariationDetails: (
featureId: string,
defaultValue: boolean
) => BKTEvaluationDetails<boolean>;
stringVariationDetails: (
featureId: string,
defaultValue: string
) => BKTEvaluationDetails<string>;
numberVariationDetails: (
featureId: string,
defaultValue: number
) => BKTEvaluationDetails<number>;
objectVariationDetails: (
featureId: string,
defaultValue: BKTValue
) => BKTEvaluationDetails<BKTValue>;
}
Object
- JavaScript
export interface BKTEvaluationDetails<T extends BKTValue> {
readonly featureId: string; // The ID of the feature flag.
readonly featureVersion: number; // The version of the feature flag.
readonly userId: string; // The ID of the user being evaluated.
readonly variationId: string; // The ID of the assigned variation.
readonly variationName: string; // The name of the assigned variation.
readonly variationValue: T; // The value of the assigned variation.
readonly reason:
| 'TARGET' // Evaluated using individual targeting.
| 'RULE' // Evaluated using a custom rule.
| 'DEFAULT' // Evaluated using the default strategy.
| 'CLIENT' // The flag is missing in the cache; the default value was returned.
| 'OFF_VARIATION' // Evaluated using the off variation.
| 'PREREQUISITE'; // Evaluated using a prerequisite.
}
Usage
- JavaScript
const showNewFeature = client?.booleanVariationDetails("YOUR_FEATURE_FLAG_ID", false);
if (showNewFeature.variationValue) {
// The Application code to show the new feature
} else {
// The code to run when the feature is off
}
Updating user evaluations
Depending on the use case, you may need to ensure the evaluations in the SDK are up to date before requesting the variation.
The fetch method uses the following parameter. Make sure to wait for its completion.
- Timeout - Default is 30 seconds (In milliseconds)
- JavaScript
const timeout = 5000; // Optional. Default is 30 seconds (In milliseconds)
await client?.fetchEvaluations(timeout);
Reporting custom events
This method lets you save user actions in your application as events. You can connect these events to metrics in the experiments console UI.
In addition, you can pass a double value to the goal event. These values will sum and show as Value total
on the experiments console UI. This is useful if you have a goal event for tracking how much a user spent on your application buying items, etc.
- JavaScript
client?.track("YOUR_GOAL_ID", 10.50);
Flushing events
This method will send all pending analytics events to the Bucketeer server as soon as possible. This process is asynchronous, so it returns before it is complete.
- JavaScript
await client?.flush();
In regular use, you don't need to call the flush method because the events are sent every 30 seconds in the background.
User attributes configuration
This feature will give you robust and granular control over what users can see on your application. You can add rules using these attributes on the console UI's feature flag's targeting tab. See more.
- JavaScript
const attributes = {
app_version: '1.0.0',
os_version: '11.0.0',
device_model: 'pixel-5',
language: 'english',
genre: 'female'
};
const user = defineBKTUser({
id: 'USER_ID',
customAttributes: attributes
});
await initializeBKTClient(config, user);
Updating user attributes
This method will update all the current user attributes. This is useful in case the user attributes update dynamically on the application after initializing the SDK.
- JavaScript
const attributes = {
app_version: '1.0.0',
os_version: '11.0.0',
device_model: 'pixel-5',
language: 'english',
genre: 'female'
};
client?.updateUserAttributes(attributes);
This updating method will override the current data.
Getting user information
This method will return the current user configured in the SDK. This is useful when you want to check the current user id and attributes before updating them through updateUserAttributes.
- JavaScript
const user = client?.currentUser();
Listening to evaluation updates
The SDK can notify when the evaluation is updated. The listener can detect both automatic polling and manual fetching.
- JavaScript
// Returned value is used when you want to remove listener
const key = client?.addEvaluationUpdateListener(() => {
const showNewFeature = client?.booleanVariation("YOUR_FEATURE_FLAG_ID", false)
if (showNewFeature) {
// The Application code to show the new feature
} else {
// The code to run when the feature is off
}
});
// Remove a listener associated with the key
client?.removeEvaluationUpdateListener(key);
// Remove all listeners
client?.clearEvaluationUpdateListeners();
Destroying client
There are cases you might want to switch the user ID or reduce resources when the application is in the background.
For those cases, you can call the destroy function, which will clear the client instance.
- JavaScript
import { destroyBKTClient } from '@bucketeer/js-client-sdk';
destroyBKTClient();
If you want to switch the user ID, please call the flush interface before calling the destroy, so that all the pending events can be sent before clearing the client instance, then call the initialize interface with the new user information.