Skip to main content

DevCycle Flutter Client SDK

The Flutter Client SDK for DevCycle! This SDK uses our Client SDK APIs to perform all user segmentation and bucketing for the SDK, providing fast response times using our globally distributed edge workers all around the world.

The SDK is available as a package on Pub. It is also open source and can be viewed on GitHub.

Pub GitHub


This version of the DevCycle Flutter Client SDK requires a minimum of version of Flutter 2, iOS 12 and Android API Version 21.


Checkout DevCycle Flutter Client SDK Releases for the latest versions of the SDKs.

Flutter CLI

The SDK can be installed into your Flutter project by running flutter pub add devcycle_flutter_client_sdk.

Pub Spec

The SDK can be installed into your Flutter project by adding the following to your pubspec.yaml:

devcycle_flutter_client_sdk: ^1.0.0

Then, run flutter pub get.


Initializing the SDK

We recommend initializing the SDK once and pass around the client instance around in your app. Using the builder pattern we can initialize the DevCycle SDK by providing the DVCUser and DevCycle mobile environment key:

import 'package:devcycle_flutter_client_sdk/devcycle_flutter_client_sdk.dart';

void main() {
runApp(const MyApp());

class MyApp extends StatefulWidget {
const MyApp({super.key});

State<MyApp> createState() => _MyAppState();

class _MyAppState extends State<MyApp> {
static final user = DVCUserBuilder().isAnonymous(true).build();
static final options = DVCOptionsBuilder().logLevel(LogLevel.error).build();

final _dvcClient = DVCClientBuilder()

void initState() {


The user object may specify a userId for a given User. If the userId is not specified, the User is considered to be anonymous.

DVC Client Builder

The DVCClient can be built using the following methods:

DVC ClientBuilder class

environmentKeyStringDevCycle environment key
userDVCUserDevCycle user object
optionsDVCOptionsDevCycle options object

DVC User Builder

The DVC user can be built using the following methods:

DVC UserBuilder class

userIdStringUnique user ID
isAnonymousBoolBoolean to indicate if the user is anonymous
emailStringUser's email
nameStringUser's name
languageStringUser's language
countryStringUser's country
customData[String: Any]Key/value map of properties to be used for targeting
privateCustomData[String: Any]Key/value map of properties to be used for targeting. Private properties will not be included in event logging.

DVC Options Builder

The SDK exposes various initialization options which can be used by passing a DVCOptions object to the options method of DVCClient.builder():

DVC OptionsBuilder class

flushEventsIntervalMsInt10000Controls the interval between flushing events to the DevCycle servers in milliseconds, defaults to 10 seconds.
disableEventLoggingBoolfalseDisables logging of SDK generated events (e.g. aggVariableEvaluated, aggVariableDefaulted) to DevCycle.
logLevelLogLevelerrorSet log level of the default logger. Defaults to error
enableEdgeDBBooleanfalseEnables the usage of EdgeDB for DevCycle that syncs User Data to DevCycle.
configCacheTTLInt604800000The maximum allowed age of a cached config in milliseconds, defaults to 7 days
disableConfigCacheBoolfalseDisable the use of cached configs

Notifying when DevCycle features are available

In the initialize call there is an optional onInitialized parameter you can use to determine when your features have been loaded:

final _dvcClient = DVCClientBuilder()
.onInitialized((error) {

Using Variable Values

To get values from your Features, the variable() method is used to fetch variable values using the variable's identifier key coupled with a default value. The default value can be of type string, boolean, number, or JSONObject:

final boolVariable = _dvcClient.variable(key: "bool_key", defaultValue: false);
final strVariable = _dvcClient.variable(key: "string_key", defaultValue: "default");
final numVariable = _dvcClient.variable(key: "num_key", defaultValue: 4);
final jsonVariable = _dvcClient.variable(key: "json_key", defaultValue: { "key": "value" });

To grab the value, there is a property on the object returned to grab the value:

if (boolVariable.value == true) {
// Run Feature Flag Code
} else {
// Run Default Code

The Variable object also contains the following params:

- `key`: the key indentifier for the Variable
- `type`: the type of the Variable, one of: `String` / `Boolean` / `Number` / `JSON`
- `value`: the Variable's value
- `defaultValue`: the Variable's default value
- `isDefaulted`: if the Variable is using the `defaultValue`
- `evalReason`: evaluation reason for why the variable was bucketed into its value

If the value is not ready, it will return the default value passed in the creation of the variable.

Variable Updates

Variable values update whenever identifyUser() or resetUser() are called, or when the project configuration changes (to learn more, visit our Realtime Updates page).

To listen for variable updates, the onUpdate() method can be used. Please note, a strong reference to the variable is needed for onUpdate to be triggered.

final variable = await _dvcClient.variable('my-variable', 'Default Value');
variable?.onUpdate((updatedVariable) {
// Variable value updated updatedVariable.value

Getting All Features / Variables

Get All Features

To get all the Features returned in the config:

Map<String, DVCFeature> features = await _dvcClient.allFeatures();

If the SDK has not finished initializing, these methods will return an empty object.

Get All Variables

To get all the variables returned in the config:

Map<String, DVCVariable> variables = await _dvcClient.allVariables();

If the SDK has not finished initializing, these methods will return an empty object.

Identifying User

To identify a different user, or the same user passed into the initialize method with more attributes, build a DVCUser object and pass it into identifyUser:

DVCUser user = DVCUserBuilder()
.email('[email protected]')
.name("My Name")
.customData({ "customkey": "customValue" })
.privateCustomData({ "customkey2": "customValue2" })


To wait on Variables that will be returned from the identify call, you can pass in a callback:

_dvcClient.identifyUser(user, (error, variables) => {
// Error or Variables for the Identified User

If error exists the called the user's configuration will not be updated and previous user's data will persist.

Reset User

To reset the user into an anonymous user, resetUser will reset to the anonymous user created before or will create one with an anonymous user_id.


To wait on the Variables of the anonymous user, you can pass in a callback:

_dvcClient.resetUser((error, variables) => {
// Error or Variables for Anonymous User

If error exists is called the user's configuration will not be updated and previous user's data will persist.

Tracking Events

To track events, pass in an object with at least a type key:

DVCEvent event = DVCEventBuilder()
.metaData({ 'key': 'value' })


The SDK will flush events every 10s or flushEventsMS specified in the options. To manually flush events, call:


An Error callback can also be passed to this method, that will be triggered if there is a non-recoverable failure when flushing events.

_dvcClient.flushEvents(([error]) => {
// Error or null for Flushing Events


EdgeDB allows you to save user data to our EdgeDB storage so that you don't have to pass in all the user data every time you identify a user. Read more about EdgeDB.

To get started, contact us at [email protected] to enable EdgeDB for your project.

Once you have EdgeDB enabled in your project, pass in the enableEdgeDB option to turn on EdgeDB mode for the SDK:

DVCUser user = DVCUserBuilder()
.email('[email protected]')
.customData({ "amountSpent": 50 })

DVCOptions options = DVCOptionsBuilder()

This will send a request to our EdgeDB API to save the custom data under the user test-user.

In the example, amountSpent is associated to the user test-user. In your next identify call for the same userId, you may omit any of the data you've sent already as it will be pulled from the EdgeDB storage when segmenting to experiments and features.