Requires PHP 8.3 and above.
composer require pablojoan/feature
use PabloJoan\Feature\Features; // Import the namespace.
$featureConfigs = [
'foo' => [
'variants' => [
'variant1' => 100, //100% chance this variable will be chosen
'variant2' => 0 //0% chance this variable will be chosen
]
],
'bar' => [
'variants' => [
'variant1' => 25, //25% chance this variable will be chosen
'variant2' => 25, //25% chance this variable will be chosen
'variant3' => 50 //50% chance this variable will be chosen
],
'bucketing' => 'id' //same id string will always return the same variant
]
];
$features = new Features($featureConfigs);
$features->isEnabled(featureName: 'foo'); // true
$features->getEnabledVariant(featureName: 'foo'); // 'variant1'
For a quick summary and common use cases, please read the rest of this README.
Feature flagging API used for operational rampups and A/B testing.
The Feature API is how we selectively enable and disable features at a very fine grain as well as enabling features for a percentage of users for operational ramp-ups and for A/B tests. A feature can be completely enabled, completely disabled, or something in between and can comprise a number of related variants.
The two main API entry points are:
$features->isEnabled(featureName: 'my_feature')
which returns true when my_feature
is enabled and, for multi-variant features:
$features->getEnabledVariant(featureName: 'my_feature')
which returns the name of the particular variant which should be used.
The single argument to each of these methods is the name of the feature to test.
A typical use of $features->isEnabled
for a single-variant feature
would look something like this:
if ($features->isEnabled(featureName: 'my_feature')) {
// do stuff
}
For a multi-variant feature, we can determine the appropriate code to run for each variant with something like this:
switch ($features->getEnabledVariant(featureName: 'my_feature')) {
case 'foo':
// do stuff appropriate for the 'foo' variant
break;
case 'bar':
// do stuff appropriate for the 'bar' variant
break;
}
If a feature is bucketed by id, then we pass the id string to
$features->isEnabled
and $features->getEnabledVariant
as a second parameter
$isMyFeatureEnabled = $features->isEnabled(
featureName: 'my_feature',
id: 'unique_id_string'
);
$variant = $features->getEnabledVariant(
featureName: 'my_feature',
id: 'unique_id_string'
);
There are a number of common configurations so before I explain the complete syntax of the feature configuration stanzas, here are some of the more common cases along with the most concise way to write the configuration.
$server_config['foo'] = ['variants' => ['enabled' => 100]];
$server_config['foo'] = ['variants' => ['enabled' => 0]];
$server_config['foo'] = ['variants' => ['blue_background' => 100]];
$server_config['foo'] = ['variants' => ['enabled' => 1]];
$server_config['foo'] = [
'variants' => [
'blue_background' => 1,
'orange_background' => 1,
'pink_background' => 1,
],
];
$server_config['foo'] = [
'variants' => ['enabled' => 10]
];
$server_config['foo'] = [
'variants' => ['enabled' => 1],
'bucketing' => 'random'
];
$server_config['foo'] = [
'variants' => ['enabled' => 40],
'bucketing' => 'id'
];
$server_config['foo'] = ['variants' => ['enabled' => 50]];
Multi-variant feature in A/B test with 20% of users seeing each variant (and 40% left in control group).
$server_config['foo'] = [
'variants' => [
'blue_background' => 20,
'orange_background' => 20,
'pink_background' => 20
],
];
Each feature’s config stanza controls when the feature is enabled and what variant should be used when it is.
The value of a feature config stanza is an array with a number of special
keys, the most important of which is 'variants'
.
The value of the 'variants'
property an array whose keys are names of variants
and whose values are the percentage of requests that should see each variant.
The remaining feature config property is 'bucketing'
. Bucketing specifies
how users are bucketed when a feature is enabled for only a percentage of users.
The default value, 'random'
, causes each request to be bucketed independently,
meaning that the same user will be in different buckets on different requests.
This is typically used for features that should have no user-visible effects
but where we want to ramp up something like the switch from master to shards
or a new version of JS code.
The bucketing value 'id'
, causes bucketing to be based on the given id.
There are a few ways to misuse the Feature API or misconfigure a feature that may be detected. (Some of these are not currently detected but may be in the future.)
-
Setting the percentage value of a variant in
'variants'
to a value less than 0 or greater than 100. -
Setting
'variants'
such that the sum of the variant percentages is greater than 100. -
Setting
'variants'
to a non-array value. -
Setting
'bucketing'
to any value that is not'id'
or'random'
.