Backed by a custom database table (one per multisite network) instead of transients, ensuring values are not lost due to cache eviction or transient purging.
When to use this
Use Expiring_Store when losing a value before its TTL has real consequences:
OAuth handshakes and short-lived tokens (e.g. PKCE code verifiers).
Locks that prevent concurrent operations (e.g. token refresh race conditions).
Any value where a missing entry causes user-facing errors or excessive API calls.
When to use transients or wp_cache instead
wp_cache: For data that only needs to live within the current request, or that benefits from a persistent object cache but can be recomputed cheaply if lost.
Transients: For data that is purely a performance optimization (caching). If the transient disappears, the worst case is a slower request while the value is recomputed. Never use transients for data whose loss would cause functional failures.
Scoping strategies
Blog-scoped (persist, get, delete): Keys are prefixed with the current blog ID. Use for data that belongs to a specific site in a multisite network.
User-scoped (*_for_user): Keys are prefixed with the given or current user ID. Use for per-user data like OAuth tokens or verification codes. Accepts an optional $user_id; when omitted (or 0), falls back to the current user. Throws No_Current_User_Exception{} when no user ID is given and no user is logged in.
Network-scoped (*_for_multisite): Keys are stored as-is without any prefix. Use for data shared across all sites in the network.
Behavior
Values are JSON-encoded for storage (not PHP-serialized) to avoid object injection risks. Any JSON-encodable value is accepted: scalars, arrays, or {@see \JsonSerializable} objects.
If a key already exists, persist overwrites it (upsert behavior). If a key is not found or has expired, get throws a Key_Not_Found_Exception{}. If a key's value cannot be decoded from JSON, get throws a Corrupted_Value_Exception{}.
Expired entries are cleaned up automatically by the hourly wpseo_cleanup_cron job and can be triggered manually via wp yoast cleanup.
Хуков нет.
Использование
$Expiring_Store = new Expiring_Store();
// use class methods
class Expiring_Store {
/**
* The repository for database operations.
*
* @var Expiring_Store_Repository_Interface
*/
private $repository;
/**
* The date helper.
*
* @var Date_Helper
*/
private $date_helper;
/**
* The constructor.
*
* @param Expiring_Store_Repository_Interface $repository The repository for database operations.
* @param Date_Helper $date_helper The date helper.
*/
public function __construct( Expiring_Store_Repository_Interface $repository, Date_Helper $date_helper ) {
$this->repository = $repository;
$this->date_helper = $date_helper;
}
/**
* Persists a value scoped to the current blog.
*
* @param string $key The key.
* @param scalar|array<string|int|float|bool|array|null>|JsonSerializable $value The value to store.
* @param int $ttl_in_seconds The time-to-live in seconds.
*
* @return void
* @throws InvalidArgumentException When the value is not JSON-encodable.
*/
public function persist( string $key, $value, int $ttl_in_seconds ): void {
$this->do_persist( $this->prefix_for_blog( $key ), $value, $ttl_in_seconds );
}
/**
* Persists a value scoped to a user.
*
* @param string $key The key.
* @param scalar|array<string|int|float|bool|array|null>|JsonSerializable $value The value to store.
* @param int $ttl_in_seconds The time-to-live in seconds.
* @param int $user_id The user ID. Defaults to the current user.
*
* @return void
* @throws InvalidArgumentException When the value is not JSON-encodable.
* @throws No_Current_User_Exception When no user ID is given and no user is logged in.
*/
public function persist_for_user( string $key, $value, int $ttl_in_seconds, int $user_id = 0 ): void {
$this->do_persist( $this->prefix_for_user( $key, $user_id ), $value, $ttl_in_seconds );
}
/**
* Persists a value shared across the entire multisite network.
*
* @param string $key The key.
* @param scalar|array<string|int|float|bool|array|null>|JsonSerializable $value The value to store.
* @param int $ttl_in_seconds The time-to-live in seconds.
*
* @return void
* @throws InvalidArgumentException When the value is not JSON-encodable.
*/
public function persist_for_multisite( string $key, $value, int $ttl_in_seconds ): void {
$this->do_persist( $key, $value, $ttl_in_seconds );
}
/**
* Persists a value scoped to the current blog, only if the key does not already exist.
*
* @param string $key The key.
* @param scalar|array<string|int|float|bool|array|null>|JsonSerializable $value The value to store.
* @param int $ttl_in_seconds The time-to-live in seconds.
*
* @return bool True if the value was inserted, false if the key already exists.
* @throws InvalidArgumentException When the value is not JSON-encodable.
*/
public function persist_if_absent( string $key, $value, int $ttl_in_seconds ): bool {
return $this->do_persist_if_absent( $this->prefix_for_blog( $key ), $value, $ttl_in_seconds );
}
/**
* Persists a value scoped to a user, only if the key does not already exist.
*
* @param string $key The key.
* @param scalar|array<string|int|float|bool|array|null>|JsonSerializable $value The value to store.
* @param int $ttl_in_seconds The time-to-live in seconds.
* @param int $user_id The user ID. Defaults to the current user.
*
* @return bool True if the value was inserted, false if the key already exists.
* @throws InvalidArgumentException When the value is not JSON-encodable.
* @throws No_Current_User_Exception When no user ID is given and no user is logged in.
*/
public function persist_if_absent_for_user( string $key, $value, int $ttl_in_seconds, int $user_id = 0 ): bool {
return $this->do_persist_if_absent( $this->prefix_for_user( $key, $user_id ), $value, $ttl_in_seconds );
}
/**
* Persists a value shared across the entire multisite network, only if the key does not already exist.
*
* @param string $key The key.
* @param scalar|array<string|int|float|bool|array|null>|JsonSerializable $value The value to store.
* @param int $ttl_in_seconds The time-to-live in seconds.
*
* @return bool True if the value was inserted, false if the key already exists.
* @throws InvalidArgumentException When the value is not JSON-encodable.
*/
public function persist_if_absent_for_multisite( string $key, $value, int $ttl_in_seconds ): bool {
return $this->do_persist_if_absent( $key, $value, $ttl_in_seconds );
}
/**
* Gets a value scoped to the current blog.
*
* @param string $key The key.
*
* @return scalar|array<string|int|float|bool|array|null> The stored value.
* @throws Key_Not_Found_Exception When the key is not found or has expired.
* @throws Corrupted_Value_Exception When the stored value cannot be decoded from JSON.
*/
public function get( string $key ) {
return $this->do_get( $this->prefix_for_blog( $key ) );
}
/**
* Gets a value scoped to a user.
*
* @param string $key The key.
* @param int $user_id The user ID. Defaults to the current user.
*
* @return scalar|array<string|int|float|bool|array|null> The stored value.
* @throws Key_Not_Found_Exception When the key is not found or has expired.
* @throws Corrupted_Value_Exception When the stored value cannot be decoded from JSON.
* @throws No_Current_User_Exception When no user ID is given and no user is logged in.
*/
public function get_for_user( string $key, int $user_id = 0 ) {
return $this->do_get( $this->prefix_for_user( $key, $user_id ) );
}
/**
* Gets a value shared across the entire multisite network.
*
* @param string $key The key.
*
* @return scalar|array<string|int|float|bool|array|null> The stored value.
* @throws Key_Not_Found_Exception When the key is not found or has expired.
* @throws Corrupted_Value_Exception When the stored value cannot be decoded from JSON.
*/
public function get_for_multisite( string $key ) {
return $this->do_get( $key );
}
/**
* Checks whether a non-expired value exists for a blog-scoped key.
*
* @param string $key The key.
*
* @return bool
*/
public function has( string $key ): bool {
return $this->do_has( $this->prefix_for_blog( $key ) );
}
/**
* Checks whether a non-expired value exists for a user-scoped key.
*
* @param string $key The key.
* @param int $user_id The user ID. Defaults to the current user.
*
* @return bool
* @throws No_Current_User_Exception When no user ID is given and no user is logged in.
*/
public function has_for_user( string $key, int $user_id = 0 ): bool {
return $this->do_has( $this->prefix_for_user( $key, $user_id ) );
}
/**
* Checks whether a non-expired value exists for a multisite-scoped key.
*
* @param string $key The key.
*
* @return bool
*/
public function has_for_multisite( string $key ): bool {
return $this->do_has( $key );
}
/**
* Deletes a value scoped to the current blog.
*
* @param string $key The key.
*
* @return void
*/
public function delete( string $key ): void {
$this->repository->delete( $this->prefix_for_blog( $key ) );
}
/**
* Deletes a value scoped to a user.
*
* @param string $key The key.
* @param int $user_id The user ID. Defaults to the current user.
*
* @return void
* @throws No_Current_User_Exception When no user ID is given and no user is logged in.
*/
public function delete_for_user( string $key, int $user_id = 0 ): void {
$this->repository->delete( $this->prefix_for_user( $key, $user_id ) );
}
/**
* Deletes a value shared across the entire multisite network.
*
* @param string $key The key.
*
* @return void
*/
public function delete_for_multisite( string $key ): void {
$this->repository->delete( $key );
}
/**
* Cleans up all expired entries.
*
* @return int The number of deleted entries.
*/
public function cleanup_expired(): int {
return $this->repository->delete_expired( $this->current_datetime() );
}
/**
* Persists a value with the given prefixed key.
*
* @param string $prefixed_key The prefixed key.
* @param string|int|float|bool|array<string|int|float|bool|array|null>|JsonSerializable $value The value to store.
* @param int $ttl_in_seconds The time-to-live in seconds.
*
* @return void
* @throws InvalidArgumentException When the value is not JSON-encodable.
*/
private function do_persist( string $prefixed_key, $value, int $ttl_in_seconds ): void {
$json = $this->json_encode_value( $value );
$exp = \gmdate( 'Y-m-d H:i:s', ( $this->date_helper->current_time() + $ttl_in_seconds ) );
$this->repository->upsert( $prefixed_key, $json, $exp );
}
/**
* Persists a value only if the prefixed key does not already exist.
*
* @param string $prefixed_key The prefixed key.
* @param string|int|float|bool|array<string|int|float|bool|array|null>|JsonSerializable $value The value to store.
* @param int $ttl_in_seconds The time-to-live in seconds.
*
* @return bool True if the value was inserted, false if the key already exists.
* @throws InvalidArgumentException When the value is not JSON-encodable.
*/
private function do_persist_if_absent( string $prefixed_key, $value, int $ttl_in_seconds ): bool {
$json = $this->json_encode_value( $value );
$now = $this->date_helper->current_time();
$exp = \gmdate( 'Y-m-d H:i:s', ( $now + $ttl_in_seconds ) );
return $this->repository->insert_if_absent( $prefixed_key, $json, $exp, \gmdate( 'Y-m-d H:i:s', $now ) );
}
/**
* Gets and decodes a value by prefixed key.
*
* @param string $prefixed_key The prefixed key.
*
* @return string|int|float|bool|array<string|int|float|bool|array|null> The stored value.
* @throws Key_Not_Found_Exception When the key is not found or has expired.
* @throws Corrupted_Value_Exception When the stored value cannot be decoded from JSON.
*/
private function do_get( string $prefixed_key ) {
$json = $this->repository->find( $prefixed_key, $this->current_datetime() );
if ( $json === null ) {
// phpcs:ignore WordPress.Security.EscapeOutput.ExceptionNotEscaped -- Internal exception message.
throw new Key_Not_Found_Exception( "Key '{$prefixed_key}' not found or expired." );
}
try {
return \json_decode( $json, true, 512, \JSON_THROW_ON_ERROR );
} catch ( JsonException $e ) {
// phpcs:ignore WordPress.Security.EscapeOutput.ExceptionNotEscaped -- This is an exception message, not output.
throw new Corrupted_Value_Exception( $prefixed_key, $e->getMessage() );
}
}
/**
* Checks whether a non-expired value exists for the given prefixed key.
*
* @param string $prefixed_key The prefixed key.
*
* @return bool
*/
private function do_has( string $prefixed_key ): bool {
return $this->repository->find( $prefixed_key, $this->current_datetime() ) !== null;
}
/**
* JSON-encodes a value.
*
* @param string|int|float|bool|array<string|int|float|bool|array|null>|JsonSerializable $value The value to encode.
*
* @return string The JSON-encoded value.
* @throws InvalidArgumentException When the value is not JSON-encodable.
*/
private function json_encode_value( $value ): string {
// phpcs:ignore Yoast.Yoast.JsonEncodeAlternative.Found -- WPSEO_Utils::format_json_encode we don't intend to output this.
$encoded = \wp_json_encode( $value );
if ( $encoded === false ) {
// phpcs:ignore WordPress.Security.EscapeOutput.ExceptionNotEscaped -- This is an exception message, not output.
throw new InvalidArgumentException( 'Expiring_Store: value must be JSON-encodable. ' . \json_last_error_msg() );
}
return $encoded;
}
/**
* Prefixes a key for blog scope.
*
* @param string $key The key.
*
* @return string The prefixed key.
*/
private function prefix_for_blog( string $key ): string {
return 'blog_' . \get_current_blog_id() . ':' . $key;
}
/**
* Prefixes a key for user scope.
*
* @param string $key The key.
* @param int $user_id The user ID. When 0, falls back to the current user.
*
* @return string The prefixed key.
* @throws No_Current_User_Exception When no user ID is given and no user is logged in.
*/
private function prefix_for_user( string $key, int $user_id = 0 ): string {
if ( $user_id <= 0 ) {
$user_id = \get_current_user_id();
}
if ( $user_id === 0 ) {
throw new No_Current_User_Exception( 'Cannot use user-scoped expiring store methods without a logged-in user.' );
}
return 'user_' . $user_id . ':' . $key;
}
/**
* Returns the current datetime in 'Y-m-d H:i:s' format.
*
* @return string The current datetime.
*/
private function current_datetime(): string {
return \gmdate( 'Y-m-d H:i:s', $this->date_helper->current_time() );
}
}