Utopia Balancer library is simple and lite library for balancing choices between multiple options. This library is aiming to be as simple and easy to learn and use. This library is maintained by the Appwrite team.
Although this library is part of the Utopia Framework project it is dependency free and can be used as standalone with any other PHP project or framework.
Install using composer:
composer require utopia-php/balancer
Balancer supports multiple algorithms. Each picks option differently, and may have different set of methods available for configuration.
- Random
Random
algorithm pick option randomly. The same option could be picked multiple times in a row. Example:
<?php
require_once '../vendor/autoload.php';
use Utopia\Balancer\Algorithm\Random;
use Utopia\Balancer\Balancer;
use Utopia\Balancer\Option;
$balancer = new Balancer(new Random());
$balancer->addFilter(fn (Option $option) => $option->getState('online', false) === true);
$balancer
->addOption(new Option([ 'hostname' => 'proxy-1', 'online' => true ]))
->addOption(new Option([ 'hostname' => 'proxy-2', 'online' => false ]))
->addOption(new Option([ 'hostname' => 'proxy-3', 'online' => true ]));
var_dump($balancer->run());
var_dump($balancer->run());
var_dump($balancer->run());
- First and Last
First
algorithm always picks first option. Similiarly, Last
algorithm always picks last option. Example:
<?php
require_once '../vendor/autoload.php';
use Utopia\Balancer\Algorithm\First;
use Utopia\Balancer\Algorithm\Last;
use Utopia\Balancer\Balancer;
use Utopia\Balancer\Option;
$balancer = new Balancer(new First());
$balancer
->addOption(new Option([ 'runtime' => 'PHP' ]))
->addOption(new Option([ 'runtime' => 'JavaScript' ]))
->addOption(new Option([ 'runtime' => 'Java' ]));
var_dump($balancer->run());
$balancer = new Balancer(new Last());
$balancer
->addOption(new Option([ 'runtime' => 'PHP' ]))
->addOption(new Option([ 'runtime' => 'JavaScript' ]))
->addOption(new Option([ 'runtime' => 'Java' ]));
var_dump($balancer->run());
- Round Robin
RoundRobin
algorithm cycles over all options starting first. Once algorithm cycles over all options, it resets back to the beginning. Example:
<?php
require_once '../vendor/autoload.php';
use Utopia\Balancer\Algorithm\RoundRobin;
use Utopia\Balancer\Balancer;
use Utopia\Balancer\Option;
$balancer = new Balancer(new RoundRobin(-1));
$balancer->addFilter(fn (Option $option) => $option->getState('online', false) === true);
$balancer
->addOption(new Option([ 'dataCenter' => 'fra-1' ]))
->addOption(new Option([ 'dataCenter' => 'fra-2' ]))
->addOption(new Option([ 'dataCenter' => 'lon-1' ]));
var_dump($balancer->run()); // fra-1
var_dump($balancer->run()); // fra-2
var_dump($balancer->run()); // lon-1
var_dump($balancer->run()); // fra-1
var_dump($balancer->run()); // fra-2
When using RoundRobin
in concurrency model, make sure to store index in atomic way. Example:
<?php
require_once '../vendor/autoload.php';
use Utopia\Balancer\Algorithm\RoundRobin;
use Utopia\Balancer\Balancer;
use Utopia\Balancer\Option;
$atomic = new Atomic(-1); // Some atomic implementation, for example: https://openswoole.com/docs/modules/swoole-atomic
function onRequest() {
$lastIndex = $atomic->get();
$algo = new RoundRobin($lastIndex);
$balancer = new Balancer();
$balancer->addFilter(fn (Option $option) => $option->getState('online', false) === true);
$balancer
->addOption(new Option([ 'dataCenter' => 'fra-1' ]))
->addOption(new Option([ 'dataCenter' => 'fra-2' ]))
->addOption(new Option([ 'dataCenter' => 'lon-1' ]));
var_dump($balancer->run());
$atomic->cmpset($lastIndex, $algo->getIndex());
}
If balancer filters cause balancer to have no options to pick from, null
will be returned. More often then not, you will need a backup logic for this scenario. You can use Group
to create a group of multiple balancers and if one fails, next can be used as fallback. Notice Group
tries balances in order you added them.
<?php
require_once '../vendor/autoload.php';
use Utopia\Balancer\Algorithm\First;
use Utopia\Balancer\Balancer;
use Utopia\Balancer\Group;
use Utopia\Balancer\Option;
// Prepare options where each has high CPU load
$options = [
new Option([ 'dataCenter' => 'fra-1', 'cpu' => 91 ]),
new Option([ 'dataCenter' => 'fra-2', 'cpu' => 95 ]),
new Option([ 'dataCenter' => 'lon-1', 'cpu' => 87 ]),
];
// Prepare balancer that allows only low CPU load options
$balancer1 = new Balancer(new First());
$balancer1->addFilter(fn ($option) => $option->getState('cpu') < 80);
// Prepare balancer that allows all options
$balancer2 = new Balancer(new First());
// Add options to both balancers
foreach ($options as $option) {
$balancer1->addOption($option);
$balancer2->addOption($option);
}
// Prepare group with both balancers
$group = new Group();
$group
->add($balancer1)
->add($balancer2);
// Run group to get option
$option = $group->run() ?? new Option([]);
\var_dump($option);
// We got fra-1 option. First balancer filtered out all options, but second balancer allowed any, and First algorithm picked first option
Utopia Framework requires PHP 8.0 or later. We recommend using the latest PHP version whenever possible.
The MIT License (MIT) http://www.opensource.org/licenses/mit-license.php