/src/Symfony/Component/Lock/Store/RedisStore.php
https://github.com/gimler/symfony · PHP · 160 lines · 100 code · 23 blank · 37 comment · 16 complexity · 2cf4080e0a92592e86f8a5d57248b541 MD5 · raw file
- <?php
- /*
- * This file is part of the Symfony package.
- *
- * (c) Fabien Potencier <fabien@symfony.com>
- *
- * For the full copyright and license information, please view the LICENSE
- * file that was distributed with this source code.
- */
- namespace Symfony\Component\Lock\Store;
- use Symfony\Component\Cache\Traits\RedisProxy;
- use Symfony\Component\Lock\Exception\InvalidArgumentException;
- use Symfony\Component\Lock\Exception\LockConflictedException;
- use Symfony\Component\Lock\Exception\LockExpiredException;
- use Symfony\Component\Lock\Key;
- use Symfony\Component\Lock\StoreInterface;
- /**
- * RedisStore is a StoreInterface implementation using Redis as store engine.
- *
- * @author Jérémy Derussé <jeremy@derusse.com>
- */
- class RedisStore implements StoreInterface
- {
- private $redis;
- private $initialTtl;
- /**
- * @param \Redis|\RedisArray|\RedisCluster|\Predis\Client $redisClient
- * @param float $initialTtl the expiration delay of locks in seconds
- */
- public function __construct($redisClient, float $initialTtl = 300.0)
- {
- if (!$redisClient instanceof \Redis && !$redisClient instanceof \RedisArray && !$redisClient instanceof \RedisCluster && !$redisClient instanceof \Predis\Client && !$redisClient instanceof RedisProxy) {
- throw new InvalidArgumentException(sprintf('%s() expects parameter 1 to be Redis, RedisArray, RedisCluster or Predis\Client, %s given', __METHOD__, \is_object($redisClient) ? \get_class($redisClient) : \gettype($redisClient)));
- }
- if ($initialTtl <= 0) {
- throw new InvalidArgumentException(sprintf('%s() expects a strictly positive TTL. Got %d.', __METHOD__, $initialTtl));
- }
- $this->redis = $redisClient;
- $this->initialTtl = $initialTtl;
- }
- /**
- * {@inheritdoc}
- */
- public function save(Key $key)
- {
- $script = '
- if redis.call("GET", KEYS[1]) == ARGV[1] then
- return redis.call("PEXPIRE", KEYS[1], ARGV[2])
- elseif redis.call("SET", KEYS[1], ARGV[1], "NX", "PX", ARGV[2]) then
- return 1
- else
- return 0
- end
- ';
- $key->reduceLifetime($this->initialTtl);
- if (!$this->evaluate($script, (string) $key, array($this->getToken($key), (int) ceil($this->initialTtl * 1000)))) {
- throw new LockConflictedException();
- }
- if ($key->isExpired()) {
- throw new LockExpiredException(sprintf('Failed to store the "%s" lock.', $key));
- }
- }
- public function waitAndSave(Key $key)
- {
- throw new InvalidArgumentException(sprintf('The store "%s" does not supports blocking locks.', \get_class($this)));
- }
- /**
- * {@inheritdoc}
- */
- public function putOffExpiration(Key $key, $ttl)
- {
- $script = '
- if redis.call("GET", KEYS[1]) == ARGV[1] then
- return redis.call("PEXPIRE", KEYS[1], ARGV[2])
- else
- return 0
- end
- ';
- $key->reduceLifetime($ttl);
- if (!$this->evaluate($script, (string) $key, array($this->getToken($key), (int) ceil($ttl * 1000)))) {
- throw new LockConflictedException();
- }
- if ($key->isExpired()) {
- throw new LockExpiredException(sprintf('Failed to put off the expiration of the "%s" lock within the specified time.', $key));
- }
- }
- /**
- * {@inheritdoc}
- */
- public function delete(Key $key)
- {
- $script = '
- if redis.call("GET", KEYS[1]) == ARGV[1] then
- return redis.call("DEL", KEYS[1])
- else
- return 0
- end
- ';
- $this->evaluate($script, (string) $key, array($this->getToken($key)));
- }
- /**
- * {@inheritdoc}
- */
- public function exists(Key $key)
- {
- return $this->redis->get((string) $key) === $this->getToken($key);
- }
- /**
- * Evaluates a script in the corresponding redis client.
- *
- * @return mixed
- */
- private function evaluate(string $script, string $resource, array $args)
- {
- if ($this->redis instanceof \Redis || $this->redis instanceof \RedisCluster || $this->redis instanceof RedisProxy) {
- return $this->redis->eval($script, array_merge(array($resource), $args), 1);
- }
- if ($this->redis instanceof \RedisArray) {
- return $this->redis->_instance($this->redis->_target($resource))->eval($script, array_merge(array($resource), $args), 1);
- }
- if ($this->redis instanceof \Predis\Client) {
- return \call_user_func_array(array($this->redis, 'eval'), array_merge(array($script, 1, $resource), $args));
- }
- throw new InvalidArgumentException(sprintf('%s() expects being initialized with a Redis, RedisArray, RedisCluster or Predis\Client, %s given', __METHOD__, \is_object($this->redis) ? \get_class($this->redis) : \gettype($this->redis)));
- }
- /**
- * Retrieves an unique token for the given key.
- */
- private function getToken(Key $key): string
- {
- if (!$key->hasState(__CLASS__)) {
- $token = base64_encode(random_bytes(32));
- $key->setState(__CLASS__, $token);
- }
- return $key->getState(__CLASS__);
- }
- }