consistent hashing for multiple data mirrors
This commit is contained in:
parent
5b9c8b736d
commit
9bde42d452
2 changed files with 291 additions and 0 deletions
|
@ -24,6 +24,8 @@
|
|||
\* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
|
||||
|
||||
$tag_n = 0; // temp hack
|
||||
$_flexihash = null;
|
||||
$_fh_last_opts = null;
|
||||
|
||||
/**
|
||||
* An object representing an entry in the images table. As of 2.2, this no
|
||||
|
@ -507,6 +509,26 @@ class Image {
|
|||
$tmpl = $plte->link;
|
||||
}
|
||||
|
||||
global $_flexihash, $_fh_last_opts;
|
||||
$matches = array();
|
||||
if(preg_match("/(.*){(.*)}(.*)/", $tmpl, &$matches)) {
|
||||
$pre = $matches[1];
|
||||
$opts = $matches[2];
|
||||
$post = $matches[3];
|
||||
|
||||
if($opts != $_fh_last_opts) {
|
||||
$_fh_last_opts = $opts;
|
||||
require_once("lib/flexihash.php");
|
||||
$_flexihash = new Flexihash();
|
||||
foreach(explode(",", $opts) as $opt) {
|
||||
$_flexihash->addTarget($opt);
|
||||
}
|
||||
}
|
||||
|
||||
$choice = $_flexihash->lookup($pre.$post);
|
||||
$tmpl = $pre.$choice.$post;
|
||||
}
|
||||
|
||||
return $tmpl;
|
||||
}
|
||||
|
||||
|
|
269
lib/flexihash.php
Normal file
269
lib/flexihash.php
Normal file
|
@ -0,0 +1,269 @@
|
|||
<?php
|
||||
|
||||
interface Flexihash_Hasher
|
||||
{
|
||||
public function hash($string);
|
||||
}
|
||||
|
||||
class Flexihash_Crc32Hasher
|
||||
implements Flexihash_Hasher
|
||||
{
|
||||
|
||||
public function hash($string)
|
||||
{
|
||||
return crc32($string);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
class Flexihash_Exception extends Exception
|
||||
{
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* A simple consistent hashing implementation with pluggable hash algorithms.
|
||||
*
|
||||
* @author Paul Annesley
|
||||
* @package Flexihash
|
||||
* @licence http://www.opensource.org/licenses/mit-license.php
|
||||
*/
|
||||
class Flexihash
|
||||
{
|
||||
|
||||
/**
|
||||
* The number of positions to hash each target to.
|
||||
*
|
||||
* @var int
|
||||
*/
|
||||
private $_replicas = 64;
|
||||
|
||||
/**
|
||||
* The hash algorithm, encapsulated in a Flexihash_Hasher implementation.
|
||||
* @var object Flexihash_Hasher
|
||||
*/
|
||||
private $_hasher;
|
||||
|
||||
/**
|
||||
* Internal counter for current number of targets.
|
||||
* @var int
|
||||
*/
|
||||
private $_targetCount = 0;
|
||||
|
||||
/**
|
||||
* Internal map of positions (hash outputs) to targets
|
||||
* @var array { position => target, ... }
|
||||
*/
|
||||
private $_positionToTarget = array();
|
||||
|
||||
/**
|
||||
* Internal map of targets to lists of positions that target is hashed to.
|
||||
* @var array { target => [ position, position, ... ], ... }
|
||||
*/
|
||||
private $_targetToPositions = array();
|
||||
|
||||
/**
|
||||
* Whether the internal map of positions to targets is already sorted.
|
||||
* @var boolean
|
||||
*/
|
||||
private $_positionToTargetSorted = false;
|
||||
|
||||
/**
|
||||
* Constructor
|
||||
* @param object $hasher Flexihash_Hasher
|
||||
* @param int $replicas Amount of positions to hash each target to.
|
||||
*/
|
||||
public function __construct(Flexihash_Hasher $hasher = null, $replicas = null)
|
||||
{
|
||||
$this->_hasher = $hasher ? $hasher : new Flexihash_Crc32Hasher();
|
||||
if (!empty($replicas)) $this->_replicas = $replicas;
|
||||
}
|
||||
|
||||
/**
|
||||
* Add a target.
|
||||
* @param string $target
|
||||
* @param float $weight
|
||||
* @chainable
|
||||
*/
|
||||
public function addTarget($target, $weight=1)
|
||||
{
|
||||
if (isset($this->_targetToPositions[$target]))
|
||||
{
|
||||
throw new Flexihash_Exception("Target '$target' already exists.");
|
||||
}
|
||||
|
||||
$this->_targetToPositions[$target] = array();
|
||||
|
||||
// hash the target into multiple positions
|
||||
for ($i = 0; $i < round($this->_replicas*$weight); $i++)
|
||||
{
|
||||
$position = $this->_hasher->hash($target . $i);
|
||||
$this->_positionToTarget[$position] = $target; // lookup
|
||||
$this->_targetToPositions[$target] []= $position; // target removal
|
||||
}
|
||||
|
||||
$this->_positionToTargetSorted = false;
|
||||
$this->_targetCount++;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Add a list of targets.
|
||||
* @param array $targets
|
||||
* @param float $weight
|
||||
* @chainable
|
||||
*/
|
||||
public function addTargets($targets, $weight=1)
|
||||
{
|
||||
foreach ($targets as $target)
|
||||
{
|
||||
$this->addTarget($target,$weight);
|
||||
}
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* Remove a target.
|
||||
* @param string $target
|
||||
* @chainable
|
||||
*/
|
||||
public function removeTarget($target)
|
||||
{
|
||||
if (!isset($this->_targetToPositions[$target]))
|
||||
{
|
||||
throw new Flexihash_Exception("Target '$target' does not exist.");
|
||||
}
|
||||
|
||||
foreach ($this->_targetToPositions[$target] as $position)
|
||||
{
|
||||
unset($this->_positionToTarget[$position]);
|
||||
}
|
||||
|
||||
unset($this->_targetToPositions[$target]);
|
||||
|
||||
$this->_targetCount--;
|
||||
|
||||
return $this;
|
||||
}
|
||||
|
||||
/**
|
||||
* A list of all potential targets
|
||||
* @return array
|
||||
*/
|
||||
public function getAllTargets()
|
||||
{
|
||||
return array_keys($this->_targetToPositions);
|
||||
}
|
||||
|
||||
/**
|
||||
* Looks up the target for the given resource.
|
||||
* @param string $resource
|
||||
* @return string
|
||||
*/
|
||||
public function lookup($resource)
|
||||
{
|
||||
$targets = $this->lookupList($resource, 1);
|
||||
if (empty($targets)) throw new Flexihash_Exception('No targets exist');
|
||||
return $targets[0];
|
||||
}
|
||||
|
||||
/**
|
||||
* Get a list of targets for the resource, in order of precedence.
|
||||
* Up to $requestedCount targets are returned, less if there are fewer in total.
|
||||
*
|
||||
* @param string $resource
|
||||
* @param int $requestedCount The length of the list to return
|
||||
* @return array List of targets
|
||||
*/
|
||||
public function lookupList($resource, $requestedCount)
|
||||
{
|
||||
if (!$requestedCount)
|
||||
throw new Flexihash_Exception('Invalid count requested');
|
||||
|
||||
// handle no targets
|
||||
if (empty($this->_positionToTarget))
|
||||
return array();
|
||||
|
||||
// optimize single target
|
||||
if ($this->_targetCount == 1)
|
||||
return array_unique(array_values($this->_positionToTarget));
|
||||
|
||||
// hash resource to a position
|
||||
$resourcePosition = $this->_hasher->hash($resource);
|
||||
|
||||
$results = array();
|
||||
$collect = false;
|
||||
|
||||
$this->_sortPositionTargets();
|
||||
|
||||
// search values above the resourcePosition
|
||||
foreach ($this->_positionToTarget as $key => $value)
|
||||
{
|
||||
// start collecting targets after passing resource position
|
||||
if (!$collect && $key > $resourcePosition)
|
||||
{
|
||||
$collect = true;
|
||||
}
|
||||
|
||||
// only collect the first instance of any target
|
||||
if ($collect && !in_array($value, $results))
|
||||
{
|
||||
$results []= $value;
|
||||
}
|
||||
|
||||
// return when enough results, or list exhausted
|
||||
if (count($results) == $requestedCount || count($results) == $this->_targetCount)
|
||||
{
|
||||
return $results;
|
||||
}
|
||||
}
|
||||
|
||||
// loop to start - search values below the resourcePosition
|
||||
foreach ($this->_positionToTarget as $key => $value)
|
||||
{
|
||||
if (!in_array($value, $results))
|
||||
{
|
||||
$results []= $value;
|
||||
}
|
||||
|
||||
// return when enough results, or list exhausted
|
||||
if (count($results) == $requestedCount || count($results) == $this->_targetCount)
|
||||
{
|
||||
return $results;
|
||||
}
|
||||
}
|
||||
|
||||
// return results after iterating through both "parts"
|
||||
return $results;
|
||||
}
|
||||
|
||||
public function __toString()
|
||||
{
|
||||
return sprintf(
|
||||
'%s{targets:[%s]}',
|
||||
get_class($this),
|
||||
implode(',', $this->getAllTargets())
|
||||
);
|
||||
}
|
||||
|
||||
// ----------------------------------------
|
||||
// private methods
|
||||
|
||||
/**
|
||||
* Sorts the internal mapping (positions to targets) by position
|
||||
*/
|
||||
private function _sortPositionTargets()
|
||||
{
|
||||
// sort by key (position) if not already
|
||||
if (!$this->_positionToTargetSorted)
|
||||
{
|
||||
ksort($this->_positionToTarget, SORT_REGULAR);
|
||||
$this->_positionToTargetSorted = true;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
Reference in a new issue