You are here

class Redis_Lock_Backend_Predis in Redis 7.2

Same name and namespace in other branches
  1. 7 lib/Redis/Lock/Backend/Predis.php \Redis_Lock_Backend_Predis

Predis lock backend implementation.

This implementation works with a single key per lock so is viable when doing client side sharding and/or using consistent hashing algorithm.

Hierarchy

Expanded class hierarchy of Redis_Lock_Backend_Predis

File

lib/Redis/Lock/Backend/Predis.php, line 9

View source
class Redis_Lock_Backend_Predis extends Redis_Lock_Backend_Default {
  public function lockAcquire($name, $timeout = 30.0) {
    $client = Redis_Client::getClient();
    $key = $this
      ->getKey($name);
    $id = $this
      ->getLockId();

    // Insure that the timeout is at least 1 second, we cannot do otherwise with
    // Redis, this is a minor change to the function signature, but in real life
    // nobody will notice with so short duration.
    $timeout = ceil(max($timeout, 1));

    // If we already have the lock, check for his owner and attempt a new EXPIRE
    // command on it.
    if (isset($this->_locks[$name])) {

      // Create a new transaction, for atomicity.
      $client
        ->watch($key);

      // Global tells us we are the owner, but in real life it could have expired
      // and another process could have taken it, check that.
      if ($client
        ->get($key) != $id) {
        $client
          ->unwatch();
        unset($this->_locks[$name]);
        return FALSE;
      }
      switch (Redis_Client_Predis::getPredisVersionMajor()) {
        case 0:
          $replies = $client
            ->pipeline(function ($pipe) use ($key, $timeout, $id) {
            $pipe
              ->multi();
            $pipe
              ->setex($key, $timeout, $id);
            $pipe
              ->exec();
          });
          break;
        default:
          $replies = $client
            ->transaction(function ($pipe) use ($key, $timeout, $id) {
            $pipe
              ->setex($key, $timeout, $id);
          });
          break;
      }
      $execReply = array_pop($replies);
      if (FALSE === $execReply[0]) {
        unset($this->_locks[$name]);
        return FALSE;
      }
      return TRUE;
    }
    else {
      $client
        ->watch($key);
      $owner = $client
        ->get($key);
      if (!empty($owner) && $owner != $id) {
        $client
          ->unwatch();
        unset($this->_locks[$name]);
        return FALSE;
      }
      $replies = $client
        ->pipeline(function ($pipe) use ($key, $timeout, $id) {
        $pipe
          ->multi();
        $pipe
          ->setex($key, $timeout, $id);
        $pipe
          ->exec();
      });
      $execReply = array_pop($replies);

      // If another client modified the $key value, transaction will be discarded
      // $result will be set to FALSE. This means atomicity have been broken and
      // the other client took the lock instead of us.
      // EXPIRE and SETEX won't return something here, EXEC return is index 0
      // This was determined debugging, seems to be Predis specific.
      if (FALSE === $execReply[0]) {
        return FALSE;
      }

      // Register the lock and return.
      return $this->_locks[$name] = TRUE;
    }
    return FALSE;
  }
  public function lockMayBeAvailable($name) {
    $client = Redis_Client::getClient();
    $key = $this
      ->getKey($name);
    $id = $this
      ->getLockId();
    $value = $client
      ->get($key);
    return empty($value) || $id == $value;
  }
  public function lockRelease($name) {
    $client = Redis_Client::getClient();
    $key = $this
      ->getKey($name);
    $id = $this
      ->getLockId();
    unset($this->_locks[$name]);

    // Ensure the lock deletion is an atomic transaction. If another thread
    // manages to removes all lock, we can not alter it anymore else we will
    // release the lock for the other thread and cause race conditions.
    $client
      ->watch($key);
    if ($client
      ->get($key) == $id) {
      $client
        ->multi();
      $client
        ->del(array(
        $key,
      ));
      $client
        ->exec();
    }
    else {
      $client
        ->unwatch();
    }
  }
  public function lockReleaseAll($lock_id = NULL) {
    if (!isset($lock_id) && empty($this->_locks)) {
      return;
    }
    $client = Redis_Client::getClient();
    $id = isset($lock_id) ? $lock_id : $this
      ->getLockId();

    // We can afford to deal with a slow algorithm here, this should not happen
    // on normal run because we should have removed manually all our locks.
    foreach ($this->_locks as $name => $foo) {
      $key = $this
        ->getKey($name);
      $owner = $client
        ->get($key);
      if (empty($owner) || $owner == $id) {
        $client
          ->del(array(
          $key,
        ));
      }
    }
  }

}

Members

Namesort descending Modifiers Type Description Overrides
Redis_AbstractBackend::$globalPrefix protected static property
Redis_AbstractBackend::$prefix private property
Redis_AbstractBackend::getClient public function Get redis client
Redis_AbstractBackend::getDefaultPrefix public static function Get global default prefix
Redis_AbstractBackend::getGlobalPrefix public static function Get site default global prefix
Redis_AbstractBackend::getPrefix final public function Get prefix
Redis_AbstractBackend::KEY_SEPARATOR constant Key components name separator
Redis_AbstractBackend::setPrefix final public function Set prefix
Redis_AbstractBackend::__construct public function Default constructor 2
Redis_Lock_Backend_Default::$_lockId protected property Current page lock token identifier.
Redis_Lock_Backend_Default::$_locks protected property Existing locks for this page.
Redis_Lock_Backend_Default::getKey public function Generate a redis key name for the current lock name Overrides Redis_AbstractBackend::getKey
Redis_Lock_Backend_Default::getLockId public function Default implementation from actual Drupal core. Overrides Redis_Lock_Backend_Interface::getLockId
Redis_Lock_Backend_Default::lockWait public function Default implementation from actual Drupal core. Overrides Redis_Lock_Backend_Interface::lockWait
Redis_Lock_Backend_Predis::lockAcquire public function Acquire lock. Overrides Redis_Lock_Backend_Interface::lockAcquire
Redis_Lock_Backend_Predis::lockMayBeAvailable public function Check if lock is available for acquire. Overrides Redis_Lock_Backend_Interface::lockMayBeAvailable
Redis_Lock_Backend_Predis::lockRelease public function Release given lock. Overrides Redis_Lock_Backend_Interface::lockRelease
Redis_Lock_Backend_Predis::lockReleaseAll public function Release all locks for the given lock token identifier. Overrides Redis_Lock_Backend_Interface::lockReleaseAll