Overview

Namespaces

  • bandwidthThrottle
    • tokenBucket
      • storage
        • scope

Classes

  • bandwidthThrottle\tokenBucket\BlockingConsumer
  • bandwidthThrottle\tokenBucket\Rate
  • bandwidthThrottle\tokenBucket\storage\FileStorage
  • bandwidthThrottle\tokenBucket\storage\IPCStorage
  • bandwidthThrottle\tokenBucket\storage\MemcachedStorage
  • bandwidthThrottle\tokenBucket\storage\MemcacheStorage
  • bandwidthThrottle\tokenBucket\storage\PDOStorage
  • bandwidthThrottle\tokenBucket\storage\PHPRedisStorage
  • bandwidthThrottle\tokenBucket\storage\PredisStorage
  • bandwidthThrottle\tokenBucket\storage\SessionStorage
  • bandwidthThrottle\tokenBucket\storage\SingleProcessStorage
  • bandwidthThrottle\tokenBucket\TokenBucket

Interfaces

  • bandwidthThrottle\tokenBucket\storage\scope\GlobalScope
  • bandwidthThrottle\tokenBucket\storage\scope\RequestScope
  • bandwidthThrottle\tokenBucket\storage\scope\SessionScope
  • bandwidthThrottle\tokenBucket\storage\Storage

Exceptions

  • bandwidthThrottle\tokenBucket\storage\StorageException
  • bandwidthThrottle\tokenBucket\TimeoutException
  • bandwidthThrottle\tokenBucket\TokenBucketException
  • Overview
  • Namespace
  • Class
  1:   2:   3:   4:   5:   6:   7:   8:   9:  10:  11:  12:  13:  14:  15:  16:  17:  18:  19:  20:  21:  22:  23:  24:  25:  26:  27:  28:  29:  30:  31:  32:  33:  34:  35:  36:  37:  38:  39:  40:  41:  42:  43:  44:  45:  46:  47:  48:  49:  50:  51:  52:  53:  54:  55:  56:  57:  58:  59:  60:  61:  62:  63:  64:  65:  66:  67:  68:  69:  70:  71:  72:  73:  74:  75:  76:  77:  78:  79:  80:  81:  82:  83:  84:  85:  86:  87:  88:  89:  90:  91:  92:  93:  94:  95:  96:  97:  98:  99: 100: 101: 102: 103: 104: 105: 106: 107: 108: 109: 110: 111: 112: 113: 114: 115: 116: 117: 118: 119: 120: 121: 122: 123: 124: 125: 126: 127: 128: 129: 130: 131: 132: 133: 134: 135: 136: 137: 138: 139: 140: 
<?php

namespace bandwidthThrottle\tokenBucket\storage;

use bandwidthThrottle\tokenBucket\storage\scope\GlobalScope;
use malkusch\lock\mutex\CASMutex;

/**
 * Memcached based storage which can be shared among processes.
 *
 * This storage is in the global scope.
 *
 * @author Markus Malkusch <markus@malkusch.de>
 * @link bitcoin:1335STSwu9hST4vcMRppEPgENMHD2r1REK Donations
 * @license WTFPL
 */
final class MemcachedStorage implements Storage, GlobalScope
{

    /**
     * @var \Memcached The memcached API.
     */
    private $memcached;
    
    /**
     * @var float The CAS token.
     */
    private $casToken;
    
    /**
     * @var string The key for the token bucket.
     */
    private $key;
    
    /**
     * @var CASMutex The mutex for this storage.
     */
    private $mutex;

    /**
     * @internal
     */
    const PREFIX = "TokenBucketD_";
    
    /**
     * Sets the memcached API and the token bucket name.
     *
     * The api needs to have at least one server in its pool. I.e.
     * it has to be added with Memcached::addServer().
     *
     * @param string     $name      The name of the shared token bucket.
     * @param \Memcached $memcached The memcached API.
     */
    public function __construct($name, \Memcached $memcached)
    {
        $this->key   = self::PREFIX . $name;
        $this->mutex = new CASMutex();
        $this->memcached = $memcached;
    }
    
    public function bootstrap($microtime)
    {
        if ($this->memcached->add($this->key, $microtime)) {
            $this->mutex->notify(); // [CAS] Stop TokenBucket::bootstrap()
            return;
        }
        if ($this->memcached->getResultCode() === \Memcached::RES_NOTSTORED) {
            // [CAS] repeat TokenBucket::bootstrap()
            return;
        }
        throw new StorageException($this->memcached->getResultMessage(), $this->memcached->getResultCode());
    }
    
    public function isBootstrapped()
    {
        if ($this->memcached->get($this->key) !== false) {
            $this->mutex->notify(); // [CAS] Stop TokenBucket::bootstrap()
            return true;
        }
        if ($this->memcached->getResultCode() === \Memcached::RES_NOTFOUND) {
            return false;
        }
        throw new StorageException($this->memcached->getResultMessage(), $this->memcached->getResultCode());
    }
    
    public function remove()
    {
        if (!$this->memcached->delete($this->key)) {
            throw new StorageException($this->memcached->getResultMessage(), $this->memcached->getResultCode());
        }
    }
    
    public function setMicrotime($microtime)
    {
        if (is_null($this->casToken)) {
            throw new StorageException("CAS token is null. Call getMicrotime() first.");
        }
        if ($this->memcached->cas($this->casToken, $this->key, $microtime)) {
            $this->mutex->notify(); // [CAS] Stop TokenBucket::consume()
            return;
        }
        if ($this->memcached->getResultCode() === \Memcached::RES_DATA_EXISTS) {
            // [CAS] repeat TokenBucket::consume()
            return;
        }
        throw new StorageException($this->memcached->getResultMessage(), $this->memcached->getResultCode());
    }

    public function getMicrotime()
    {
        $getDelayed = $this->memcached->getDelayed([$this->key], true);
        if (!$getDelayed) {
            throw new StorageException($this->memcached->getResultMessage(), $this->memcached->getResultCode());
        }
        
        $result = $this->memcached->fetchAll();
        if (!$result) {
            throw new StorageException($this->memcached->getResultMessage(), $this->memcached->getResultCode());
        }
        
        $microtime = $result[0]["value"];
        $this->casToken = $result[0]["cas"];
        if ($this->casToken === null) {
            throw new StorageException("Failed to aquire a CAS token.");
        }
        
        return (double) $microtime;
    }

    public function getMutex()
    {
        return $this->mutex;
    }

    public function letMicrotimeUnchanged()
    {
        $this->mutex->notify();
    }
}
API documentation generated by ApiGen