Files
wp-multisite-waas/dependencies/amphp/cache/lib/AtomicCache.php
2024-11-30 18:24:12 -07:00

238 lines
9.0 KiB
PHP

<?php
namespace WP_Ultimo\Dependencies\Amp\Cache;
use WP_Ultimo\Dependencies\Amp\Promise;
use WP_Ultimo\Dependencies\Amp\Serialization\SerializationException;
use WP_Ultimo\Dependencies\Amp\Sync\KeyedMutex;
use WP_Ultimo\Dependencies\Amp\Sync\Lock;
use function WP_Ultimo\Dependencies\Amp\call;
/**
* @template TValue
*/
final class AtomicCache
{
/** @var SerializedCache<TValue> */
private $cache;
/** @var KeyedMutex */
private $mutex;
/**
* @param SerializedCache<TValue> $cache
* @param KeyedMutex $mutex
*/
public function __construct(SerializedCache $cache, KeyedMutex $mutex)
{
$this->cache = $cache;
$this->mutex = $mutex;
}
/**
* Obtains the lock for the given key, then invokes the $create callback with the current cached value (which may
* be null if the key did not exist in the cache). The value returned from the callback is stored in the cache and
* the promise returned from this method is resolved with the value.
*
* @param string $key
* @param callable(string, mixed|null): mixed $create Receives $key and $value as parameters.
* @param int|null $ttl Timeout in seconds. The default `null` $ttl value indicates no timeout.
*
* @return Promise<mixed>
*
* @psalm-param callable(string, TValue|null):(TValue|Promise<TValue>|\Generator<mixed, mixed, mixed, TValue>)
* $create
* @psalm-return Promise<TValue>
*
* @throws CacheException If the $create callback throws an exception while generating the value.
* @throws SerializationException If serializing the value returned from the callback fails.
*/
public function compute(string $key, callable $create, ?int $ttl = null) : Promise
{
return call(function () use($key, $create, $ttl) : \Generator {
$lock = (yield from $this->lock($key));
\assert($lock instanceof Lock);
try {
$value = (yield $this->cache->get($key));
return yield from $this->create($create, $key, $value, $ttl);
} finally {
$lock->release();
}
});
}
/**
* Attempts to get the value for the given key. If the key is not found, the key is locked, the $create callback
* is invoked with the key as the first parameter. The value returned from the callback is stored in the cache and
* the promise returned from this method is resolved with the value.
*
* @param string $key Cache key.
* @param callable(string): mixed $create Receives $key as parameter.
* @param int|null $ttl Timeout in seconds. The default `null` $ttl value indicates no timeout.
*
* @return Promise<mixed>
*
* @psalm-param callable(string, TValue|null):(TValue|Promise<TValue>|\Generator<mixed, mixed, mixed, TValue>)
* $create
* @psalm-return Promise<TValue>
*
* @throws CacheException If the $create callback throws an exception while generating the value.
* @throws SerializationException If serializing the value returned from the callback fails.
*/
public function computeIfAbsent(string $key, callable $create, ?int $ttl = null) : Promise
{
return call(function () use($key, $create, $ttl) : \Generator {
$value = (yield $this->cache->get($key));
if ($value !== null) {
return $value;
}
$lock = (yield from $this->lock($key));
\assert($lock instanceof Lock);
try {
// Attempt to get the value again, since it may have been set while obtaining the lock.
$value = (yield $this->cache->get($key));
if ($value !== null) {
return $value;
}
return yield from $this->create($create, $key, null, $ttl);
} finally {
$lock->release();
}
});
}
/**
* Attempts to get the value for the given key. If the key exists, the key is locked, the $create callback
* is invoked with the key as the first parameter and the current key value as the second parameter. The value
* returned from the callback is stored in the cache and the promise returned from this method is resolved with
* the value.
*
* @param string $key Cache key.
* @param callable(string, mixed): mixed $create Receives $key and $value as parameters.
* @param int|null $ttl Timeout in seconds. The default `null` $ttl value indicates no timeout.
*
* @return Promise<mixed>
*
* @psalm-param callable(string, TValue|null): (TValue|Promise<TValue>|\Generator<mixed, mixed, mixed, TValue>) $create
* @psalm-return Promise<TValue>
*
* @throws CacheException If the $create callback throws an exception while generating the value.
* @throws SerializationException If serializing the value returned from the callback fails.
*/
public function computeIfPresent(string $key, callable $create, ?int $ttl = null) : Promise
{
return call(function () use($key, $create, $ttl) : \Generator {
$value = (yield $this->cache->get($key));
if ($value === null) {
return null;
}
$lock = (yield from $this->lock($key));
\assert($lock instanceof Lock);
try {
// Attempt to get the value again, since it may have been set while obtaining the lock.
$value = (yield $this->cache->get($key));
if ($value === null) {
return null;
}
return yield from $this->create($create, $key, $value, $ttl);
} finally {
$lock->release();
}
});
}
/**
* The lock is obtained for the key before setting the value.
*
* @param string $key Cache key.
* @param mixed $value Value to cache.
* @param int|null $ttl Timeout in seconds. The default `null` $ttl value indicates no timeout.
*
* @return Promise<void> Resolves either successfully or fails with a CacheException on failure.
*
* @psalm-param TValue $value
* @psalm-return Promise<void>
*
* @throws CacheException
* @throws SerializationException
*
* @see SerializedCache::set()
*/
public function set(string $key, $value, ?int $ttl = null) : Promise
{
return call(function () use($key, $value, $ttl) : \Generator {
$lock = (yield from $this->lock($key));
\assert($lock instanceof Lock);
try {
(yield $this->cache->set($key, $value, $ttl));
} finally {
$lock->release();
}
});
}
/**
* Returns the cached value for the key or the given default value if the key does not exist.
*
* @template TDefault
*
* @param string $key Cache key.
* @param mixed $default Default value returned if the key does not exist. Null by default.
*
* @return Promise<mixed|null> Resolved with null iff $default is null.
*
* @psalm-param TDefault $default
* @psalm-return Promise<TValue|TDefault>
*
* @throws CacheException
* @throws SerializationException
*
* @see SerializedCache::get()
*/
public function get(string $key, $default = null) : Promise
{
return call(function () use($key, $default) : \Generator {
$value = (yield $this->cache->get($key));
if ($value === null) {
return $default;
}
return $value;
});
}
/**
* The lock is obtained for the key before deleting the key.
*
* @param string $key
*
* @return Promise<bool|null>
*
* @see SerializedCache::delete()
*/
public function delete(string $key) : Promise
{
return call(function () use($key) : \Generator {
$lock = (yield from $this->lock($key));
\assert($lock instanceof Lock);
try {
return (yield $this->cache->delete($key));
} finally {
$lock->release();
}
});
}
private function lock(string $key) : \Generator
{
try {
return (yield $this->mutex->acquire($key));
} catch (\Throwable $exception) {
throw new CacheException(\sprintf('Exception thrown when obtaining the lock for key "%s"', $key), 0, $exception);
}
}
/**
* @psalm-param TValue|null $value
* @psalm-return \Generator<mixed, mixed, mixed, TValue>
*/
private function create(callable $create, string $key, $value, ?int $ttl) : \Generator
{
try {
$value = (yield call($create, $key, $value));
} catch (\Throwable $exception) {
throw new CacheException(\sprintf('Exception thrown while creating the value for key "%s"', $key), 0, $exception);
}
(yield $this->cache->set($key, $value, $ttl));
return $value;
}
}