Zumba Swivel is a library that allows PHP applications to manage features to multiple users via buckets. It consists with 10 buckets, allowing the same code have up to 10 different behaviors.
This plugin is a bridge between CakePHP and Swivel. It provides a helper, component and behavior classes to be used in your CakePHP application.
You can install Swivel Cake into your project using composer.
For existing applications you can add the following to your composer.json
file:
"require": {
"zumba/swivel-cake": "1.*"
}
And run php composer.phar update
After installing, you should tell your application to load the plugin:
CakePlugin::load('Swivel', ['bootstrap' => true]);
The plugin has default configurations and is ready to use. However, you can customize some of the configurations.
Configuration | Default Value | Description |
---|---|---|
Cookie.enabled | true |
If cookie should be set at all |
Cookie.name | Swivel_Bucket |
Cookie name used to store the client bucket number. |
Cookie.expire | 0 |
Expiration, in seconds, of the cookie. Setting 0 means a session cookie. |
Cookie.path | / |
Cookie's path. |
Cookie.domain | env('HTTP_HOST') |
Domain name used for the cookie. |
Cookie.secure | false |
If cookie can be only used in secure transmissions, ie. HTTPS |
Cookie.httpOnly | false |
If cookie can be accessed via other sources other than HTTP, ie. JavaScript |
BucketIndex | null |
Defines the user's bucket. Leaving null it will auto-generate a number between 1 and 10. |
LoaderAlias | SwivelManager |
Name that will be used to store the instance on Cake's ClassRegistry . |
Logger | null |
Instance to receive the logs. Setting to null will make the log be discarded. |
Metrics | null |
Metrics instance. |
ModelAlias | Swivel.SwivelFeature |
Model name that will provide the swivel mapping configuration. |
To set custom configurations, you need to create a file in APP/Config/swivel.php
and set the fields you
want to override. Here is the default configuration file:
<?php
$config = [
'Swivel' => [
'Cookie' => [
'enabled' => true,
'name' => 'Swivel_Bucket',
'expire' => 0,
'path' => '/',
'domain' => env('HTTP_HOST'),
'secure' => false,
'httpOnly' => false
],
'BucketIndex' => null,
'LoaderAlias' => 'SwivelManager',
'Logger' => null,
'Metrics' => null,
'ModelAlias' => 'Swivel.SwivelFeature',
]
];
Let's say for an example that you want to reserve one bucket for your testing and give the other 9 buckets to your customers, you can do something like this:
<?php
// Saving bucket 1 for internal testing
$bucketIndex = isset($_COOKIE['Swivel_Bucket']) ? $_COOKIE['Swivel_Bucket'] : mt_rand(2, 10);
$config = [
'Swivel' => [
'BucketIndex' => $bucketIndex
]
];
In order to Swivel to work, you need to specify which features are enabled for each bucket.
The default behavior from swivel-cake is to load the features from database. This is done via
the built-in class Swivel.SwivelFeature
. This class However expects the table swivel_buckets
to exist in your default
database configuration.
This is the minimum table structure:
CREATE TABLE `swivel_features` (
`id` int(10) unsigned NOT NULL AUTO_INCREMENT,
`slug` varchar(255) NOT NULL,
`buckets` varchar(20) NOT NULL DEFAULT '1,2,3,4,5,6,7,8,9,10',
PRIMARY KEY (`id`),
UNIQUE KEY `slug_UNIQUE` (`slug`)
);
Feel free to add more fields if you want. For example, at Zumba we have the field modified that automatically auto-update when we change the buckets configuration. If you rename one of the pre-defined fields you will have to extends the plugin model and update accordingly.
Note the buckets are in a string field, separated by comma. You should not add spaces between the numbers.
You can also load the feature list from any other source, ie. from a webservice that you use to share across all your apps.
In order to do that, create a model and make this model to implement SwivelModelInterface
interface. Example, I will call my model MySwivelFeature
:
<?php
App::uses('AppModel', 'Model');
App::uses('SwivelModelInterface', 'Swivel.Lib');
class SwivelFeature extends AppModel implements SwivelModelInterface {
public $useTable = false;
public function getMapData()
{
// @todo Load data from some source
// @todo Format this data in a KEY/VALUE array, where KEY is the feature
// and VALUE is the buckets in an array format, ie [1, 2, 3]
// @todo Return the formatted data
return [
'FeatureA' => [1, 2, 3, 4, 5, 6, 7, 8, 9, 10],
'FeatureB' => [1, 2, 3],
'FeatureC' => [2, 4, 6, 8, 10],
];
}
}
Either the component, behavior and helper implement the methods forFeature
and invoke
,
which are part of swivel. You can check the details of these methods in
Swivel's documentation.
To exemplify in CakePHP application:
<?php
class UsersController extends AppController
{
public $components = ['Swivel.Swivel'];
public $uses = ['User', 'MyCoolWidget'];
public function index()
{
$this->set('users', $this->User->find(/* ... */));
$this->Swivel->invoke('MyCoolWidget', function() {
return $this->set('widget', $this->MyCoolWidget->find(/* ... */));
});
}
public function view($id = null)
{
$this->Swivel->forFeature('Redesign')
->addBehavior('userView', [$this, 'renderNewView'], [$id])
->defaultBehavior([$this, 'renderOldView'], [$id])
->execute();
}
protected function renderOldView($id)
{
// @todo implement
$this->render('oldView');
}
protected function renderNewView($id)
{
// @todo implement
$this->render('newView');
}
}
This is just an example of how you can use the plugin in the controller, but you can also use it on your models and views.