-
Notifications
You must be signed in to change notification settings - Fork 89
/
Copy pathCache_Xcache.php
359 lines (320 loc) · 11.5 KB
/
Cache_Xcache.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
<?php
/**
* File: Cache_Xcache.php
*
* @package W3TC
*/
namespace W3TC;
/**
* Class Cache_Xcache
*
* phpcs:disable PSR2.Classes.PropertyDeclaration.Underscore
* phpcs:disable PSR2.Methods.MethodDeclaration.Underscore
* phpcs:disable WordPress.PHP.DiscouragedPHPFunctions.serialize_serialize
* phpcs:disable WordPress.PHP.DiscouragedPHPFunctions.serialize_unserialize
* phpcs:disable WordPress.PHP.NoSilencedErrors.Discouraged
*/
class Cache_Xcache extends Cache_Base {
/**
* Used for faster flushing
*
* @var integer $_key_version
*/
private $_key_version = array();
/**
* Adds a new item to the cache if it does not already exist.
*
* If the item does not exist in the cache, it is added with the specified expiration and group. If it already exists,
* the method returns false.
*
* @param string $key The unique key to identify the cached item.
* @param mixed $var The value to store in the cache (passed by reference).
* @param int $expire The expiration time in seconds. Defaults to 0 (no expiration).
* @param string $group The group to which the key belongs. Defaults to an empty string.
*
* @return bool True if the item was added successfully, false if the item already exists.
*/
public function add( $key, &$var, $expire = 0, $group = '' ) {
if ( false === $this->get( $key, $group ) ) {
return $this->set( $key, $var, $expire, $group );
}
return false;
}
/**
* Sets a value in the cache, overwriting any existing value.
*
* This method sets a value in the cache for a given key, with an optional expiration time and group. If the value
* does not have a `key_version`, it is assigned the current group key version.
*
* @param string $key The unique key to identify the cached item.
* @param mixed $var The value to store in the cache.
* @param int $expire The expiration time in seconds. Defaults to 0 (no expiration).
* @param string $group The group to which the key belongs. Defaults to an empty string.
*
* @return bool True if the value was successfully stored, false otherwise.
*/
public function set( $key, $var, $expire = 0, $group = '' ) {
if ( ! isset( $var['key_version'] ) ) {
$var['key_version'] = $this->_get_key_version( $group );
}
$storage_key = $this->get_item_key( $key );
return xcache_set( $storage_key, serialize( $var ), $expire );
}
/**
* Retrieves a cached item along with its version status.
*
* This method retrieves the cached value for a given key, checking if the key version matches the current group key
* version. It also determines whether the cached value is expired and returns a flag indicating if old data is used.
*
* @param string $key The unique key to identify the cached item.
* @param string $group The group to which the key belongs. Defaults to an empty string.
*
* @return array An array containing the cached value (or null if not found) and a boolean indicating old data usage.
*/
public function get_with_old( $key, $group = '' ) {
$has_old_data = false;
$storage_key = $this->get_item_key( $key );
$v = @unserialize( xcache_get( $storage_key ) );
if ( ! is_array( $v ) || ! isset( $v['key_version'] ) ) {
return array( null, $has_old_data );
}
$key_version = $this->_get_key_version( $group );
if ( $v['key_version'] === $key_version ) {
return array( $v, $has_old_data );
}
if ( $v['key_version'] > $key_version ) {
if ( ! empty( $v['key_version_at_creation'] ) && $v['key_version_at_creation'] !== $key_version ) {
$this->_set_key_version( $v['key_version'], $group );
}
return array( $v, $has_old_data );
}
// key version is old.
if ( ! $this->_use_expired_data ) {
return array( null, $has_old_data );
}
// if we have expired data - update it for future use and let current process recalculate it.
$expires_at = isset( $v['expires_at'] ) ? $v['expires_at'] : null;
if ( null === $expires_at || time() > $expires_at ) {
$v['expires_at'] = time() + 30;
xcache_set( $storage_key, serialize( $v ), 0 );
$has_old_data = true;
return array( null, $has_old_data );
}
// return old version.
return array( $v, $has_old_data );
}
/**
* Replaces an existing cached item with a new value.
*
* This method updates the value for a given key only if the key already exists in the cache.
*
* @param string $key The unique key to identify the cached item.
* @param mixed $var The new value to store in the cache (passed by reference).
* @param int $expire The expiration time in seconds. Defaults to 0 (no expiration).
* @param string $group The group to which the key belongs. Defaults to an empty string.
*
* @return bool True if the value was replaced successfully, false if the key does not exist.
*/
public function replace( $key, &$var, $expire = 0, $group = '' ) {
if ( $this->get( $key, $group ) !== false ) {
return $this->set( $key, $var, $expire, $group );
}
return false;
}
/**
* Deletes a cached item, optionally keeping expired data.
*
* If expired data usage is enabled, the key version is set to 0 instead of completely removing the item.
* Otherwise, the item is fully deleted from the cache.
*
* @param string $key The unique key to identify the cached item.
* @param string $group The group to which the key belongs. Defaults to an empty string.
*
* @return bool True if the item was deleted successfully, false otherwise.
*/
public function delete( $key, $group = '' ) {
$storage_key = $this->get_item_key( $key );
if ( $this->_use_expired_data ) {
$v = @unserialize( xcache_get( $storage_key ) );
if ( is_array( $v ) ) {
$v['key_version'] = 0;
xcache_set( $storage_key, serialize( $v ), 0 );
return true;
}
}
return xcache_unset( $storage_key );
}
/**
* Completely removes a cached item from the cache.
*
* This method fully deletes the cached item, bypassing any logic for handling expired data.
*
* @param string $key The unique key to identify the cached item.
* @param string $group The group to which the key belongs. Defaults to an empty string.
*
* @return bool True if the item was removed successfully, false otherwise.
*/
public function hard_delete( $key, $group = '' ) {
$storage_key = $this->get_item_key( $key );
return xcache_unset( $storage_key );
}
/**
* Flushes the cache for a specific group or all groups.
*
* This increments the key version for the specified group, effectively invalidating all cache entries
* associated with the current key version.
*
* @param string $group (Optional) The cache group to flush. Default is an empty string, which applies to all groups.
*
* @return bool True on success.
*/
public function flush( $group = '' ) {
$this->_get_key_version( $group ); // initialize $this->_key_version.
$this->_key_version[ $group ]++;
$this->_set_key_version( $this->_key_version[ $group ], $group );
return true;
}
/**
* Gets the key version extension for ahead-of-time cache generation.
*
* This provides the current key version and the next version to be used for generating ahead-of-time cache.
*
* @param string $group The cache group to retrieve the extension for.
*
* @return array An associative array with:
* - 'key_version' (int): The next key version.
* - 'key_version_at_creation' (int): The current key version.
*/
public function get_ahead_generation_extension( $group ) {
$v = $this->_get_key_version( $group );
return array(
'key_version' => $v + 1,
'key_version_at_creation' => $v,
);
}
/**
* Updates the key version for a cache group after ahead-of-time generation.
*
* If the provided key version is higher than the current version, the key version is updated.
*
* @param string $group The cache group to update.
* @param array $extension {
* The extension data containing 'key_version'.
*
* @type string $key_version The version of the cache key.
* }
*
* @return void
*/
public function flush_group_after_ahead_generation( $group, $extension ) {
$v = $this->_get_key_version( $group );
if ( $extension['key_version'] > $v ) {
$this->_set_key_version( $extension['key_version'], $group );
}
}
/**
* Checks if Wincache is available on the server.
*
* @return bool True if Wincache functions are available, false otherwise.
*/
public function available() {
return function_exists( 'xcache_set' );
}
/**
* Retrieves the current key version for a cache group.
*
* If no version exists, initializes the version to 1.
*
* @param string $group (Optional) The cache group to retrieve the key version for. Default is an empty string.
*
* @return int The current key version for the specified group.
*/
private function _get_key_version( $group = '' ) {
if ( ! isset( $this->_key_version[ $group ] ) || $this->_key_version[ $group ] <= 0 ) {
$v = xcache_get( $this->_get_key_version_key( $group ) );
$v = intval( $v );
$this->_key_version[ $group ] = ( $v > 0 ? $v : 1 );
}
return $this->_key_version[ $group ];
}
/**
* Sets the key version for a cache group.
*
* @param int $v The key version to set.
* @param string $group (Optional) The cache group to set the key version for. Default is an empty string.
*
* @return void
*/
private function _set_key_version( $v, $group = '' ) {
xcache_set( $this->_get_key_version_key( $group ), $v, 0 );
}
/**
* Sets a value conditionally if the old value matches the expected value.
*
* This method attempts to simulate an atomic check-and-set operation. If the current value does not
* match the old value, the operation fails.
*
* @param string $key The cache key to update.
* @param array $old_value {
* The expected current value.
*
* @type string $content The expected content to compare.
* }
* @param array $new_value The new value to set.
*
* @return bool True if the operation succeeds, false otherwise.
*/
public function set_if_maybe_equals( $key, $old_value, $new_value ) {
// cant guarantee atomic action here, filelocks fail often.
$value = $this->get( $key );
if ( isset( $old_value['content'] ) && $value['content'] !== $old_value['content'] ) {
return false;
}
return $this->set( $key, $new_value );
}
/**
* Increments a counter by the specified value.
*
* If the counter does not exist, initializes it with a value of 0 before incrementing.
*
* @param string $key The key of the counter to increment.
* @param int $value The value to increment the counter by.
*
* @return int|bool The new counter value on success, or false on failure.
*/
public function counter_add( $key, $value ) {
if ( 0 === $value ) {
return true;
}
$storage_key = $this->get_item_key( $key );
$r = xcache_inc( $storage_key, $value );
if ( ! $r ) { // it doesnt initialize counter by itself.
$this->counter_set( $key, 0 );
}
return $r;
}
/**
* Sets the value of a counter.
*
* @param string $key The key of the counter to set.
* @param int $value The value to set the counter to.
*
* @return bool True on success, false on failure.
*/
public function counter_set( $key, $value ) {
$storage_key = $this->get_item_key( $key );
return xcache_set( $storage_key, $value );
}
/**
* Retrieves the value of a counter.
*
* @param string $key The key of the counter to retrieve.
*
* @return int The current counter value, or 0 if the counter does not exist.
*/
public function counter_get( $key ) {
$storage_key = $this->get_item_key( $key );
$v = (int) xcache_get( $storage_key );
return $v;
}
}