add( $key, $data, $group, (int) $expire ); } /** * Closes the cache. * * This function has ceased to do anything since WordPress 2.5. The * functionality was removed along with the rest of the persistent cache. This * does not mean that plugins can't implement this function when they need to * make sure that the cache is cleaned up after WordPress no longer needs it. * * @return bool Always returns True */ function wp_cache_close() { return true; } /** * Decrement numeric cache item's value * * @uses $wp_object_cache Object Cache Class * @see WP_Object_Cache::decr() * * @param int|string $key The cache key to increment * @param int $offset The amount by which to decrement the item's value. Default is 1. * @param string $group The group the key is in. * @return false|int False on failure, the item's new value on success. */ function wp_cache_decr( $key, $offset = 1, $group = '' ) { global $wp_object_cache; return $wp_object_cache->decr( $key, $offset, $group ); } /** * Removes the cache contents matching key and group. * * @uses $wp_object_cache Object Cache Class * @see WP_Object_Cache::delete() * * @param int|string $key What the contents in the cache are called * @param string $group Where the cache contents are grouped * @return bool True on successful removal, false on failure */ function wp_cache_delete( $key, $group = '' ) { global $wp_object_cache; return $wp_object_cache->delete( $key, $group ); } /** * Removes cache contents for a given group. * * @uses $wp_object_cache Object Cache Class * @see WP_Object_Cache::delete_group() * * @param string $group Where the cache contents are grouped * @return bool True on successful removal, false on failure */ function wp_cache_delete_group( $group ) { global $wp_object_cache; return $wp_object_cache->delete_group( $group ); } /** * Removes all cache items. * * @uses $wp_object_cache Object Cache Class * @see WP_Object_Cache::flush() * * @return bool False on failure, true on success */ function wp_cache_flush() { global $wp_object_cache; return $wp_object_cache->flush(); } /** * Retrieves the cache contents from the cache by key and group. * * @uses $wp_object_cache Object Cache Class * @see WP_Object_Cache::get() * * @param int|string $key What the contents in the cache are called * @param string $group Where the cache contents are grouped * @param bool $force Whether to force an update of the local cache from the persistent cache (default is false) * @param &bool $found Whether key was found in the cache. Disambiguates a return of false, a storable value. * @return bool|mixed False on failure to retrieve contents or the cache contents on success */ function wp_cache_get( $key, $group = '', $force = false, &$found = null ) { global $wp_object_cache; return $wp_object_cache->get( $key, $group, $force, $found ); } /** * Retrieves multiple values from the cache in one call. * * @see WP_Object_Cache::get_multiple() * @global WP_Object_Cache $wp_object_cache Object cache global instance. * * @param array $keys Array of keys under which the cache contents are stored. * @param string $group Optional. Where the cache contents are grouped. Default empty. * @param bool $force Optional. Whether to force an update of the local cache * from the persistent cache. Default false. * @return array Array of values organized into groups. */ function wp_cache_get_multiple( $keys, $group = '', $force = false ) { global $wp_object_cache; return $wp_object_cache->get_multiple( $keys, $group, $force ); } /** * Removes all cache items from the in-memory runtime cache. * * @see WP_Object_Cache::flush() * * @return bool True on success, false on failure. */ function wp_cache_flush_runtime() { global $wp_object_cache; return $wp_object_cache->flush( false ); } /** * Removes all cache items in a group, if the object cache implementation supports it. * * Before calling this function, always check for group flushing support using the * `wp_cache_supports( 'flush_group' )` function. * * @see WP_Object_Cache::flush_group() * @global WP_Object_Cache $wp_object_cache Object cache global instance. * * @param string $group Name of group to remove from cache. * @return bool True if group was flushed, false otherwise. */ function wp_cache_flush_group( $group ) { global $wp_object_cache; return $wp_object_cache->flush_group( $group ); } /** * Increment numeric cache item's value * * @uses $wp_object_cache Object Cache Class * @see WP_Object_Cache::incr() * * @param int|string $key The cache key to increment * @param int $offset The amount by which to increment the item's value. Default is 1. * @param string $group The group the key is in. * @return false|int False on failure, the item's new value on success. */ function wp_cache_incr( $key, $offset = 1, $group = '' ) { global $wp_object_cache; return $wp_object_cache->incr( $key, $offset, $group ); } /** * Sets up Object Cache Global and assigns it. * * @global WP_Object_Cache $wp_object_cache WordPress Object Cache */ function wp_cache_init() { global $wp_object_cache; if ( ! ( $wp_object_cache instanceof WP_Object_Cache ) ) { $wp_object_cache = new WP_Object_Cache(); } } /** * Replaces the contents of the cache with new data. * * @uses $wp_object_cache Object Cache Class * @see WP_Object_Cache::replace() * * @param int|string $key What to call the contents in the cache * @param mixed $data The contents to store in the cache * @param string $group Where to group the cache contents * @param int $expire When to expire the cache contents * @return bool False if not exists, true if contents were replaced */ function wp_cache_replace( $key, $data, $group = '', $expire = WP_REDIS_DEFAULT_EXPIRE_SECONDS ) { global $wp_object_cache; return $wp_object_cache->replace( $key, $data, $group, (int) $expire ); } /** * Saves the data to the cache. * * @uses $wp_object_cache Object Cache Class * @see WP_Object_Cache::set() * * @param int|string $key What to call the contents in the cache * @param mixed $data The contents to store in the cache * @param string $group Where to group the cache contents * @param int $expire When to expire the cache contents * @return bool False on failure, true on success */ function wp_cache_set( $key, $data, $group = '', $expire = WP_REDIS_DEFAULT_EXPIRE_SECONDS ) { global $wp_object_cache; return $wp_object_cache->set( $key, $data, $group, (int) $expire ); } /** * Switch the internal blog id. * * This changes the blog id used to create keys in blog specific groups. * * @param int $blog_id Blog ID */ function wp_cache_switch_to_blog( $blog_id ) { global $wp_object_cache; return $wp_object_cache->switch_to_blog( $blog_id ); } /** * Adds a group or set of groups to the list of global groups. * * @param string|array $groups A group or an array of groups to add */ function wp_cache_add_global_groups( $groups ) { global $wp_object_cache; return $wp_object_cache->add_global_groups( $groups ); } /** * Adds a group or set of groups to the list of non-persistent groups. * * @param string|array $groups A group or an array of groups to add */ function wp_cache_add_non_persistent_groups( $groups ) { global $wp_object_cache; $wp_object_cache->add_non_persistent_groups( $groups ); } /** * Adds a group or set of groups to the list of groups that use Redis hashes. * * @param string|array $groups A group or an array of groups to add. */ function wp_cache_add_redis_hash_groups( $groups ) { global $wp_object_cache; $wp_object_cache->add_redis_hash_groups( $groups ); } /** * Reset internal cache keys and structures. If the cache backend uses global * blog or site IDs as part of its cache keys, this function instructs the * backend to reset those keys and perform any cleanup since blog or site IDs * have changed since cache init. * * This function is deprecated. Use wp_cache_switch_to_blog() instead of this * function when preparing the cache for a blog switch. For clearing the cache * during unit tests, consider using wp_cache_init(). wp_cache_init() is not * recommended outside of unit tests as the performance penalty for using it is * high. * * @deprecated 3.5.0 */ function wp_cache_reset() { _deprecated_function( __FUNCTION__, '3.5' ); global $wp_object_cache; return $wp_object_cache->reset(); } /** * Determines whether the object cache implementation supports a particular feature. * * @since 6.1.0 * * @param string $feature Name of the feature to check for. Possible values include: * 'add_multiple', 'set_multiple', 'get_multiple', 'delete_multiple', * 'flush_runtime', 'flush_group'. * @return bool True if the feature is supported, false otherwise. */ function wp_cache_supports( $feature ) { switch ( $feature ) { case 'get_multiple': case 'flush_runtime': case 'flush_group': return true; case 'add_multiple': case 'set_multiple': case 'delete_multiple': default: return false; } } /** * WordPress Object Cache * * The WordPress Object Cache is used to save on trips to the database. The * Object Cache stores all of the cache data to memory and makes the cache * contents available by using a key, which is used to name and later retrieve * the cache contents. * * The Object Cache can be replaced by other caching mechanisms by placing files * in the wp-content folder which is looked at in wp-settings. If that file * exists, then this file will not be included. */ #[AllowDynamicProperties] class WP_Object_Cache { /** * Holds the cached objects * * @var array */ public $cache = []; /** * The amount of times the cache data was already stored in the cache. * * @var int */ public $cache_hits = 0; /** * Amount of times the cache did not have the request in cache * * @var int */ public $cache_misses = 0; /** * The amount of times a request was made to Redis * * @var int */ public $redis_calls = []; /** * List of global groups * * @var array */ public $global_groups = []; /** * List of non-persistent groups * * @var array */ public $non_persistent_groups = []; /** * List of groups which use Redis hashes. * * @var array */ public $redis_hash_groups = []; /** * The blog prefix to prepend to keys in non-global groups. * * @var int */ public $blog_prefix; /** * Whether or not Redis is connected * * @var bool */ public $is_redis_connected = false; /** * Whether or not the object cache thinks Redis needs a flush * * @var bool */ public $do_redis_failback_flush = false; /** * The last triggered error * * @var string */ public $last_triggered_error = ''; /** * Whether or not to use true cache groups, instead of flattening. * * @var bool */ const USE_GROUPS = WP_REDIS_USE_CACHE_GROUPS; /** * Adds data to the cache if it doesn't already exist. * * @uses WP_Object_Cache::_exists Checks to see if the cache already has data. * @uses WP_Object_Cache::set Sets the data after the checking the cache * contents existence. * * @param int|string $key What to call the contents in the cache * @param mixed $data The contents to store in the cache * @param string $group Where to group the cache contents * @param int $expire When to expire the cache contents * @return bool False if cache key and group already exist, true on success */ public function add( $key, $data, $group = 'default', $expire = WP_REDIS_DEFAULT_EXPIRE_SECONDS ) { if ( empty( $group ) ) { $group = 'default'; } if ( function_exists( 'wp_suspend_cache_addition' ) && wp_suspend_cache_addition() ) { return false; } if ( $this->_exists( $key, $group ) ) { return false; } return $this->set( $key, $data, $group, (int) $expire ); } /** * Sets the list of global groups. * * @param array $groups List of groups that are global. */ public function add_global_groups( $groups ) { $groups = (array) $groups; // Allow force ignoring of global groups. if ( is_array( WP_REDIS_IGNORE_GLOBAL_GROUPS ) ) { $groups = array_diff( $groups, WP_REDIS_IGNORE_GLOBAL_GROUPS ); } $groups = array_fill_keys( $groups, true ); $this->global_groups = array_merge( $this->global_groups, $groups ); } /** * Sets the list of non-persistent groups. * * @param array $groups List of groups that are non-persistent. */ public function add_non_persistent_groups( $groups ) { $groups = (array) $groups; $groups = array_fill_keys( $groups, true ); $this->non_persistent_groups = array_merge( $this->non_persistent_groups, $groups ); } /** * Sets the list of groups that use Redis hashes. * * @param array $groups List of groups that use Redis hashes. */ public function add_redis_hash_groups( $groups ) { $groups = (array) $groups; $groups = array_fill_keys( $groups, true ); $this->redis_hash_groups = array_merge( $this->redis_hash_groups, $groups ); } /** * Decrement numeric cache item's value * * @param int|string $key The cache key to increment * @param int $offset The amount by which to decrement the item's value. Default is 1. * @param string $group The group the key is in. * @return false|int False on failure, the item's new value on success. */ public function decr( $key, $offset = 1, $group = 'default' ) { if ( empty( $group ) ) { $group = 'default'; } // The key needs to exist in order to be decremented. if ( ! $this->_exists( $key, $group ) ) { return false; } $offset = (int) $offset; // If this isn't a persistent group, we have to sort this out ourselves, grumble grumble. if ( ! $this->_should_persist( $group ) ) { $existing = $this->_get_internal( $key, $group ); if ( empty( $existing ) || ! is_numeric( $existing ) ) { $existing = 0; } else { $existing -= $offset; } if ( $existing < 0 ) { $existing = 0; } $this->_set_internal( $key, $group, $existing ); return $existing; } if ( $this->_should_use_redis_hashes( $group ) ) { $redis_safe_group = $this->_key( '', $group ); $result = $this->_call_redis( 'hIncrBy', $redis_safe_group, $key, -$offset, $group ); if ( $result < 0 ) { $result = 0; $this->_call_redis( 'hSet', $redis_safe_group, $key, $result ); } } else { $id = $this->_key( $key, $group ); $result = $this->_call_redis( 'decrBy', $id, $offset ); if ( $result < 0 ) { $result = 0; $this->_call_redis( 'set', $id, $result ); } } if ( is_int( $result ) ) { $this->_set_internal( $key, $group, $result ); } return $result; } /** * Remove the contents of the cache key in the group * * If the cache key does not exist in the group and $force parameter is set * to false, then nothing will happen. The $force parameter is set to false * by default. * * @param int|string $key What the contents in the cache are called * @param string $group Where the cache contents are grouped * @param bool $force Optional. Whether to force the unsetting of the cache * key in the group * @return bool False if the contents weren't deleted and true on success */ public function delete( $key, $group = 'default', $force = false ) { if ( empty( $group ) ) { $group = 'default'; } if ( ! $force && ! $this->_exists( $key, $group ) ) { return false; } if ( $this->_should_persist( $group ) ) { if ( $this->_should_use_redis_hashes( $group ) ) { $redis_safe_group = $this->_key( '', $group ); $result = $this->_call_redis( 'hDel', $redis_safe_group, $key ); } else { $id = $this->_key( $key, $group ); $result = $this->_call_redis( 'del', $id ); } if ( 1 !== $result ) { return false; } } $this->_unset_internal( $key, $group ); return true; } /** * Remove the contents of all cache keys in the group. * * @param string $group Where the cache contents are grouped. * @return boolean True on success, false on failure. */ public function delete_group( $group ) { if ( ! $this->_should_use_redis_hashes( $group ) ) { return false; } $multisite_safe_group = $this->multisite && ! isset( $this->global_groups[ $group ] ) ? $this->blog_prefix . $group : $group; $redis_safe_group = $this->_key( '', $group ); if ( $this->_should_persist( $group ) ) { $result = $this->_call_redis( 'del', $redis_safe_group ); if ( 1 !== $result ) { return false; } } elseif ( ! $this->_should_persist( $group ) && ! isset( $this->cache[ $multisite_safe_group ] ) ) { return false; } unset( $this->cache[ $multisite_safe_group ] ); return true; } /** * Clears the object cache of all data. * * By default, this will flush the session cache as well as Redis, but we * can leave the redis cache intact if we want. This is helpful when, for * instance, you're running a batch process and want to clear the session * store to reduce the memory footprint, but you don't want to have to * re-fetch all the values from the database. * * @param bool $redis Should we flush redis as well as the session cache? * @return bool Always returns true */ public function flush( $redis = true ) { $this->cache = []; if ( $redis ) { $this->_call_redis( 'flushdb' ); } return true; } /** * Removes all cache items in a group. * * @param string $group Name of group to remove from cache. * @return true Always returns true. */ public function flush_group( $group ) { if ( ! $this->_should_use_redis_hashes( $group ) ) { return false; } $multisite_safe_group = $this->multisite && ! isset( $this->global_groups[ $group ] ) ? $this->blog_prefix . $group : $group; $redis_safe_group = $this->_key( '', $group ); if ( $this->_should_persist( $group ) ) { $result = $this->_call_redis( 'del', $redis_safe_group ); if ( 1 !== $result ) { return false; } } elseif ( ! $this->_should_persist( $group ) && ! isset( $this->cache[ $multisite_safe_group ] ) ) { return false; } unset( $this->cache[ $multisite_safe_group ] ); return true; } /** * Retrieves the cache contents, if it exists * * The contents will be first attempted to be retrieved by searching by the * key in the cache group. If the cache is hit (success) then the contents * are returned. * * On failure, the number of cache misses will be incremented. * * @param int|string $key What the contents in the cache are called * @param string $group Where the cache contents are grouped * @param string $force Whether to force a refetch rather than relying on the local cache (default is false) * @param bool $found Optional. Whether the key was found in the cache. Disambiguates a return of false, a storable value. Passed by reference. Default null. * @return bool|mixed False on failure to retrieve contents or the cache contents on success */ public function get( $key, $group = 'default', $force = false, &$found = null ) { if ( empty( $group ) ) { $group = 'default'; } // Key is set internally, so we can use this value. if ( $this->_isset_internal( $key, $group ) && ! $force ) { $this->cache_hits += 1; $found = true; return $this->_get_internal( $key, $group ); } // Not a persistent group, so don't try Redis if the value doesn't exist internally. if ( ! $this->_should_persist( $group ) ) { $this->cache_misses += 1; $found = false; return false; } if ( $this->_should_use_redis_hashes( $group ) ) { $redis_safe_group = $this->_key( '', $group ); $value = $this->_call_redis( 'hGet', $redis_safe_group, $key ); } else { $id = $this->_key( $key, $group ); $value = $this->_call_redis( 'get', $id ); } // PhpRedis returns `false` when the key doesn't exist. if ( false === $value ) { $this->cache_misses += 1; $found = false; return false; } // All non-numeric values are serialized. $value = is_numeric( $value ) ? intval( $value ) : unserialize( $value ); $this->_set_internal( $key, $group, $value ); $this->cache_hits += 1; $found = true; return $value; } /** * Retrieves multiple values from the cache in one call. * * @param array $keys Array of keys under which the cache contents are stored. * @param string $group Optional. Where the cache contents are grouped. Default empty. * @param bool $force Optional. Whether to force an update of the local cache * from the persistent cache. Default false. * @return array Array of values organized into groups. */ public function get_multiple( $keys, $group = 'default', $force = false ) { if ( empty( $group ) ) { $group = 'default'; } // Get unique keys. $keys = array_unique( $keys ); $cache = []; if ( ! $this->_should_persist( $group ) ) { foreach ( $keys as $key ) { $cache[ $key ] = $this->_isset_internal( $key, $group ) ? $this->_get_internal( $key, $group ) : false; false !== $cache[ $key ] ? $this->cache_hits++ : $this->cache_misses++; } return $cache; } // Attempt to fetch values from the internal cache. if ( ! $force ) { foreach ( $keys as $key ) { if ( $this->_isset_internal( $key, $group ) ) { $cache[ $key ] = $this->_get_internal( $key, $group ); ++$this->cache_hits; } } } $remaining_keys = array_values( array_diff( $keys, array_keys( $cache ) ) ); // If all keys were satisfied by the internal cache, we're sorted. if ( empty( $remaining_keys ) ) { return $cache; } if ( $this->_should_use_redis_hashes( $group ) ) { $redis_safe_group = $this->_key( '', $group ); $results = $this->_call_redis( 'hmGet', $redis_safe_group, $remaining_keys ); $results = is_array( $results ) ? array_values( $results ) : $results; } else { $ids = []; foreach ( $remaining_keys as $key ) { $ids[] = $this->_key( $key, $group ); } $results = $this->_call_redis( 'mget', $ids ); } // Process the results from the Redis call. foreach ( $remaining_keys as $i => $key ) { $value = isset( $results[ $i ] ) ? $results[ $i ] : false; if ( false !== $value ) { // All non-numeric values are serialized. $value = is_numeric( $value ) ? intval( $value ) : unserialize( $value ); $this->_set_internal( $key, $group, $value ); ++$this->cache_hits; } else { ++$this->cache_misses; } $cache[ $key ] = $value; } // Make sure return values are returned in the order of the passed keys. $return_cache = []; foreach ( $keys as $key ) { $return_cache[ $key ] = isset( $cache[ $key ] ) ? $cache[ $key ] : false; } return $return_cache; } /** * Increment numeric cache item's value * * @param int|string $key The cache key to increment * @param int $offset The amount by which to increment the item's value. Default is 1. * @param string $group The group the key is in. * @return false|int False on failure, the item's new value on success. */ public function incr( $key, $offset = 1, $group = 'default' ) { if ( empty( $group ) ) { $group = 'default'; } // The key needs to exist in order to be incremented. if ( ! $this->_exists( $key, $group ) ) { return false; } $offset = (int) $offset; // If this isn't a persistent group, we have to sort this out ourselves, grumble grumble. if ( ! $this->_should_persist( $group ) ) { $existing = $this->_get_internal( $key, $group ); if ( empty( $existing ) || ! is_numeric( $existing ) ) { $existing = 1; } else { $existing += $offset; } if ( $existing < 0 ) { $existing = 0; } $this->_set_internal( $key, $group, $existing ); return $existing; } if ( $this->_should_use_redis_hashes( $group ) ) { $redis_safe_group = $this->_key( '', $group ); $result = $this->_call_redis( 'hIncrBy', $redis_safe_group, $key, $offset, $group ); if ( $result < 0 ) { $result = 0; $this->_call_redis( 'hSet', $redis_safe_group, $key, $result ); } } else { $id = $this->_key( $key, $group ); $result = $this->_call_redis( 'incrBy', $id, $offset ); if ( $result < 0 ) { $result = 0; $this->_call_redis( 'set', $id, $result ); } } if ( is_int( $result ) ) { $this->_set_internal( $key, $group, $result ); } return $result; } /** * Replace the contents in the cache, if contents already exist * * @see WP_Object_Cache::set() * * @param int|string $key What to call the contents in the cache * @param mixed $data The contents to store in the cache * @param string $group Where to group the cache contents * @param int $expire When to expire the cache contents * @return bool False if not exists, true if contents were replaced */ public function replace( $key, $data, $group = 'default', $expire = WP_REDIS_DEFAULT_EXPIRE_SECONDS ) { if ( empty( $group ) ) { $group = 'default'; } if ( ! $this->_exists( $key, $group ) ) { return false; } return $this->set( $key, $data, $group, (int) $expire ); } /** * Reset keys * * @deprecated 3.5.0 */ public function reset() { _deprecated_function( __FUNCTION__, '3.5', 'switch_to_blog()' ); } /** * Sets the data contents into the cache * * The cache contents is grouped by the $group parameter followed by the * $key. This allows for duplicate ids in unique groups. Therefore, naming of * the group should be used with care and should follow normal function * naming guidelines outside of core WordPress usage. * * The $expire parameter is not used, because the cache will automatically * expire for each time a page is accessed and PHP finishes. The method is * more for cache plugins which use files. * * @param int|string $key What to call the contents in the cache * @param mixed $data The contents to store in the cache * @param string $group Where to group the cache contents * @param int $expire TTL for the data, in seconds * @return bool Always returns true */ public function set( $key, $data, $group = 'default', $expire = WP_REDIS_DEFAULT_EXPIRE_SECONDS ) { if ( empty( $group ) ) { $group = 'default'; } if ( is_object( $data ) ) { $data = clone $data; } $this->_set_internal( $key, $group, $data ); if ( ! $this->_should_persist( $group ) ) { return true; } // If this is an integer, store it as such. Otherwise, serialize it. if ( ! is_numeric( $data ) || intval( $data ) !== $data ) { $data = serialize( $data ); } // Redis doesn't support expire on hash group keys. if ( $this->_should_use_redis_hashes( $group ) ) { $redis_safe_group = $this->_key( '', $group ); $this->_call_redis( 'hSet', $redis_safe_group, $key, $data ); return true; } $id = $this->_key( $key, $group ); if ( empty( $expire ) ) { $this->_call_redis( 'set', $id, $data ); } else { $this->_call_redis( 'setex', $id, $expire, $data ); } return true; } /** * Echoes the stats of the caching. * * Gives the cache hits, and cache misses. Also prints every cached group, * key and the data. */ public function stats() { $total_redis_calls = 0; foreach ( $this->redis_calls as $method => $calls ) { $total_redis_calls += $calls; } $out = []; $out[] = '
';
$out[] = 'Cache Hits:' . (int) $this->cache_hits . '
';
$out[] = 'Cache Misses:' . (int) $this->cache_misses . '
';
$out[] = 'Redis Client:' . get_class( $this->redis ) . '
';
$out[] = 'Redis Calls:' . (int) $total_redis_calls . ':
';
foreach ( $this->redis_calls as $method => $calls ) {
$out[] = ' - ' . esc_html( $method ) . ': ' . (int) $calls . '
';
}
$out[] = '