add($key, $data, $group, $expire); } function wp_cache_incr($key, $n = 1, $group = '') { global $wp_object_cache; return $wp_object_cache->incr($key, $n, $group); } function wp_cache_decr($key, $n = 1, $group = '') { global $wp_object_cache; return $wp_object_cache->decr($key, $n, $group); } function wp_cache_close() { global $wp_object_cache; return $wp_object_cache->close(); } function wp_cache_delete($key, $group = '') { global $wp_object_cache; return $wp_object_cache->delete($key, $group); } function wp_cache_flush() { global $wp_object_cache; return $wp_object_cache->flush(); } function wp_cache_get($key, $group = '', $force = false) { global $wp_object_cache; return $wp_object_cache->get($key, $group, $force); } function wp_cache_get_multi($groups, $force = false) { global $wp_object_cache; return $wp_object_cache->get_multi($groups, $force); } function wp_cache_init() { global $wp_object_cache; $wp_object_cache = new WP_Object_Cache(); } function wp_cache_replace($key, $data, $group = '', $expire = 0) { global $wp_object_cache; return $wp_object_cache->replace($key, $data, $group, $expire); } function wp_cache_set($key, $data, $group = '', $expire = 0) { global $wp_object_cache; if ( defined( 'WP_INSTALLING' ) == false ) { return $wp_object_cache->set( $key, $data, $group, $expire ); } else { return $wp_object_cache->delete( $key, $group ); } } function wp_cache_switch_to_blog( $blog_id ) { global $wp_object_cache; return $wp_object_cache->switch_to_blog( $blog_id ); } function wp_cache_add_global_groups( $groups ) { global $wp_object_cache; $wp_object_cache->add_global_groups($groups); } function wp_cache_add_non_persistent_groups( $groups ) { global $wp_object_cache; $wp_object_cache->add_non_persistent_groups($groups); } class WP_Object_Cache { var $global_groups = array(); var $no_mc_groups = array(); var $cache = array(); var $mc = array(); var $stats = array(); var $group_ops = array(); var $cache_enabled = true; var $default_expiration = 0; public function get_alloptions() { // Check our internal cache, to avoid the more expensive get-multi $key = $this->key( 'alloptions', 'options' ); if ( isset( $this->cache[ $key ] ) ) { return $this->cache[ $key ]; } $keys = $this->get( 'alloptionskeys', 'options' ); if ( empty( $keys ) || ! is_array( $keys ) ) { return array(); } $data = $this->get_multi( array( 'options' => array_keys( $keys ) ) ); if ( empty( $data ) || empty( $data['options'] ) ) { return array(); } $this->cache[ $key ] = $data['options']; return $this->cache[ $key ]; } public function set_alloptions( $data ) { $internal_cache_key = $this->key( 'alloptions', 'options' ); $existing = $internal_cache = $this->get_alloptions(); $keys = $this->get( 'alloptionskeys', 'options' ); if ( empty( $keys ) || ! is_array( $keys ) ) { $keys = array(); } // While you could use array_diff here, it ends up being a bit more // complicated than just checking foreach ( $data as $key => $value ) { if ( isset( $existing[ $key ] ) && $existing[ $key ] === $value ) { continue; } if ( ! isset( $keys[ $key ] ) ) { $keys[ $key ] = true; } if ( ! $this->set( $key, $value, 'options' ) ) { return false; } $internal_cache[ $key ] = $value; } // Remove deleted elements foreach ( $existing as $key => $value ) { if ( isset( $data[ $key ] ) ) { continue; } if ( isset( $keys[ $key ] ) ) { unset( $keys[ $key ] ); } if ( ! $this->delete( $key, 'options' ) ) { return false; } unset( $internal_cache[ $key ] ); } if ( ! $this->set( 'alloptionskeys', $keys, 'options' ) ) { return false; } $this->cache[ $internal_cache_key ] = $internal_cache; return true; } public function delete_alloptions() { $key = $this->key( 'alloptions', 'options' ); $this->cache[ $key ] = array(); return $this->delete( 'alloptionskeys', 'options' ); } function add($id, $data, $group = 'default', $expire = 0) { if ( $id === 'alloptions' && $group === 'options' ) { return $this->set_alloptions( $data ); } $key = $this->key($id, $group); if ( is_object( $data ) ) { $data = clone $data; } if ( in_array($group, $this->no_mc_groups) ) { $this->cache[$key] = $data; return true; } elseif ( isset($this->cache[$key]) && $this->cache[$key] !== false ) { return false; } $mc =& $this->get_mc($group); $expire = ($expire == 0) ? $this->default_expiration : $expire; $time = microtime(true); $result = $mc->add($key, $data, false, $expire); $time_taken = microtime(true) - $time; if ( false !== $result ) { @ ++$this->stats['add']; $this->stats['add_time'] += $time_taken; $this->group_ops[$group][] = "add $id"; $this->cache[$key] = $data; } return $result; } function add_global_groups($groups) { if ( ! is_array( $groups ) ) { $groups = (array) $groups; } $this->global_groups = array_merge($this->global_groups, $groups); $this->global_groups = array_unique($this->global_groups); } function add_non_persistent_groups($groups) { if ( ! is_array( $groups ) ) { $groups = (array) $groups; } $this->no_mc_groups = array_merge($this->no_mc_groups, $groups); $this->no_mc_groups = array_unique($this->no_mc_groups); } function incr($id, $n = 1, $group = 'default' ) { $key = $this->key($id, $group); $mc =& $this->get_mc($group); $this->cache[ $key ] = $mc->increment( $key, $n ); return $this->cache[ $key ]; } function decr($id, $n = 1, $group = 'default' ) { $key = $this->key($id, $group); $mc =& $this->get_mc($group); $this->cache[ $key ] = $mc->decrement( $key, $n ); return $this->cache[ $key ]; } function close() { foreach ( $this->mc as $bucket => $mc ) { $mc->close(); } } function delete($id, $group = 'default') { if ( $id === 'alloptions' && $group === 'options' ) { return $this->delete_alloptions(); } $key = $this->key($id, $group); if ( in_array($group, $this->no_mc_groups) ) { unset($this->cache[$key]); return true; } $mc =& $this->get_mc($group); $time = microtime(true); $result = $mc->delete($key); $time_taken = microtime(true) - $time; @ ++$this->stats['delete']; $this->stats['delete_time'] += $time_taken; $this->group_ops[$group][] = "delete $id"; if ( false !== $result ) { unset( $this->cache[ $key ] ); } return $result; } function flush() { // Return true is flushing is disabled if ( ! WP_MEMCACHE_DISABLE_FLUSHING ) { return true; } // Did someone try and wipe our stats? >:( // This occurs during unit tests, where WP reaches in and resets the // stats array. if ( empty( $this->stats ) ) { $this->reset_stats(); } $ret = true; foreach ( array_keys( $this->mc ) as $group ) { $ret &= $this->mc[ $group ]->flush(); } return $ret; } /** * Flush the local (in-memory) object cache * * Forces all future requests to fetch from memcache. Can be used to * alleviate memory pressure in long-running requests. */ public function flush_local() { $this->cache = array(); $this->group_ops = array(); } protected function reset_stats() { $this->stats = array( 'get' => 0, 'get_time' => 0, 'add' => 0, 'add_time' => 0, 'delete' => 0, 'delete_time' => 0, 'set' => 0, 'set_time' => 0 ); } function get($id, $group = 'default', $force = false) { if ( $id === 'alloptions' && $group === 'options' ) { return $this->get_alloptions(); } $key = $this->key($id, $group); $mc =& $this->get_mc($group); if ( isset($this->cache[$key]) && ( !$force || in_array($group, $this->no_mc_groups) ) ) { if ( is_object( $this->cache[ $key ] ) ) { $value = clone $this->cache[ $key ]; } else { $value = $this->cache[ $key ]; } } else if ( in_array($group, $this->no_mc_groups) ) { $this->cache[$key] = $value = false; } else { $time = microtime(true); $value = $mc->get($key); if ( NULL === $value ) $value = false; $time_taken = microtime(true) - $time; $this->cache[$key] = $value; @ ++$this->stats['get']; $this->stats['get_time'] += $time_taken; $this->group_ops[$group][] = "get $id"; } if ( 'checkthedatabaseplease' === $value ) { unset( $this->cache[$key] ); $value = false; } return $value; } function get_multi( $groups ) { /* format: $get['group-name'] = array( 'key1', 'key2' ); */ $return = array(); $to_get = array(); foreach ( $groups as $group => $ids ) { $mc =& $this->get_mc( $group ); $return[ $group ] = array(); foreach ( $ids as $id ) { $key = $this->key( $id, $group ); if ( isset( $this->cache[ $key ] ) ) { if ( is_object( $this->cache[ $key ] ) ) { $return[ $group ][ $id ] = clone $this->cache[ $key ]; } else { $return[ $group ][ $id ] = $this->cache[ $key ]; } continue; } else if ( in_array( $group, $this->no_mc_groups ) ) { $return[ $group ][ $id ] = false; continue; } else { $to_get[ $key ] = array( $group, $id ); } } } if ( $to_get ) { $vals = $mc->get( array_keys( $to_get ) ); foreach ( $to_get as $key => $bits ) { if ( ! isset( $vals[ $key ] ) ) { continue; } list( $group, $id ) = $bits; $return[ $group ][ $id ] = $vals[ $key ]; $this->cache[ $key ] = $vals[ $key ]; } } @ ++$this->stats['get_multi']; $this->group_ops[$group][] = "get_multi $id"; return $return; } function key($key, $group) { if ( empty( $group ) ) { $group = 'default'; } if ( false !== array_search( $group, $this->global_groups ) ) { $prefix = $this->global_prefix; } else { $prefix = $this->blog_prefix; } return preg_replace('/\s+/', '', WP_CACHE_KEY_SALT . ":$prefix$group:$key"); } function replace( $id, $data, $group = 'default', $expire = 0 ) { $key = $this->key( $id, $group ); $expire = ( $expire == 0 ) ? $this->default_expiration : $expire; $mc =& $this->get_mc( $group ); if ( is_object( $data ) ) { $data = clone $data; } $result = $mc->replace( $key, $data, false, $expire ); if ( false !== $result ) { $this->cache[ $key ] = $data; } return $result; } function set($id, $data, $group = 'default', $expire = 0) { if ( $id === 'alloptions' && $group === 'options' ) { return $this->set_alloptions( $data ); } $key = $this->key($id, $group); if ( isset( $this->cache[ $key ] ) && ( 'checkthedatabaseplease' === $this->cache[ $key ] ) ) { return false; } if ( is_object( $data ) ) { $data = clone $data; } $this->cache[ $key ] = $data; if ( in_array( $group, $this->no_mc_groups ) ) { return true; } $expire = ($expire == 0) ? $this->default_expiration : $expire; $mc =& $this->get_mc($group); $time = microtime(true); /** * If the expiry exceeds 30 days, we have to sent the expire * as a unix timestamp. This is because PHP Memcache extension * uses this hueristic. * * To make this consistant, we always use absolute unix timestamps. */ if ( $expire ) { $expire += time(); } $result = $mc->set($key, $data, false, $expire); $time_taken = microtime(true) - $time; @ ++$this->stats['set']; $this->stats['set_time'] += $time_taken; $this->group_ops[$group][] = "set $id"; return $result; } function switch_to_blog( $blog_id ) { global $wpdb; $table_prefix = $wpdb->prefix; $blog_id = (int) $blog_id; $this->blog_prefix = ( is_multisite() ? $blog_id : $table_prefix ) . ':'; } function colorize_debug_line($line) { $colors = array( 'get' => 'green', 'set' => 'purple', 'add' => 'blue', 'delete' => 'red'); $cmd = substr($line, 0, strpos($line, ' ')); $cmd2 = "$cmd"; return $cmd2 . substr($line, strlen($cmd)) . "\n"; } function stats() { echo "
\n";
foreach ( $this->stats as $stat => $n ) {
if ( ! $n ) {
continue;
}
echo "$stat $n";
echo "
\n";
}
echo "
\n";
$lines = array();
foreach ( $ops as $op ) {
$lines[] = $this->colorize_debug_line($op);
}
print_r($lines);
echo "\n";
}
}
function &get_mc($group) {
if ( isset( $this->mc[ $group ] ) ) {
return $this->mc[ $group ];
}
return $this->mc['default'];
}
function failure_callback( $host, $port ) {
if ( !WP_MEMCACHE_DISABLE_LOGGING ) {
error_log( "Memcache Connection failure for $host:$port\n" );
}
}
function __construct() {
global $memcached_servers, $blog_id, $table_prefix;
if ( isset( $memcached_servers ) ) {
$buckets = $memcached_servers;
} else {
$buckets = array( '127.0.0.1:11211' );
}
reset( $buckets );
if ( is_int( key( $buckets ) ) ) {
$buckets = array( 'default' => $buckets );
}
foreach ( $buckets as $bucket => $servers) {
$this->mc[$bucket] = new Memcache();
foreach ( $servers as $server ) {
list ( $node, $port ) = explode(':', $server);
if ( ! $port ) {
$port = ini_get( 'memcache.default_port' );
}
$port = intval( $port );
if ( ! $port ) {
$port = 11211;
}
$this->mc[$bucket]->addServer($node, $port, WP_MEMCACHE_PERSISTENT, WP_MEMCACHE_WEIGHT, WP_MEMCACHE_TIMEOUT, WP_MEMCACHE_RETRY, true, array($this, 'failure_callback'));
$this->mc[$bucket]->setCompressThreshold(20000, 0.2);
}
}
$this->global_prefix = '';
$this->blog_prefix = '';
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 ) . ':';
}
$this->reset_stats();
$this->cache_hits =& $this->stats['get'];
$this->cache_misses =& $this->stats['add'];
}
}