add($key, $value, $group, $expiration); } /** * 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 a numeric item's value. * * @param string $key The key under which to store the value. * @param int $offset The amount by which to decrement the item's value. * @param string $group The group value appended to the $key. * * @global WP_Object_Cache $wp_object_cache * * @return int|bool Returns item's new value on success or FALSE on failure. */ function wp_cache_decr($key, $offset = 1, $group = '') { global $wp_object_cache; return $wp_object_cache->decrement($key, $offset, $group); } /** * Remove the item from the cache. * * @param string $key The key under which to store the value. * @param string $group The group value appended to the $key. * @param int $time The amount of time the server will wait to delete the item in seconds. * * @global WP_Object_Cache $wp_object_cache * * @return bool Returns TRUE on success or FALSE on failure. */ function wp_cache_delete($key, $group = '', $time = 0) { global $wp_object_cache; return $wp_object_cache->delete($key, $group, $time); } /** * Invalidate all items in the cache. If `WP_REDIS_SELECTIVE_FLUSH` is `true`, * only keys prefixed with the `WP_CACHE_KEY_SALT` are flushed. * * @param int $delay Number of seconds to wait before invalidating the items. * * @global WP_Object_Cache $wp_object_cache * * @return bool Returns TRUE on success or FALSE on failure. */ function wp_cache_flush($delay = 0) { global $wp_object_cache; return $wp_object_cache->flush($delay); } /** * Retrieve object from cache. * * Gets an object from cache based on $key and $group. * * @param string $key The key under which to store the value. * @param string $group The group value appended to the $key. * @param bool $force Optional. Whether to force an update of the local cache from the persistent * cache. Default 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. * * @global WP_Object_Cache $wp_object_cache * * @return bool|mixed Cached object value. */ function wp_cache_get($key, $group = '', $force = false, &$found = null) { global $wp_object_cache; return $wp_object_cache->get($key, $group, $force, $found); } /** * Retrieve multiple values from cache. * * Gets multiple values from cache, including across multiple groups * * Usage: array( 'group0' => array( 'key0', 'key1', 'key2', ), 'group1' => array( 'key0' ) ) * * Mirrors the Memcached Object Cache plugin's argument and return-value formats * * @param array $groups Array of groups and keys to retrieve * * @global WP_Object_Cache $wp_object_cache * * @return bool|mixed Array of cached values, keys in the format $group:$key. Non-existent keys false */ function wp_cache_get_multi($groups) { global $wp_object_cache; return $wp_object_cache->get_multi($groups); } /** * Increment a numeric item's value. * * @param string $key The key under which to store the value. * @param int $offset The amount by which to increment the item's value. * @param string $group The group value appended to the $key. * * @global WP_Object_Cache $wp_object_cache * * @return int|bool Returns item's new value on success or FALSE on failure. */ function wp_cache_incr($key, $offset = 1, $group = '') { global $wp_object_cache; return $wp_object_cache->increment($key, $offset, $group); } /** * Sets up Object Cache Global and assigns it. * * @global WP_Object_Cache $wp_object_cache WordPress Object Cache * * @return void */ function wp_cache_init() { global $wp_object_cache; if (! ($wp_object_cache instanceof WP_Object_Cache)) { $wp_object_cache = new WP_Object_Cache; } } /** * Replaces a value in cache. * * This method is similar to "add"; however, is does not successfully set a value if * the object's key is not already set in cache. * * @param string $key The key under which to store the value. * @param mixed $value The value to store. * @param string $group The group value appended to the $key. * @param int $expiration The expiration time, defaults to 0. * * @global WP_Object_Cache $wp_object_cache * * @return bool Returns TRUE on success or FALSE on failure. */ function wp_cache_replace($key, $value, $group = '', $expiration = 0) { global $wp_object_cache; return $wp_object_cache->replace($key, $value, $group, $expiration); } /** * Sets a value in cache. * * The value is set whether or not this key already exists in Redis. * * @param string $key The key under which to store the value. * @param mixed $value The value to store. * @param string $group The group value appended to the $key. * @param int $expiration The expiration time, defaults to 0. * * @global WP_Object_Cache $wp_object_cache * * @return bool Returns TRUE on success or FALSE on failure. */ function wp_cache_set($key, $value, $group = '', $expiration = 0) { global $wp_object_cache; return $wp_object_cache->set($key, $value, $group, $expiration); } /** * Switch the interal blog id. * * This changes the blog id used to create keys in blog specific groups. * * @param int $_blog_id Blog ID * * @global WP_Object_Cache $wp_object_cache * * @return bool */ 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 Redis groups. * * @param string|array $groups A group or an array of groups to add. * * @global WP_Object_Cache $wp_object_cache * * @return void */ function wp_cache_add_global_groups($groups) { global $wp_object_cache; $wp_object_cache->add_global_groups($groups); } /** * Adds a group or set of groups to the list of non-Redis groups. * * @param string|array $groups A group or an array of groups to add. * * @global WP_Object_Cache $wp_object_cache * * @return void */ function wp_cache_add_non_persistent_groups($groups) { global $wp_object_cache; $wp_object_cache->add_non_persistent_groups($groups); } class WP_Object_Cache { /** * The Redis client. * * @var mixed */ private $redis; /** * Track if Redis is available * * @var bool */ private $redis_connected = false; /** * Holds the non-Redis objects. * * @var array */ public $cache = array(); /** * Name of the used Redis client * * @var bool */ public $redis_client = null; /** * List of global groups. * * @var array */ public $global_groups = array( 'blog-details', 'blog-id-cache', 'blog-lookup', 'global-posts', 'networks', 'rss', 'sites', 'site-details', 'site-lookup', 'site-options', 'site-transient', 'users', 'useremail', 'userlogins', 'usermeta', 'user_meta', 'userslugs', ); /** * List of groups not saved to Redis. * * @var array */ public $ignored_groups = array( 'counts', 'plugins' ); /** * Prefix used for global groups. * * @var string */ public $global_prefix = ''; /** * Prefix used for non-global groups. * * @var string */ public $blog_prefix = ''; /** * Track how many requests were found in cache * * @var int */ public $cache_hits = 0; /** * Track how may requests were not cached * * @var int */ public $cache_misses = 0; /** * Instantiate the Redis class. * * @param bool $fail_gracefully */ public function __construct($fail_gracefully = true) { global $blog_id, $table_prefix; $parameters = array( 'scheme' => 'tcp', 'host' => '127.0.0.1', 'port' => 6379 ); foreach (array( 'scheme', 'host', 'port', 'path', 'password', 'database' ) as $setting) { $constant = sprintf('WP_REDIS_%s', strtoupper($setting)); if (defined($constant)) { $parameters[ $setting ] = constant($constant); } } if (defined('WP_REDIS_GLOBAL_GROUPS') && is_array(WP_REDIS_GLOBAL_GROUPS)) { $this->global_groups = WP_REDIS_GLOBAL_GROUPS; } if (defined('WP_REDIS_IGNORED_GROUPS') && is_array(WP_REDIS_IGNORED_GROUPS)) { $this->ignored_groups = WP_REDIS_IGNORED_GROUPS; } $client = defined('WP_REDIS_CLIENT') ? WP_REDIS_CLIENT : null; if (class_exists('Redis') && strcasecmp('predis', $client) !== 0) { $client = defined('HHVM_VERSION') ? 'hhvm' : 'pecl'; } else { $client = 'predis'; } try { if (strcasecmp('hhvm', $client) === 0) { $this->redis_client = sprintf('HHVM Extension (v%s)', HHVM_VERSION); $this->redis = new Redis(); // Adjust host and port, if the scheme is `unix` if (strcasecmp('unix', $parameters[ 'scheme' ]) === 0) { $parameters[ 'host' ] = 'unix://' . $parameters[ 'path' ]; $parameters[ 'port' ] = 0; } $this->redis->connect($parameters[ 'host' ], $parameters[ 'port' ]); } if (strcasecmp('pecl', $client) === 0) { $this->redis_client = sprintf('PhpRedis (v%s)', phpversion('redis')); if (defined('WP_REDIS_SHARDS')) { $this->redis = new RedisArray(array_values(WP_REDIS_SHARDS)); } elseif (defined('WP_REDIS_CLUSTER')) { $this->redis = new RedisCluster(null, array_values(WP_REDIS_CLUSTER)); } else { $this->redis = new Redis(); if (strcasecmp('unix', $parameters[ 'scheme' ]) === 0) { $this->redis->connect($parameters[ 'path' ]); } else { $this->redis->connect($parameters[ 'host' ], $parameters[ 'port' ]); } } } if (strcasecmp('pecl', $client) === 0 || strcasecmp('hhvm', $client) === 0) { if (isset($parameters[ 'password' ])) { $this->redis->auth($parameters[ 'password' ]); } if (isset($parameters[ 'database' ])) { $this->redis->select($parameters[ 'database' ]); } } if (strcasecmp('predis', $client) === 0) { $this->redis_client = 'Predis'; // Require PHP 5.4 or greater if (version_compare(PHP_VERSION, '5.4.0', '<')) { throw new Exception; } // Load bundled Predis library if (! class_exists('Predis\Client')) { $plugin_dir = defined('WP_PLUGIN_DIR') ? WP_PLUGIN_DIR : WP_CONTENT_DIR . '/plugins'; require_once $plugin_dir . '/redis-cache/includes/predis/autoload.php'; } $options = array(); if (defined('WP_REDIS_SHARDS')) { $parameters = WP_REDIS_SHARDS; } elseif (defined('WP_REDIS_SENTINEL')) { $parameters = WP_REDIS_SERVERS; $options[ 'replication' ] = 'sentinel'; $options[ 'service' ] = WP_REDIS_SENTINEL; } elseif (defined('WP_REDIS_SERVERS')) { $parameters = WP_REDIS_SERVERS; $options[ 'replication' ] = true; } elseif (defined('WP_REDIS_CLUSTER')) { $parameters = WP_REDIS_CLUSTER; $options[ 'cluster' ] = 'redis'; } foreach (array( 'WP_REDIS_SERVERS', 'WP_REDIS_SHARDS', 'WP_REDIS_CLUSTER' ) as $constant) { if (defined('WP_REDIS_PASSWORD') && defined($constant)) { $options[ 'parameters' ][ 'password' ] = WP_REDIS_PASSWORD; } } $this->redis = new Predis\Client($parameters, $options); $this->redis->connect(); $this->redis_client .= sprintf(' (v%s)', Predis\Client::VERSION); } // Throws exception if Redis is unavailable $this->redis->ping(); $this->redis_connected = true; } catch (Exception $exception) { // When Redis is unavailable, fall back to the internal back by forcing all groups to be "no redis" groups $this->ignored_groups = array_unique(array_merge($this->ignored_groups, $this->global_groups)); $this->redis_connected = false; if (! $fail_gracefully) { throw $exception; } } // Assign global and blog prefixes for use with keys if (function_exists('is_multisite')) { $this->global_prefix = (is_multisite() || defined('CUSTOM_USER_TABLE') && defined('CUSTOM_USER_META_TABLE')) ? '' : $table_prefix; $this->blog_prefix = (is_multisite() ? $blog_id : $table_prefix); } } /** * Is Redis available? * * @return bool */ public function redis_status() { return $this->redis_connected; } /** * Returns the Redis instance. * * @return mixed */ public function redis_instance() { return $this->redis; } /** * Adds a value to cache. * * If the specified key already exists, the value is not stored and the function * returns false. * * @param string $key The key under which to store the value. * @param mixed $value The value to store. * @param string $group The group value appended to the $key. * @param int $expiration The expiration time, defaults to 0. * @return bool Returns TRUE on success or FALSE on failure. */ public function add($key, $value, $group = 'default', $expiration = 0) { return $this->add_or_replace(true, $key, $value, $group, $expiration); } /** * Replace a value in the cache. * * If the specified key doesn't exist, the value is not stored and the function * returns false. * * @param string $key The key under which to store the value. * @param mixed $value The value to store. * @param string $group The group value appended to the $key. * @param int $expiration The expiration time, defaults to 0. * @return bool Returns TRUE on success or FALSE on failure. */ public function replace($key, $value, $group = 'default', $expiration = 0) { return $this->add_or_replace(false, $key, $value, $group, $expiration); } /** * Add or replace a value in the cache. * * Add does not set the value if the key exists; replace does not replace if the value doesn't exist. * * @param bool $add True if should only add if value doesn't exist, false to only add when value already exists * @param string $key The key under which to store the value. * @param mixed $value The value to store. * @param string $group The group value appended to the $key. * @param int $expiration The expiration time, defaults to 0. * @return bool Returns TRUE on success or FALSE on failure. */ protected function add_or_replace($add, $key, $value, $group = 'default', $expiration = 0) { $cache_addition_suspended = function_exists('wp_suspend_cache_addition') ? wp_suspend_cache_addition() : false; if ( $add && $cache_addition_suspended ) { return false; } $result = true; $derived_key = $this->build_key($key, $group); // save if group not excluded and redis is up if (! in_array($group, $this->ignored_groups) && $this->redis_status()) { $exists = $this->redis->exists($derived_key); if ($add == $exists) { return false; } $expiration = $this->validate_expiration($expiration); if ($expiration) { $result = $this->parse_redis_response($this->redis->setex($derived_key, $expiration, $this->maybe_serialize($value))); } else { $result = $this->parse_redis_response($this->redis->set($derived_key, $this->maybe_serialize($value))); } } $exists = isset($this->cache[ $derived_key ]); if ($add == $exists) { return false; } if ($result) { $this->add_to_internal_cache($derived_key, $value); } return $result; } /** * Remove the item from the cache. * * @param string $key The key under which to store the value. * @param string $group The group value appended to the $key. * @return bool Returns TRUE on success or FALSE on failure. */ public function delete($key, $group = 'default') { $result = false; $derived_key = $this->build_key($key, $group); if (isset($this->cache[ $derived_key ])) { unset($this->cache[ $derived_key ]); $result = true; } if ($this->redis_status() && ! in_array($group, $this->ignored_groups)) { $result = $this->parse_redis_response($this->redis->del($derived_key)); } if (function_exists('do_action')) { do_action('redis_object_cache_delete', $key, $group); } return $result; } /** * Invalidate all items in the cache. If `WP_REDIS_SELECTIVE_FLUSH` is `true`, * only keys prefixed with the `WP_CACHE_KEY_SALT` are flushed. * * @param int $delay Number of seconds to wait before invalidating the items. * @return bool Returns TRUE on success or FALSE on failure. */ public function flush($delay = 0) { $delay = abs(intval($delay)); if ($delay) { sleep($delay); } $result = false; $this->cache = array(); if ($this->redis_status()) { $salt = defined('WP_CACHE_KEY_SALT') ? trim(WP_CACHE_KEY_SALT) : null; $selective = defined('WP_REDIS_SELECTIVE_FLUSH') ? WP_REDIS_SELECTIVE_FLUSH : null; if ($salt && $selective) { $script = " local i = 0 for _,k in ipairs(redis.call('keys', '{$salt}*')) do redis.call('del', k) i = i + 1 end return i "; $result = $this->parse_redis_response($this->redis->eval( $script, $this->redis instanceof Predis\Client ? 0 : [] )); } else { $result = $this->parse_redis_response($this->redis->flushdb()); } if (function_exists('do_action')) { do_action('redis_object_cache_flush', $result, $delay, $selective, $salt); } } return $result; } /** * Retrieve object from cache. * * Gets an object from cache based on $key and $group. * * @param string $key The key under which to store the value. * @param string $group The group value appended to the $key. * @param string $force Optional. Whether to force a refetch rather than relying on the local * cache. Default 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 Cached object value. */ public function get($key, $group = 'default', $force = false, &$found = null) { $derived_key = $this->build_key($key, $group); if (isset($this->cache[ $derived_key ]) && ! $force) { $found = true; $this->cache_hits++; return is_object($this->cache[ $derived_key ]) ? clone $this->cache[ $derived_key ] : $this->cache[ $derived_key ]; } elseif (in_array($group, $this->ignored_groups) || ! $this->redis_status()) { $found = false; $this->cache_misses++; return false; } $result = $this->redis->get($derived_key); if ($result === null || $result === false) { $found = false; $this->cache_misses++; return false; } else { $found = true; $this->cache_hits++; $value = $this->maybe_unserialize($result); } $this->add_to_internal_cache($derived_key, $value); $value = is_object($value) ? clone $value : $value; if (function_exists('do_action')) { do_action('redis_object_cache_get', $key, $value, $group, $force, $found); } if (function_exists('apply_filters') && function_exists('has_filter')) { if (has_filter('redis_object_cache_get')) { return apply_filters('redis_object_cache_get', $value, $key, $group, $force, $found); } } return $value; } /** * Retrieve multiple values from cache. * * Gets multiple values from cache, including across multiple groups * * Usage: array( 'group0' => array( 'key0', 'key1', 'key2', ), 'group1' => array( 'key0' ) ) * * Mirrors the Memcached Object Cache plugin's argument and return-value formats * * @param array $groups Array of groups and keys to retrieve * @return bool|mixed Array of cached values, keys in the format $group:$key. Non-existent keys null. */ public function get_multi($groups) { if (empty($groups) || ! is_array($groups)) { return false; } // Retrieve requested caches and reformat results to mimic Memcached Object Cache's output $cache = array(); foreach ($groups as $group => $keys) { if (in_array($group, $this->ignored_groups) || ! $this->redis_status()) { foreach ($keys as $key) { $cache[ $this->build_key($key, $group) ] = $this->get($key, $group); } } else { // Reformat arguments as expected by Redis $derived_keys = array(); foreach ($keys as $key) { $derived_keys[] = $this->build_key($key, $group); } // Retrieve from cache in a single request $group_cache = $this->redis->mget($derived_keys); // Build an array of values looked up, keyed by the derived cache key $group_cache = array_combine($derived_keys, $group_cache); // Restores cached data to its original data type $group_cache = array_map(array( $this, 'maybe_unserialize' ), $group_cache); // Redis returns null for values not found in cache, but expected return value is false in this instance $group_cache = array_map(array( $this, 'filter_redis_get_multi' ), $group_cache); $cache = array_merge($cache, $group_cache); } } // Add to the internal cache the found values from Redis foreach ($cache as $key => $value) { if ($value) { $this->cache_hits++; $this->add_to_internal_cache($key, $value); } else { $this->cache_misses++; } } return $cache; } /** * Sets a value in cache. * * The value is set whether or not this key already exists in Redis. * * @param string $key The key under which to store the value. * @param mixed $value The value to store. * @param string $group The group value appended to the $key. * @param int $expiration The expiration time, defaults to 0. * @return bool Returns TRUE on success or FALSE on failure. */ public function set($key, $value, $group = 'default', $expiration = 0) { $result = true; $derived_key = $this->build_key($key, $group); // save if group not excluded from redis and redis is up if (! in_array($group, $this->ignored_groups) && $this->redis_status()) { $expiration = $this->validate_expiration($expiration); if ($expiration) { $result = $this->parse_redis_response($this->redis->setex($derived_key, $expiration, $this->maybe_serialize($value))); } else { $result = $this->parse_redis_response($this->redis->set($derived_key, $this->maybe_serialize($value))); } } // if the set was successful, or we didn't go to redis if ($result) { $this->add_to_internal_cache($derived_key, $value); } if (function_exists('do_action')) { do_action('redis_object_cache_set', $key, $value, $group, $expiration); } return $result; } /** * Increment a Redis counter by the amount specified * * @param string $key * @param int $offset * @param string $group * @return int|bool */ public function increment($key, $offset = 1, $group = 'default') { $derived_key = $this->build_key($key, $group); $offset = (int) $offset; // If group is a non-Redis group, save to internal cache, not Redis if (in_array($group, $this->ignored_groups) || ! $this->redis_status()) { $value = $this->get_from_internal_cache($derived_key, $group); $value += $offset; $this->add_to_internal_cache($derived_key, $value); return $value; } // Save to Redis $result = $this->parse_redis_response($this->redis->incrBy($derived_key, $offset)); $this->add_to_internal_cache($derived_key, (int) $this->redis->get($derived_key)); return $result; } /** * Alias of `increment()`. * * @param string $key * @param int $offset * @param string $group * @return bool */ public function incr($key, $offset = 1, $group = 'default') { return $this->increment($key, $offset, $group); } /** * Decrement a Redis counter by the amount specified * * @param string $key * @param int $offset * @param string $group * @return int|bool */ public function decrement($key, $offset = 1, $group = 'default') { $derived_key = $this->build_key($key, $group); $offset = (int) $offset; // If group is a non-Redis group, save to internal cache, not Redis if (in_array($group, $this->ignored_groups) || ! $this->redis_status()) { $value = $this->get_from_internal_cache($derived_key, $group); $value -= $offset; $this->add_to_internal_cache($derived_key, $value); return $value; } // Save to Redis $result = $this->parse_redis_response($this->redis->decrBy($derived_key, $offset)); $this->add_to_internal_cache($derived_key, (int) $this->redis->get($derived_key)); return $result; } /** * Render data about current cache requests * * @return string */ public function stats() { ?>
Redis Status: redis_status() ? 'Connected' : 'Not Connected'; ?>
Redis Client: redis_client; ?>
Cache Hits: cache_hits; ?>
Cache Misses: cache_misses; ?>