1<?php
2// This file is part of Moodle - http://moodle.org/
3//
4// Moodle is free software: you can redistribute it and/or modify
5// it under the terms of the GNU General Public License as published by
6// the Free Software Foundation, either version 3 of the License, or
7// (at your option) any later version.
8//
9// Moodle is distributed in the hope that it will be useful,
10// but WITHOUT ANY WARRANTY; without even the implied warranty of
11// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12// GNU General Public License for more details.
13//
14// You should have received a copy of the GNU General Public License
15// along with Moodle.  If not, see <http://www.gnu.org/licenses/>.
16
17/**
18 * Cache loaders
19 *
20 * This file is part of Moodle's cache API, affectionately called MUC.
21 * It contains the components that are required in order to use caching.
22 *
23 * @package    core
24 * @category   cache
25 * @copyright  2012 Sam Hemelryk
26 * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
27 */
28
29defined('MOODLE_INTERNAL') || die();
30
31/**
32 * The main cache class.
33 *
34 * This class if the first class that any end developer will interact with.
35 * In order to create an instance of a cache that they can work with they must call one of the static make methods belonging
36 * to this class.
37 *
38 * @package    core
39 * @category   cache
40 * @copyright  2012 Sam Hemelryk
41 * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
42 */
43class cache implements cache_loader {
44
45    /**
46     * We need a timestamp to use within the cache API.
47     * This stamp needs to be used for all ttl and time based operations to ensure that we don't end up with
48     * timing issues.
49     * @var int
50     */
51    protected static $now;
52
53    /**
54     * A purge token used to distinguish between multiple cache purges in the same second.
55     * This is in the format <microtime>-<random string>.
56     *
57     * @var string
58     */
59    protected static $purgetoken;
60
61    /**
62     * The definition used when loading this cache if there was one.
63     * @var cache_definition
64     */
65    private $definition = false;
66
67    /**
68     * The cache store that this loader will make use of.
69     * @var cache_store
70     */
71    private $store;
72
73    /**
74     * The next cache loader in the chain if there is one.
75     * If a cache request misses for the store belonging to this loader then the loader
76     * stored here will be checked next.
77     * If there is a loader here then $datasource must be false.
78     * @var cache_loader|false
79     */
80    private $loader = false;
81
82    /**
83     * The data source to use if we need to load data (because if doesn't exist in the cache store).
84     * If there is a data source here then $loader above must be false.
85     * @var cache_data_source|false
86     */
87    private $datasource = false;
88
89    /**
90     * Used to quickly check if the store supports key awareness.
91     * This is set when the cache is initialised and is used to speed up processing.
92     * @var bool
93     */
94    private $supportskeyawareness = null;
95
96    /**
97     * Used to quickly check if the store supports ttl natively.
98     * This is set when the cache is initialised and is used to speed up processing.
99     * @var bool
100     */
101    private $supportsnativettl = null;
102
103    /**
104     * Gets set to true if the cache is going to be using a static array for acceleration.
105     * The array statically caches items used during the lifetime of the request. This greatly speeds up interaction
106     * with the cache in areas where it will be repetitively hit for the same information such as with strings.
107     * There are several other variables to control how this static acceleration array works.
108     * @var bool
109     */
110    private $staticacceleration = false;
111
112    /**
113     * The static acceleration array.
114     * Items will be stored in this cache as they were provided. This ensure there is no unnecessary processing taking place.
115     * @var array
116     */
117    private $staticaccelerationarray = array();
118
119    /**
120     * The number of items in the static acceleration array. Avoids count calls like you wouldn't believe.
121     * @var int
122     */
123    private $staticaccelerationcount = 0;
124
125    /**
126     * An array containing just the keys being used in the static acceleration array.
127     * This seems redundant perhaps but is used when managing the size of the static acceleration array.
128     * Items are added to the end of the array and the when we need to reduce the size of the cache we use the
129     * key that is first on this array.
130     * @var array
131     */
132    private $staticaccelerationkeys = array();
133
134    /**
135     * The maximum size of the static acceleration array.
136     *
137     * If set to false there is no max size.
138     * Caches that make use of static acceleration should seriously consider setting this to something reasonably small, but
139     * still large enough to offset repetitive calls.
140     *
141     * @var int|false
142     */
143    private $staticaccelerationsize = false;
144
145    /**
146     * Gets set to true during initialisation if the definition is making use of a ttl.
147     * Used to speed up processing.
148     * @var bool
149     */
150    private $hasattl = false;
151
152    /**
153     * Gets set to the class name of the store during initialisation. This is used several times in the cache class internally
154     * and having it here helps speed up processing.
155     * @var strubg
156     */
157    protected $storetype = 'unknown';
158
159    /**
160     * Gets set to true if we want to collect performance information about the cache API.
161     * @var bool
162     */
163    protected $perfdebug = false;
164
165    /**
166     * Determines if this loader is a sub loader, not the top of the chain.
167     * @var bool
168     */
169    protected $subloader = false;
170
171    /**
172     * Creates a new cache instance for a pre-defined definition.
173     *
174     * @param string $component The component for the definition
175     * @param string $area The area for the definition
176     * @param array $identifiers Any additional identifiers that should be provided to the definition.
177     * @param string $unused Used to be datasourceaggregate but that was removed and this is now unused.
178     * @return cache_application|cache_session|cache_store
179     */
180    public static function make($component, $area, array $identifiers = array(), $unused = null) {
181        $factory = cache_factory::instance();
182        return $factory->create_cache_from_definition($component, $area, $identifiers);
183    }
184
185    /**
186     * Creates a new cache instance based upon the given params.
187     *
188     * @param int $mode One of cache_store::MODE_*
189     * @param string $component The component this cache relates to.
190     * @param string $area The area this cache relates to.
191     * @param array $identifiers Any additional identifiers that should be provided to the definition.
192     * @param array $options An array of options, available options are:
193     *   - simplekeys : Set to true if the keys you will use are a-zA-Z0-9_
194     *   - simpledata : Set to true if the type of the data you are going to store is scalar, or an array of scalar vars
195     *   - staticacceleration : If set to true the cache will hold onto data passing through it.
196     *   - staticaccelerationsize : The max size for the static acceleration array.
197     * @return cache_application|cache_session|cache_store
198     */
199    public static function make_from_params($mode, $component, $area, array $identifiers = array(), array $options = array()) {
200        $factory = cache_factory::instance();
201        return $factory->create_cache_from_params($mode, $component, $area, $identifiers, $options);
202    }
203
204    /**
205     * Constructs a new cache instance.
206     *
207     * You should not call this method from your code, instead you should use the cache::make methods.
208     *
209     * This method is public so that the cache_factory is able to instantiate cache instances.
210     * Ideally we would make this method protected and expose its construction to the factory method internally somehow.
211     * The factory class is responsible for this in order to centralise the storage of instances once created. This way if needed
212     * we can force a reset of the cache API (used during unit testing).
213     *
214     * @param cache_definition $definition The definition for the cache instance.
215     * @param cache_store $store The store that cache should use.
216     * @param cache_loader|cache_data_source $loader The next loader in the chain or the data source if there is one and there
217     *      are no other cache_loaders in the chain.
218     */
219    public function __construct(cache_definition $definition, cache_store $store, $loader = null) {
220        global $CFG;
221        $this->definition = $definition;
222        $this->store = $store;
223        $this->storetype = get_class($store);
224        $this->perfdebug = (!empty($CFG->perfdebug) and $CFG->perfdebug > 7);
225        if ($loader instanceof cache_loader) {
226            $this->set_loader($loader);
227        } else if ($loader instanceof cache_data_source) {
228            $this->set_data_source($loader);
229        }
230        $this->definition->generate_definition_hash();
231        $this->staticacceleration = $this->definition->use_static_acceleration();
232        if ($this->staticacceleration) {
233            $this->staticaccelerationsize = $this->definition->get_static_acceleration_size();
234        }
235        $this->hasattl = ($this->definition->get_ttl() > 0);
236    }
237
238    /**
239     * Set the loader for this cache.
240     *
241     * @param   cache_loader $loader
242     */
243    protected function set_loader(cache_loader $loader): void {
244        $this->loader = $loader;
245
246        // Mark the loader as a sub (chained) loader.
247        $this->loader->set_is_sub_loader(true);
248    }
249
250    /**
251     * Set the data source for this cache.
252     *
253     * @param   cache_data_source $datasource
254     */
255    protected function set_data_source(cache_data_source $datasource): void {
256        $this->datasource = $datasource;
257    }
258
259    /**
260     * Used to inform the loader of its state as a sub loader, or as the top of the chain.
261     *
262     * This is important as it ensures that we do not have more than one loader keeping static acceleration data.
263     * Subloaders need to be "pure" loaders in the sense that they are used to store and retrieve information from stores or the
264     * next loader/data source in the chain.
265     * Nothing fancy, nothing flash.
266     *
267     * @param bool $setting
268     */
269    protected function set_is_sub_loader($setting = true) {
270        if ($setting) {
271            $this->subloader = true;
272            // Subloaders should not keep static acceleration data.
273            $this->staticacceleration = false;
274            $this->staticaccelerationsize = false;
275        } else {
276            $this->subloader = true;
277            $this->staticacceleration = $this->definition->use_static_acceleration();
278            if ($this->staticacceleration) {
279                $this->staticaccelerationsize = $this->definition->get_static_acceleration_size();
280            }
281        }
282    }
283
284    /**
285     * Alters the identifiers that have been provided to the definition.
286     *
287     * This is an advanced method and should not be used unless really needed.
288     * It allows the developer to slightly alter the definition without having to re-establish the cache.
289     * It will cause more processing as the definition will need to clear and reprepare some of its properties.
290     *
291     * @param array $identifiers
292     */
293    public function set_identifiers(array $identifiers) {
294        if ($this->definition->set_identifiers($identifiers)) {
295            // As static acceleration uses input keys and not parsed keys
296            // it much be cleared when the identifier set is changed.
297            $this->staticaccelerationarray = array();
298            if ($this->staticaccelerationsize !== false) {
299                $this->staticaccelerationkeys = array();
300                $this->staticaccelerationcount = 0;
301            }
302        }
303    }
304
305    /**
306     * Process any outstanding invalidation events for the cache we are registering,
307     *
308     * Identifiers and event invalidation are not compatible with each other at this time.
309     * As a result the cache does not need to consider identifiers when working out what to invalidate.
310     */
311    protected function handle_invalidation_events() {
312        if (!$this->definition->has_invalidation_events()) {
313            return;
314        }
315
316        // Each cache stores the current 'lastinvalidation' value within the cache itself.
317        $lastinvalidation = $this->get('lastinvalidation');
318        if ($lastinvalidation === false) {
319            // There is currently no  value for the lastinvalidation token, therefore the token is not set, and there
320            // can be nothing to invalidate.
321            // Set the lastinvalidation value to the current purge token and return early.
322            $this->set('lastinvalidation', self::get_purge_token());
323            return;
324        } else if ($lastinvalidation == self::get_purge_token()) {
325            // The current purge request has already been fully handled by this cache.
326            return;
327        }
328
329        /*
330         * Now that the whole cache check is complete, we check the meaning of any specific cache invalidation events.
331         * These are stored in the core/eventinvalidation cache as an multi-dimensinoal array in the form:
332         *  [
333         *      eventname => [
334         *          keyname => purgetoken,
335         *      ]
336         *  ]
337         *
338         * The 'keyname' value is used to delete a specific key in the cache.
339         * If the keyname is set to the special value 'purged', then the whole cache is purged instead.
340         *
341         * The 'purgetoken' is the token that this key was last purged.
342         * a) If the purgetoken matches the last invalidation, then the key/cache is not purged.
343         * b) If the purgetoken is newer than the last invalidation, then the key/cache is not purged.
344         * c) If the purge token is older than the last invalidation, or it has a different token component, then the
345         *    cache is purged.
346         *
347         * Option b should not happen under normal operation, but may happen in race condition whereby a long-running
348         * request's cache is cleared in another process during that request, and prior to that long-running request
349         * creating the cache. In such a condition, it would be incorrect to clear that cache.
350         */
351        $cache = self::make('core', 'eventinvalidation');
352        $events = $cache->get_many($this->definition->get_invalidation_events());
353        $todelete = array();
354        $purgeall = false;
355
356        // Iterate the returned data for the events.
357        foreach ($events as $event => $keys) {
358            if ($keys === false) {
359                // No data to be invalidated yet.
360                continue;
361            }
362
363            // Look at each key and check the timestamp.
364            foreach ($keys as $key => $purgetoken) {
365                // If the timestamp of the event is more than or equal to the last invalidation (happened between the last
366                // invalidation and now), then we need to invaliate the key.
367                if (self::compare_purge_tokens($purgetoken, $lastinvalidation) > 0) {
368                    if ($key === 'purged') {
369                        $purgeall = true;
370                        break;
371                    } else {
372                        $todelete[] = $key;
373                    }
374                }
375            }
376        }
377        if ($purgeall) {
378            $this->purge();
379        } else if (!empty($todelete)) {
380            $todelete = array_unique($todelete);
381            $this->delete_many($todelete);
382        }
383        // Set the time of the last invalidation.
384        if ($purgeall || !empty($todelete)) {
385            $this->set('lastinvalidation', self::get_purge_token(true));
386        }
387    }
388
389    /**
390     * Retrieves the value for the given key from the cache.
391     *
392     * @param string|int $key The key for the data being requested.
393     *      It can be any structure although using a scalar string or int is recommended in the interests of performance.
394     *      In advanced cases an array may be useful such as in situations requiring the multi-key functionality.
395     * @param int $strictness One of IGNORE_MISSING | MUST_EXIST
396     * @return mixed|false The data from the cache or false if the key did not exist within the cache.
397     * @throws coding_exception
398     */
399    public function get($key, $strictness = IGNORE_MISSING) {
400        // 1. Get it from the static acceleration array if we can (only when it is enabled and it has already been requested/set).
401        $usesstaticacceleration = $this->use_static_acceleration();
402
403        if ($usesstaticacceleration) {
404            $result = $this->static_acceleration_get($key);
405            if ($result !== false) {
406                return $result;
407            }
408        }
409
410        // 2. Parse the key.
411        $parsedkey = $this->parse_key($key);
412
413        // 3. Get it from the store. Obviously wasn't in the static acceleration array.
414        $result = $this->store->get($parsedkey);
415        if ($result !== false) {
416            if ($result instanceof cache_ttl_wrapper) {
417                if ($result->has_expired()) {
418                    $this->store->delete($parsedkey);
419                    $result = false;
420                } else {
421                    $result = $result->data;
422                }
423            }
424            if ($usesstaticacceleration) {
425                $this->static_acceleration_set($key, $result);
426            }
427            if ($result instanceof cache_cached_object) {
428                $result = $result->restore_object();
429            }
430        }
431
432        // 4. Load if from the loader/datasource if we don't already have it.
433        $setaftervalidation = false;
434        if ($result === false) {
435            if ($this->perfdebug) {
436                cache_helper::record_cache_miss($this->store, $this->definition);
437            }
438            if ($this->loader !== false) {
439                // We must pass the original (unparsed) key to the next loader in the chain.
440                // The next loader will parse the key as it sees fit. It may be parsed differently
441                // depending upon the capabilities of the store associated with the loader.
442                $result = $this->loader->get($key);
443            } else if ($this->datasource !== false) {
444                $result = $this->datasource->load_for_cache($key);
445            }
446            $setaftervalidation = ($result !== false);
447        } else if ($this->perfdebug) {
448            cache_helper::record_cache_hit($this->store, $this->definition);
449        }
450        // 5. Validate strictness.
451        if ($strictness === MUST_EXIST && $result === false) {
452            throw new coding_exception('Requested key did not exist in any cache stores and could not be loaded.');
453        }
454        // 6. Set it to the store if we got it from the loader/datasource.
455        if ($setaftervalidation) {
456            $this->set($key, $result);
457        }
458        // 7. Make sure we don't pass back anything that could be a reference.
459        //    We don't want people modifying the data in the cache.
460        if (!$this->store->supports_dereferencing_objects() && !is_scalar($result)) {
461            // If data is an object it will be a reference.
462            // If data is an array if may contain references.
463            // We want to break references so that the cache cannot be modified outside of itself.
464            // Call the function to unreference it (in the best way possible).
465            $result = $this->unref($result);
466        }
467        return $result;
468    }
469
470    /**
471     * Retrieves an array of values for an array of keys.
472     *
473     * Using this function comes with potential performance implications.
474     * Not all cache stores will support get_many/set_many operations and in order to replicate this functionality will call
475     * the equivalent singular method for each item provided.
476     * This should not deter you from using this function as there is a performance benefit in situations where the cache store
477     * does support it, but you should be aware of this fact.
478     *
479     * @param array $keys The keys of the data being requested.
480     *      Each key can be any structure although using a scalar string or int is recommended in the interests of performance.
481     *      In advanced cases an array may be useful such as in situations requiring the multi-key functionality.
482     * @param int $strictness One of IGNORE_MISSING or MUST_EXIST.
483     * @return array An array of key value pairs for the items that could be retrieved from the cache.
484     *      If MUST_EXIST was used and not all keys existed within the cache then an exception will be thrown.
485     *      Otherwise any key that did not exist will have a data value of false within the results.
486     * @throws coding_exception
487     */
488    public function get_many(array $keys, $strictness = IGNORE_MISSING) {
489
490        $keysparsed = array();
491        $parsedkeys = array();
492        $resultpersist = array();
493        $resultstore = array();
494        $keystofind = array();
495
496        // First up check the persist cache for each key.
497        $isusingpersist = $this->use_static_acceleration();
498        foreach ($keys as $key) {
499            $pkey = $this->parse_key($key);
500            if (is_array($pkey)) {
501                $pkey = $pkey['key'];
502            }
503            $keysparsed[$key] = $pkey;
504            $parsedkeys[$pkey] = $key;
505            $keystofind[$pkey] = $key;
506            if ($isusingpersist) {
507                $value = $this->static_acceleration_get($key);
508                if ($value !== false) {
509                    $resultpersist[$pkey] = $value;
510                    unset($keystofind[$pkey]);
511                }
512            }
513        }
514
515        // Next assuming we didn't find all of the keys in the persist cache try loading them from the store.
516        if (count($keystofind)) {
517            $resultstore = $this->store->get_many(array_keys($keystofind));
518            // Process each item in the result to "unwrap" it.
519            foreach ($resultstore as $key => $value) {
520                if ($value instanceof cache_ttl_wrapper) {
521                    if ($value->has_expired()) {
522                        $value = false;
523                    } else {
524                        $value = $value->data;
525                    }
526                }
527                if ($value !== false && $this->use_static_acceleration()) {
528                    $this->static_acceleration_set($keystofind[$key], $value);
529                }
530                if ($value instanceof cache_cached_object) {
531                    $value = $value->restore_object();
532                }
533                $resultstore[$key] = $value;
534            }
535        }
536
537        // Merge the result from the persis cache with the results from the store load.
538        $result = $resultpersist + $resultstore;
539        unset($resultpersist);
540        unset($resultstore);
541
542        // Next we need to find any missing values and load them from the loader/datasource next in the chain.
543        $usingloader = ($this->loader !== false);
544        $usingsource = (!$usingloader && ($this->datasource !== false));
545        if ($usingloader || $usingsource) {
546            $missingkeys = array();
547            foreach ($result as $key => $value) {
548                if ($value === false) {
549                    $missingkeys[] = $parsedkeys[$key];
550                }
551            }
552            if (!empty($missingkeys)) {
553                if ($usingloader) {
554                    $resultmissing = $this->loader->get_many($missingkeys);
555                } else {
556                    $resultmissing = $this->datasource->load_many_for_cache($missingkeys);
557                }
558                foreach ($resultmissing as $key => $value) {
559                    $result[$keysparsed[$key]] = $value;
560                    if ($value !== false) {
561                        $this->set($key, $value);
562                    }
563                }
564                unset($resultmissing);
565            }
566            unset($missingkeys);
567        }
568
569        // Create an array with the original keys and the found values. This will be what we return.
570        $fullresult = array();
571        foreach ($result as $key => $value) {
572            if (!is_scalar($value)) {
573                // If data is an object it will be a reference.
574                // If data is an array if may contain references.
575                // We want to break references so that the cache cannot be modified outside of itself.
576                // Call the function to unreference it (in the best way possible).
577                $value = $this->unref($value);
578            }
579            $fullresult[$parsedkeys[$key]] = $value;
580        }
581        unset($result);
582
583        // Final step is to check strictness.
584        if ($strictness === MUST_EXIST) {
585            foreach ($keys as $key) {
586                if (!array_key_exists($key, $fullresult)) {
587                    throw new coding_exception('Not all the requested keys existed within the cache stores.');
588                }
589            }
590        }
591
592        if ($this->perfdebug) {
593            $hits = 0;
594            $misses = 0;
595            foreach ($fullresult as $value) {
596                if ($value === false) {
597                    $misses++;
598                } else {
599                    $hits++;
600                }
601            }
602            cache_helper::record_cache_hit($this->store, $this->definition, $hits);
603            cache_helper::record_cache_miss($this->store, $this->definition, $misses);
604        }
605
606        // Return the result. Phew!
607        return $fullresult;
608    }
609
610    /**
611     * Sends a key => value pair to the cache.
612     *
613     * <code>
614     * // This code will add four entries to the cache, one for each url.
615     * $cache->set('main', 'http://moodle.org');
616     * $cache->set('docs', 'http://docs.moodle.org');
617     * $cache->set('tracker', 'http://tracker.moodle.org');
618     * $cache->set('qa', 'http://qa.moodle.net');
619     * </code>
620     *
621     * @param string|int $key The key for the data being requested.
622     *      It can be any structure although using a scalar string or int is recommended in the interests of performance.
623     *      In advanced cases an array may be useful such as in situations requiring the multi-key functionality.
624     * @param mixed $data The data to set against the key.
625     * @return bool True on success, false otherwise.
626     */
627    public function set($key, $data) {
628        if ($this->perfdebug) {
629            cache_helper::record_cache_set($this->store, $this->definition);
630        }
631        if ($this->loader !== false) {
632            // We have a loader available set it there as well.
633            // We have to let the loader do its own parsing of data as it may be unique.
634            $this->loader->set($key, $data);
635        }
636        $usestaticacceleration = $this->use_static_acceleration();
637
638        if (is_object($data) && $data instanceof cacheable_object) {
639            $data = new cache_cached_object($data);
640        } else if (!$this->store->supports_dereferencing_objects() && !is_scalar($data)) {
641            // If data is an object it will be a reference.
642            // If data is an array if may contain references.
643            // We want to break references so that the cache cannot be modified outside of itself.
644            // Call the function to unreference it (in the best way possible).
645            $data = $this->unref($data);
646        }
647
648        if ($usestaticacceleration) {
649            $this->static_acceleration_set($key, $data);
650        }
651
652        if ($this->has_a_ttl() && !$this->store_supports_native_ttl()) {
653            $data = new cache_ttl_wrapper($data, $this->definition->get_ttl());
654        }
655        $parsedkey = $this->parse_key($key);
656
657        return $this->store->set($parsedkey, $data);
658    }
659
660    /**
661     * Removes references where required.
662     *
663     * @param stdClass|array $data
664     * @return mixed What ever was put in but without any references.
665     */
666    protected function unref($data) {
667        if ($this->definition->uses_simple_data()) {
668            return $data;
669        }
670        // Check if it requires serialisation in order to produce a reference free copy.
671        if ($this->requires_serialisation($data)) {
672            // Damn, its going to have to be serialise.
673            $data = serialize($data);
674            // We unserialise immediately so that we don't have to do it every time on get.
675            $data = unserialize($data);
676        } else if (!is_scalar($data)) {
677            // Its safe to clone, lets do it, its going to beat the pants of serialisation.
678            $data = $this->deep_clone($data);
679        }
680        return $data;
681    }
682
683    /**
684     * Checks to see if a var requires serialisation.
685     *
686     * @param mixed $value The value to check.
687     * @param int $depth Used to ensure we don't enter an endless loop (think recursion).
688     * @return bool Returns true if the value is going to require serialisation in order to ensure a reference free copy
689     *      or false if its safe to clone.
690     */
691    protected function requires_serialisation($value, $depth = 1) {
692        if (is_scalar($value)) {
693            return false;
694        } else if (is_array($value) || $value instanceof stdClass || $value instanceof Traversable) {
695            if ($depth > 5) {
696                // Skrew it, mega-deep object, developer you suck, we're just going to serialise.
697                return true;
698            }
699            foreach ($value as $key => $subvalue) {
700                if ($this->requires_serialisation($subvalue, $depth++)) {
701                    return true;
702                }
703            }
704        }
705        // Its not scalar, array, or stdClass so we'll need to serialise.
706        return true;
707    }
708
709    /**
710     * Creates a reference free clone of the given value.
711     *
712     * @param mixed $value
713     * @return mixed
714     */
715    protected function deep_clone($value) {
716        if (is_object($value)) {
717            // Objects must be cloned to begin with.
718            $value = clone $value;
719        }
720        if (is_array($value) || is_object($value)) {
721            foreach ($value as $key => $subvalue) {
722                $value[$key] = $this->deep_clone($subvalue);
723            }
724        }
725        return $value;
726    }
727
728    /**
729     * Sends several key => value pairs to the cache.
730     *
731     * Using this function comes with potential performance implications.
732     * Not all cache stores will support get_many/set_many operations and in order to replicate this functionality will call
733     * the equivalent singular method for each item provided.
734     * This should not deter you from using this function as there is a performance benefit in situations where the cache store
735     * does support it, but you should be aware of this fact.
736     *
737     * <code>
738     * // This code will add four entries to the cache, one for each url.
739     * $cache->set_many(array(
740     *     'main' => 'http://moodle.org',
741     *     'docs' => 'http://docs.moodle.org',
742     *     'tracker' => 'http://tracker.moodle.org',
743     *     'qa' => ''http://qa.moodle.net'
744     * ));
745     * </code>
746     *
747     * @param array $keyvaluearray An array of key => value pairs to send to the cache.
748     * @return int The number of items successfully set. It is up to the developer to check this matches the number of items.
749     *      ... if they care that is.
750     */
751    public function set_many(array $keyvaluearray) {
752        if ($this->loader !== false) {
753            // We have a loader available set it there as well.
754            // We have to let the loader do its own parsing of data as it may be unique.
755            $this->loader->set_many($keyvaluearray);
756        }
757        $data = array();
758        $simulatettl = $this->has_a_ttl() && !$this->store_supports_native_ttl();
759        $usestaticaccelerationarray = $this->use_static_acceleration();
760        $needsdereferencing = !$this->store->supports_dereferencing_objects();
761        foreach ($keyvaluearray as $key => $value) {
762            if (is_object($value) && $value instanceof cacheable_object) {
763                $value = new cache_cached_object($value);
764            } else if ($needsdereferencing && !is_scalar($value)) {
765                // If data is an object it will be a reference.
766                // If data is an array if may contain references.
767                // We want to break references so that the cache cannot be modified outside of itself.
768                // Call the function to unreference it (in the best way possible).
769                $value = $this->unref($value);
770            }
771            if ($usestaticaccelerationarray) {
772                $this->static_acceleration_set($key, $value);
773            }
774            if ($simulatettl) {
775                $value = new cache_ttl_wrapper($value, $this->definition->get_ttl());
776            }
777            $data[$key] = array(
778                'key' => $this->parse_key($key),
779                'value' => $value
780            );
781        }
782        $successfullyset = $this->store->set_many($data);
783        if ($this->perfdebug && $successfullyset) {
784            cache_helper::record_cache_set($this->store, $this->definition, $successfullyset);
785        }
786        return $successfullyset;
787    }
788
789    /**
790     * Test is a cache has a key.
791     *
792     * The use of the has methods is strongly discouraged. In a high load environment the cache may well change between the
793     * test and any subsequent action (get, set, delete etc).
794     * Instead it is recommended to write your code in such a way they it performs the following steps:
795     * <ol>
796     * <li>Attempt to retrieve the information.</li>
797     * <li>Generate the information.</li>
798     * <li>Attempt to set the information</li>
799     * </ol>
800     *
801     * Its also worth mentioning that not all stores support key tests.
802     * For stores that don't support key tests this functionality is mimicked by using the equivalent get method.
803     * Just one more reason you should not use these methods unless you have a very good reason to do so.
804     *
805     * @param string|int $key
806     * @param bool $tryloadifpossible If set to true, the cache doesn't contain the key, and there is another cache loader or
807     *      data source then the code will try load the key value from the next item in the chain.
808     * @return bool True if the cache has the requested key, false otherwise.
809     */
810    public function has($key, $tryloadifpossible = false) {
811        if ($this->static_acceleration_has($key)) {
812            // Hoorah, that was easy. It exists in the static acceleration array so we definitely have it.
813            return true;
814        }
815        $parsedkey = $this->parse_key($key);
816
817        if ($this->has_a_ttl() && !$this->store_supports_native_ttl()) {
818            // The data has a TTL and the store doesn't support it natively.
819            // We must fetch the data and expect a ttl wrapper.
820            $data = $this->store->get($parsedkey);
821            $has = ($data instanceof cache_ttl_wrapper && !$data->has_expired());
822        } else if (!$this->store_supports_key_awareness()) {
823            // The store doesn't support key awareness, get the data and check it manually... puke.
824            // Either no TTL is set of the store supports its handling natively.
825            $data = $this->store->get($parsedkey);
826            $has = ($data !== false);
827        } else {
828            // The store supports key awareness, this is easy!
829            // Either no TTL is set of the store supports its handling natively.
830            $has = $this->store->has($parsedkey);
831        }
832        if (!$has && $tryloadifpossible) {
833            if ($this->loader !== false) {
834                $result = $this->loader->get($parsedkey);
835            } else if ($this->datasource !== null) {
836                $result = $this->datasource->load_for_cache($key);
837            }
838            $has = ($result !== null);
839            if ($has) {
840                $this->set($key, $result);
841            }
842        }
843        return $has;
844    }
845
846    /**
847     * Test is a cache has all of the given keys.
848     *
849     * It is strongly recommended to avoid the use of this function if not absolutely required.
850     * In a high load environment the cache may well change between the test and any subsequent action (get, set, delete etc).
851     *
852     * Its also worth mentioning that not all stores support key tests.
853     * For stores that don't support key tests this functionality is mimicked by using the equivalent get method.
854     * Just one more reason you should not use these methods unless you have a very good reason to do so.
855     *
856     * @param array $keys
857     * @return bool True if the cache has all of the given keys, false otherwise.
858     */
859    public function has_all(array $keys) {
860        if (($this->has_a_ttl() && !$this->store_supports_native_ttl()) || !$this->store_supports_key_awareness()) {
861            foreach ($keys as $key) {
862                if (!$this->has($key)) {
863                    return false;
864                }
865            }
866            return true;
867        }
868        $parsedkeys = array_map(array($this, 'parse_key'), $keys);
869        return $this->store->has_all($parsedkeys);
870    }
871
872    /**
873     * Test if a cache has at least one of the given keys.
874     *
875     * It is strongly recommended to avoid the use of this function if not absolutely required.
876     * In a high load environment the cache may well change between the test and any subsequent action (get, set, delete etc).
877     *
878     * Its also worth mentioning that not all stores support key tests.
879     * For stores that don't support key tests this functionality is mimicked by using the equivalent get method.
880     * Just one more reason you should not use these methods unless you have a very good reason to do so.
881     *
882     * @param array $keys
883     * @return bool True if the cache has at least one of the given keys
884     */
885    public function has_any(array $keys) {
886        if (($this->has_a_ttl() && !$this->store_supports_native_ttl()) || !$this->store_supports_key_awareness()) {
887            foreach ($keys as $key) {
888                if ($this->has($key)) {
889                    return true;
890                }
891            }
892            return false;
893        }
894
895        if ($this->use_static_acceleration()) {
896            foreach ($keys as $id => $key) {
897                if ($this->static_acceleration_has($key)) {
898                    return true;
899                }
900            }
901        }
902        $parsedkeys = array_map(array($this, 'parse_key'), $keys);
903        return $this->store->has_any($parsedkeys);
904    }
905
906    /**
907     * Delete the given key from the cache.
908     *
909     * @param string|int $key The key to delete.
910     * @param bool $recurse When set to true the key will also be deleted from all stacked cache loaders and their stores.
911     *     This happens by default and ensure that all the caches are consistent. It is NOT recommended to change this.
912     * @return bool True of success, false otherwise.
913     */
914    public function delete($key, $recurse = true) {
915        $this->static_acceleration_delete($key);
916        if ($recurse && $this->loader !== false) {
917            // Delete from the bottom of the stack first.
918            $this->loader->delete($key, $recurse);
919        }
920        $parsedkey = $this->parse_key($key);
921        return $this->store->delete($parsedkey);
922    }
923
924    /**
925     * Delete all of the given keys from the cache.
926     *
927     * @param array $keys The key to delete.
928     * @param bool $recurse When set to true the key will also be deleted from all stacked cache loaders and their stores.
929     *     This happens by default and ensure that all the caches are consistent. It is NOT recommended to change this.
930     * @return int The number of items successfully deleted.
931     */
932    public function delete_many(array $keys, $recurse = true) {
933        if ($this->use_static_acceleration()) {
934            foreach ($keys as $key) {
935                $this->static_acceleration_delete($key);
936            }
937        }
938        if ($recurse && $this->loader !== false) {
939            // Delete from the bottom of the stack first.
940            $this->loader->delete_many($keys, $recurse);
941        }
942        $parsedkeys = array_map(array($this, 'parse_key'), $keys);
943        return $this->store->delete_many($parsedkeys);
944    }
945
946    /**
947     * Purges the cache store, and loader if there is one.
948     *
949     * @return bool True on success, false otherwise
950     */
951    public function purge() {
952        // 1. Purge the static acceleration array.
953        $this->static_acceleration_purge();
954        // 2. Purge the store.
955        $this->store->purge();
956        // 3. Optionally pruge any stacked loaders.
957        if ($this->loader) {
958            $this->loader->purge();
959        }
960        return true;
961    }
962
963    /**
964     * Parses the key turning it into a string (or array is required) suitable to be passed to the cache store.
965     *
966     * @param string|int $key As passed to get|set|delete etc.
967     * @return string|array String unless the store supports multi-identifiers in which case an array if returned.
968     */
969    protected function parse_key($key) {
970        // First up if the store supports multiple keys we'll go with that.
971        if ($this->store->supports_multiple_identifiers()) {
972            $result = $this->definition->generate_multi_key_parts();
973            $result['key'] = $key;
974            return $result;
975        }
976        // If not we need to generate a hash and to for that we use the cache_helper.
977        return cache_helper::hash_key($key, $this->definition);
978    }
979
980    /**
981     * Returns true if the cache is making use of a ttl.
982     * @return bool
983     */
984    protected function has_a_ttl() {
985        return $this->hasattl;
986    }
987
988    /**
989     * Returns true if the cache store supports native ttl.
990     * @return bool
991     */
992    protected function store_supports_native_ttl() {
993        if ($this->supportsnativettl === null) {
994            $this->supportsnativettl = ($this->store->supports_native_ttl());
995        }
996        return $this->supportsnativettl;
997    }
998
999    /**
1000     * Returns the cache definition.
1001     *
1002     * @return cache_definition
1003     */
1004    protected function get_definition() {
1005        return $this->definition;
1006    }
1007
1008    /**
1009     * Returns the cache store
1010     *
1011     * @return cache_store
1012     */
1013    protected function get_store() {
1014        return $this->store;
1015    }
1016
1017    /**
1018     * Returns the loader associated with this instance.
1019     *
1020     * @since Moodle 2.4.4
1021     * @return cache|false
1022     */
1023    protected function get_loader() {
1024        return $this->loader;
1025    }
1026
1027    /**
1028     * Returns the data source associated with this cache.
1029     *
1030     * @since Moodle 2.4.4
1031     * @return cache_data_source|false
1032     */
1033    protected function get_datasource() {
1034        return $this->datasource;
1035    }
1036
1037    /**
1038     * Returns true if the store supports key awareness.
1039     *
1040     * @return bool
1041     */
1042    protected function store_supports_key_awareness() {
1043        if ($this->supportskeyawareness === null) {
1044            $this->supportskeyawareness = ($this->store instanceof cache_is_key_aware);
1045        }
1046        return $this->supportskeyawareness;
1047    }
1048
1049    /**
1050     * Returns true if the store natively supports locking.
1051     *
1052     * @return bool
1053     */
1054    protected function store_supports_native_locking() {
1055        if ($this->nativelocking === null) {
1056            $this->nativelocking = ($this->store instanceof cache_is_lockable);
1057        }
1058        return $this->nativelocking;
1059    }
1060
1061    /**
1062     * @deprecated since 2.6
1063     * @see cache::use_static_acceleration()
1064     */
1065    protected function is_using_persist_cache() {
1066        throw new coding_exception('cache::is_using_persist_cache() can not be used anymore.' .
1067            ' Please use cache::use_static_acceleration() instead.');
1068    }
1069
1070    /**
1071     * Returns true if this cache is making use of the static acceleration array.
1072     *
1073     * @return bool
1074     */
1075    protected function use_static_acceleration() {
1076        return $this->staticacceleration;
1077    }
1078
1079    /**
1080     * @see cache::static_acceleration_has
1081     * @deprecated since 2.6
1082     */
1083    protected function is_in_persist_cache() {
1084        throw new coding_exception('cache::is_in_persist_cache() can not be used anymore.' .
1085            ' Please use cache::static_acceleration_has() instead.');
1086    }
1087
1088    /**
1089     * Returns true if the requested key exists within the static acceleration array.
1090     *
1091     * @param string $key The parsed key
1092     * @return bool
1093     */
1094    protected function static_acceleration_has($key) {
1095        // This could be written as a single line, however it has been split because the ttl check is faster than the instanceof
1096        // and has_expired calls.
1097        if (!$this->staticacceleration || !isset($this->staticaccelerationarray[$key])) {
1098            return false;
1099        }
1100        return true;
1101    }
1102
1103    /**
1104     * @deprecated since 2.6
1105     * @see cache::static_acceleration_get
1106     */
1107    protected function get_from_persist_cache() {
1108        throw new coding_exception('cache::get_from_persist_cache() can not be used anymore.' .
1109            ' Please use cache::static_acceleration_get() instead.');
1110    }
1111
1112    /**
1113     * Returns the item from the static acceleration array if it exists there.
1114     *
1115     * @param string $key The parsed key
1116     * @return mixed|false Dereferenced data from the static acceleration array or false if it wasn't there.
1117     */
1118    protected function static_acceleration_get($key) {
1119        if (!$this->staticacceleration || !isset($this->staticaccelerationarray[$key])) {
1120            $result = false;
1121        } else {
1122            $data = $this->staticaccelerationarray[$key]['data'];
1123
1124            if ($data instanceof cache_cached_object) {
1125                $result = $data->restore_object();
1126            } else if ($this->staticaccelerationarray[$key]['serialized']) {
1127                $result = unserialize($data);
1128            } else {
1129                $result = $data;
1130            }
1131        }
1132        if ($result !== false) {
1133            if ($this->perfdebug) {
1134                cache_helper::record_cache_hit(cache_store::STATIC_ACCEL, $this->definition);
1135            }
1136            if ($this->staticaccelerationsize > 1 && $this->staticaccelerationcount > 1) {
1137                // Check to see if this is the last item on the static acceleration keys array.
1138                if (end($this->staticaccelerationkeys) !== $key) {
1139                    // It isn't the last item.
1140                    // Move the item to the end of the array so that it is last to be removed.
1141                    unset($this->staticaccelerationkeys[$key]);
1142                    $this->staticaccelerationkeys[$key] = $key;
1143                }
1144            }
1145            return $result;
1146        } else {
1147            if ($this->perfdebug) {
1148                cache_helper::record_cache_miss(cache_store::STATIC_ACCEL, $this->definition);
1149            }
1150            return false;
1151        }
1152    }
1153
1154    /**
1155     * @deprecated since 2.6
1156     * @see cache::static_acceleration_set
1157     */
1158    protected function set_in_persist_cache() {
1159        throw new coding_exception('cache::set_in_persist_cache() can not be used anymore.' .
1160            ' Please use cache::static_acceleration_set() instead.');
1161    }
1162
1163    /**
1164     * Sets a key value pair into the static acceleration array.
1165     *
1166     * @param string $key The parsed key
1167     * @param mixed $data
1168     * @return bool
1169     */
1170    protected function static_acceleration_set($key, $data) {
1171        if ($this->staticaccelerationsize !== false && isset($this->staticaccelerationkeys[$key])) {
1172            $this->staticaccelerationcount--;
1173            unset($this->staticaccelerationkeys[$key]);
1174        }
1175
1176        // We serialize anything that's not;
1177        // 1. A known scalar safe value.
1178        // 2. A definition that says it's simpledata.  We trust it that it doesn't contain dangerous references.
1179        // 3. An object that handles dereferencing by itself.
1180        if (is_scalar($data) || $this->definition->uses_simple_data()
1181                || $data instanceof cache_cached_object) {
1182            $this->staticaccelerationarray[$key]['data'] = $data;
1183            $this->staticaccelerationarray[$key]['serialized'] = false;
1184        } else {
1185            $this->staticaccelerationarray[$key]['data'] = serialize($data);
1186            $this->staticaccelerationarray[$key]['serialized'] = true;
1187        }
1188        if ($this->staticaccelerationsize !== false) {
1189            $this->staticaccelerationcount++;
1190            $this->staticaccelerationkeys[$key] = $key;
1191            if ($this->staticaccelerationcount > $this->staticaccelerationsize) {
1192                $dropkey = array_shift($this->staticaccelerationkeys);
1193                unset($this->staticaccelerationarray[$dropkey]);
1194                $this->staticaccelerationcount--;
1195            }
1196        }
1197        return true;
1198    }
1199
1200    /**
1201     * @deprecated since 2.6
1202     * @see cache::static_acceleration_delete()
1203     */
1204    protected function delete_from_persist_cache() {
1205        throw new coding_exception('cache::delete_from_persist_cache() can not be used anymore.' .
1206            ' Please use cache::static_acceleration_delete() instead.');
1207    }
1208
1209    /**
1210     * Deletes an item from the static acceleration array.
1211     *
1212     * @param string|int $key As given to get|set|delete
1213     * @return bool True on success, false otherwise.
1214     */
1215    protected function static_acceleration_delete($key) {
1216        unset($this->staticaccelerationarray[$key]);
1217        if ($this->staticaccelerationsize !== false && isset($this->staticaccelerationkeys[$key])) {
1218            unset($this->staticaccelerationkeys[$key]);
1219            $this->staticaccelerationcount--;
1220        }
1221        return true;
1222    }
1223
1224    /**
1225     * Purge the static acceleration cache.
1226     */
1227    protected function static_acceleration_purge() {
1228        $this->staticaccelerationarray = array();
1229        if ($this->staticaccelerationsize !== false) {
1230            $this->staticaccelerationkeys = array();
1231            $this->staticaccelerationcount = 0;
1232        }
1233    }
1234
1235    /**
1236     * Returns the timestamp from the first request for the time from the cache API.
1237     *
1238     * This stamp needs to be used for all ttl and time based operations to ensure that we don't end up with
1239     * timing issues.
1240     *
1241     * @param   bool    $float Whether to use floating precision accuracy.
1242     * @return  int|float
1243     */
1244    public static function now($float = false) {
1245        if (self::$now === null) {
1246            self::$now = microtime(true);
1247        }
1248
1249        if ($float) {
1250            return self::$now;
1251        } else {
1252            return (int) self::$now;
1253        }
1254    }
1255
1256    /**
1257     * Get a 'purge' token used for cache invalidation handling.
1258     *
1259     * Note: This function is intended for use from within the Cache API only and not by plugins, or cache stores.
1260     *
1261     * @param   bool    $reset  Whether to reset the token and generate a new one.
1262     * @return  string
1263     */
1264    public static function get_purge_token($reset = false) {
1265        if (self::$purgetoken === null || $reset) {
1266            self::$now = null;
1267            self::$purgetoken = self::now(true) . '-' . uniqid('', true);
1268        }
1269
1270        return self::$purgetoken;
1271    }
1272
1273    /**
1274     * Compare a pair of purge tokens.
1275     *
1276     * If the two tokens are identical, then the return value is 0.
1277     * If the time component of token A is newer than token B, then a positive value is returned.
1278     * If the time component of token B is newer than token A, then a negative value is returned.
1279     *
1280     * Note: This function is intended for use from within the Cache API only and not by plugins, or cache stores.
1281     *
1282     * @param   string  $tokena
1283     * @param   string  $tokenb
1284     * @return  int
1285     */
1286    public static function compare_purge_tokens($tokena, $tokenb) {
1287        if ($tokena === $tokenb) {
1288            // There is an exact match.
1289            return 0;
1290        }
1291
1292        // The token for when the cache was last invalidated.
1293        list($atime) = explode('-', "{$tokena}-", 2);
1294
1295        // The token for this cache.
1296        list($btime) = explode('-', "{$tokenb}-", 2);
1297
1298        if ($atime >= $btime) {
1299            // Token A is newer.
1300            return 1;
1301        } else {
1302            // Token A is older.
1303            return -1;
1304        }
1305    }
1306
1307    /**
1308     * Subclasses may support purging cache of all data belonging to the
1309     * current user.
1310     */
1311    public function purge_current_user() {
1312    }
1313}
1314
1315/**
1316 * An application cache.
1317 *
1318 * This class is used for application caches returned by the cache::make methods.
1319 * On top of the standard functionality it also allows locking to be required and or manually operated.
1320 *
1321 * This cache class should never be interacted with directly. Instead you should always use the cache::make methods.
1322 * It is technically possible to call those methods through this class however there is no guarantee that you will get an
1323 * instance of this class back again.
1324 *
1325 * @internal don't use me directly.
1326 *
1327 * @package    core
1328 * @category   cache
1329 * @copyright  2012 Sam Hemelryk
1330 * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
1331 */
1332class cache_application extends cache implements cache_loader_with_locking {
1333
1334    /**
1335     * Lock identifier.
1336     * This is used to ensure the lock belongs to the cache instance + definition + user.
1337     * @var string
1338     */
1339    protected $lockidentifier;
1340
1341    /**
1342     * Gets set to true if the cache's primary store natively supports locking.
1343     * If it does then we use that, otherwise we need to instantiate a second store to use for locking.
1344     * @var cache_store
1345     */
1346    protected $nativelocking = null;
1347
1348    /**
1349     * Gets set to true if the cache is going to be using locking.
1350     * This isn't a requirement, it doesn't need to use locking (most won't) and this bool is used to quickly check things.
1351     * If required then locking will be forced for the get|set|delete operation.
1352     * @var bool
1353     */
1354    protected $requirelocking = false;
1355
1356    /**
1357     * Gets set to true if the cache must use read locking (get|has).
1358     * @var bool
1359     */
1360    protected $requirelockingread = false;
1361
1362    /**
1363     * Gets set to true if the cache must use write locking (set|delete)
1364     * @var bool
1365     */
1366    protected $requirelockingwrite = false;
1367
1368    /**
1369     * Gets set to a cache_store to use for locking if the caches primary store doesn't support locking natively.
1370     * @var cache_lock_interface
1371     */
1372    protected $cachelockinstance;
1373
1374    /**
1375     * Overrides the cache construct method.
1376     *
1377     * You should not call this method from your code, instead you should use the cache::make methods.
1378     *
1379     * @param cache_definition $definition
1380     * @param cache_store $store
1381     * @param cache_loader|cache_data_source $loader
1382     */
1383    public function __construct(cache_definition $definition, cache_store $store, $loader = null) {
1384        parent::__construct($definition, $store, $loader);
1385        $this->nativelocking = $this->store_supports_native_locking();
1386        if ($definition->require_locking()) {
1387            $this->requirelocking = true;
1388            $this->requirelockingread = $definition->require_locking_read();
1389            $this->requirelockingwrite = $definition->require_locking_write();
1390        }
1391
1392        $this->handle_invalidation_events();
1393    }
1394
1395    /**
1396     * Returns the identifier to use
1397     *
1398     * @staticvar int $instances Counts the number of instances. Used as part of the lock identifier.
1399     * @return string
1400     */
1401    public function get_identifier() {
1402        static $instances = 0;
1403        if ($this->lockidentifier === null) {
1404            $this->lockidentifier = md5(
1405                $this->get_definition()->generate_definition_hash() .
1406                sesskey() .
1407                $instances++ .
1408                'cache_application'
1409            );
1410        }
1411        return $this->lockidentifier;
1412    }
1413
1414    /**
1415     * Fixes the instance up after a clone.
1416     */
1417    public function __clone() {
1418        // Force a new idenfitier.
1419        $this->lockidentifier = null;
1420    }
1421
1422    /**
1423     * Acquires a lock on the given key.
1424     *
1425     * This is done automatically if the definition requires it.
1426     * It is recommended to use a definition if you want to have locking although it is possible to do locking without having
1427     * it required by the definition.
1428     * The problem with such an approach is that you cannot ensure that code will consistently use locking. You will need to
1429     * rely on the integrators review skills.
1430     *
1431     * @param string|int $key The key as given to get|set|delete
1432     * @return bool Returns true if the lock could be acquired, false otherwise.
1433     */
1434    public function acquire_lock($key) {
1435        $key = $this->parse_key($key);
1436        if ($this->nativelocking) {
1437            return $this->get_store()->acquire_lock($key, $this->get_identifier());
1438        } else {
1439            $this->ensure_cachelock_available();
1440            return $this->cachelockinstance->lock($key, $this->get_identifier());
1441        }
1442    }
1443
1444    /**
1445     * Checks if this cache has a lock on the given key.
1446     *
1447     * @param string|int $key The key as given to get|set|delete
1448     * @return bool|null Returns true if there is a lock and this cache has it, null if no one has a lock on that key, false if
1449     *      someone else has the lock.
1450     */
1451    public function check_lock_state($key) {
1452        $key = $this->parse_key($key);
1453        if ($this->nativelocking) {
1454            return $this->get_store()->check_lock_state($key, $this->get_identifier());
1455        } else {
1456            $this->ensure_cachelock_available();
1457            return $this->cachelockinstance->check_state($key, $this->get_identifier());
1458        }
1459    }
1460
1461    /**
1462     * Releases the lock this cache has on the given key
1463     *
1464     * @param string|int $key
1465     * @return bool True if the operation succeeded, false otherwise.
1466     */
1467    public function release_lock($key) {
1468        $key = $this->parse_key($key);
1469        if ($this->nativelocking) {
1470            return $this->get_store()->release_lock($key, $this->get_identifier());
1471        } else {
1472            $this->ensure_cachelock_available();
1473            return $this->cachelockinstance->unlock($key, $this->get_identifier());
1474        }
1475    }
1476
1477    /**
1478     * Ensure that the dedicated lock store is ready to go.
1479     *
1480     * This should only happen if the cache store doesn't natively support it.
1481     */
1482    protected function ensure_cachelock_available() {
1483        if ($this->cachelockinstance === null) {
1484            $this->cachelockinstance = cache_helper::get_cachelock_for_store($this->get_store());
1485        }
1486    }
1487
1488    /**
1489     * Sends a key => value pair to the cache.
1490     *
1491     * <code>
1492     * // This code will add four entries to the cache, one for each url.
1493     * $cache->set('main', 'http://moodle.org');
1494     * $cache->set('docs', 'http://docs.moodle.org');
1495     * $cache->set('tracker', 'http://tracker.moodle.org');
1496     * $cache->set('qa', 'http://qa.moodle.net');
1497     * </code>
1498     *
1499     * @param string|int $key The key for the data being requested.
1500     * @param mixed $data The data to set against the key.
1501     * @return bool True on success, false otherwise.
1502     */
1503    public function set($key, $data) {
1504        if ($this->requirelockingwrite && !$this->acquire_lock($key)) {
1505            return false;
1506        }
1507        $result = parent::set($key, $data);
1508        if ($this->requirelockingwrite && !$this->release_lock($key)) {
1509            debugging('Failed to release cache lock on set operation... this should not happen.', DEBUG_DEVELOPER);
1510        }
1511        return $result;
1512    }
1513
1514    /**
1515     * Sends several key => value pairs to the cache.
1516     *
1517     * Using this function comes with potential performance implications.
1518     * Not all cache stores will support get_many/set_many operations and in order to replicate this functionality will call
1519     * the equivalent singular method for each item provided.
1520     * This should not deter you from using this function as there is a performance benefit in situations where the cache store
1521     * does support it, but you should be aware of this fact.
1522     *
1523     * <code>
1524     * // This code will add four entries to the cache, one for each url.
1525     * $cache->set_many(array(
1526     *     'main' => 'http://moodle.org',
1527     *     'docs' => 'http://docs.moodle.org',
1528     *     'tracker' => 'http://tracker.moodle.org',
1529     *     'qa' => ''http://qa.moodle.net'
1530     * ));
1531     * </code>
1532     *
1533     * @param array $keyvaluearray An array of key => value pairs to send to the cache.
1534     * @return int The number of items successfully set. It is up to the developer to check this matches the number of items.
1535     *      ... if they care that is.
1536     */
1537    public function set_many(array $keyvaluearray) {
1538        if ($this->requirelockingwrite) {
1539            $locks = array();
1540            foreach ($keyvaluearray as $id => $pair) {
1541                $key = $pair['key'];
1542                if ($this->acquire_lock($key)) {
1543                    $locks[] = $key;
1544                } else {
1545                    unset($keyvaluearray[$id]);
1546                }
1547            }
1548        }
1549        $result = parent::set_many($keyvaluearray);
1550        if ($this->requirelockingwrite) {
1551            foreach ($locks as $key) {
1552                if ($this->release_lock($key)) {
1553                    debugging('Failed to release cache lock on set_many operation... this should not happen.', DEBUG_DEVELOPER);
1554                }
1555            }
1556        }
1557        return $result;
1558    }
1559
1560    /**
1561     * Retrieves the value for the given key from the cache.
1562     *
1563     * @param string|int $key The key for the data being requested.
1564     * @param int $strictness One of IGNORE_MISSING | MUST_EXIST
1565     * @return mixed|false The data from the cache or false if the key did not exist within the cache.
1566     */
1567    public function get($key, $strictness = IGNORE_MISSING) {
1568        if ($this->requirelockingread && $this->check_lock_state($key) === false) {
1569            // Read locking required and someone else has the read lock.
1570            return false;
1571        }
1572        return parent::get($key, $strictness);
1573    }
1574
1575    /**
1576     * Retrieves an array of values for an array of keys.
1577     *
1578     * Using this function comes with potential performance implications.
1579     * Not all cache stores will support get_many/set_many operations and in order to replicate this functionality will call
1580     * the equivalent singular method for each item provided.
1581     * This should not deter you from using this function as there is a performance benefit in situations where the cache store
1582     * does support it, but you should be aware of this fact.
1583     *
1584     * @param array $keys The keys of the data being requested.
1585     * @param int $strictness One of IGNORE_MISSING or MUST_EXIST.
1586     * @return array An array of key value pairs for the items that could be retrieved from the cache.
1587     *      If MUST_EXIST was used and not all keys existed within the cache then an exception will be thrown.
1588     *      Otherwise any key that did not exist will have a data value of false within the results.
1589     * @throws coding_exception
1590     */
1591    public function get_many(array $keys, $strictness = IGNORE_MISSING) {
1592        if ($this->requirelockingread) {
1593            foreach ($keys as $id => $key) {
1594                $lock =$this->acquire_lock($key);
1595                if (!$lock) {
1596                    if ($strictness === MUST_EXIST) {
1597                        throw new coding_exception('Could not acquire read locks for all of the items being requested.');
1598                    } else {
1599                        // Can't return this as we couldn't get a read lock.
1600                        unset($keys[$id]);
1601                    }
1602                }
1603
1604            }
1605        }
1606        return parent::get_many($keys, $strictness);
1607    }
1608
1609    /**
1610     * Delete the given key from the cache.
1611     *
1612     * @param string|int $key The key to delete.
1613     * @param bool $recurse When set to true the key will also be deleted from all stacked cache loaders and their stores.
1614     *     This happens by default and ensure that all the caches are consistent. It is NOT recommended to change this.
1615     * @return bool True of success, false otherwise.
1616     */
1617    public function delete($key, $recurse = true) {
1618        if ($this->requirelockingwrite && !$this->acquire_lock($key)) {
1619            return false;
1620        }
1621        $result = parent::delete($key, $recurse);
1622        if ($this->requirelockingwrite && !$this->release_lock($key)) {
1623            debugging('Failed to release cache lock on delete operation... this should not happen.', DEBUG_DEVELOPER);
1624        }
1625        return $result;
1626    }
1627
1628    /**
1629     * Delete all of the given keys from the cache.
1630     *
1631     * @param array $keys The key to delete.
1632     * @param bool $recurse When set to true the key will also be deleted from all stacked cache loaders and their stores.
1633     *     This happens by default and ensure that all the caches are consistent. It is NOT recommended to change this.
1634     * @return int The number of items successfully deleted.
1635     */
1636    public function delete_many(array $keys, $recurse = true) {
1637        if ($this->requirelockingwrite) {
1638            $locks = array();
1639            foreach ($keys as $id => $key) {
1640                if ($this->acquire_lock($key)) {
1641                    $locks[] = $key;
1642                } else {
1643                    unset($keys[$id]);
1644                }
1645            }
1646        }
1647        $result = parent::delete_many($keys, $recurse);
1648        if ($this->requirelockingwrite) {
1649            foreach ($locks as $key) {
1650                if ($this->release_lock($key)) {
1651                    debugging('Failed to release cache lock on delete_many operation... this should not happen.', DEBUG_DEVELOPER);
1652                }
1653            }
1654        }
1655        return $result;
1656    }
1657}
1658
1659/**
1660 * A session cache.
1661 *
1662 * This class is used for session caches returned by the cache::make methods.
1663 *
1664 * It differs from the application loader in a couple of noteable ways:
1665 *    1. Sessions are always expected to exist.
1666 *       Because of this we don't ever use the static acceleration array.
1667 *    2. Session data for a loader instance (store + definition) is consolidate into a
1668 *       single array for storage within the store.
1669 *       Along with this we embed a lastaccessed time with the data. This way we can
1670 *       check sessions for a last access time.
1671 *    3. Session stores are required to support key searching and must
1672 *       implement cache_is_searchable. This ensures stores used for the cache can be
1673 *       targetted for garbage collection of session data.
1674 *
1675 * This cache class should never be interacted with directly. Instead you should always use the cache::make methods.
1676 * It is technically possible to call those methods through this class however there is no guarantee that you will get an
1677 * instance of this class back again.
1678 *
1679 * @todo we should support locking in the session as well. Should be pretty simple to set up.
1680 *
1681 * @internal don't use me directly.
1682 * @method cache_store|cache_is_searchable get_store() Returns the cache store which must implement both cache_is_searchable.
1683 *
1684 * @package    core
1685 * @category   cache
1686 * @copyright  2012 Sam Hemelryk
1687 * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
1688 */
1689class cache_session extends cache {
1690    /**
1691     * The user the session has been established for.
1692     * @var int
1693     */
1694    protected static $loadeduserid = null;
1695
1696    /**
1697     * The userid this cache is currently using.
1698     * @var int
1699     */
1700    protected $currentuserid = null;
1701
1702    /**
1703     * The session id we are currently using.
1704     * @var array
1705     */
1706    protected $sessionid = null;
1707
1708    /**
1709     * The session data for the above session id.
1710     * @var array
1711     */
1712    protected $session = null;
1713
1714    /**
1715     * Constant used to prefix keys.
1716     */
1717    const KEY_PREFIX = 'sess_';
1718
1719    /**
1720     * This is the key used to track last access.
1721     */
1722    const LASTACCESS = '__lastaccess__';
1723
1724    /**
1725     * Override the cache::construct method.
1726     *
1727     * This function gets overriden so that we can process any invalidation events if need be.
1728     * If the definition doesn't have any invalidation events then this occurs exactly as it would for the cache class.
1729     * Otherwise we look at the last invalidation time and then check the invalidation data for events that have occured
1730     * between then now.
1731     *
1732     * You should not call this method from your code, instead you should use the cache::make methods.
1733     *
1734     * @param cache_definition $definition
1735     * @param cache_store $store
1736     * @param cache_loader|cache_data_source $loader
1737     */
1738    public function __construct(cache_definition $definition, cache_store $store, $loader = null) {
1739        // First up copy the loadeduserid to the current user id.
1740        $this->currentuserid = self::$loadeduserid;
1741        $this->set_session_id();
1742        parent::__construct($definition, $store, $loader);
1743
1744        // This will trigger check tracked user. If this gets removed a call to that will need to be added here in its place.
1745        $this->set(self::LASTACCESS, cache::now());
1746
1747        $this->handle_invalidation_events();
1748    }
1749
1750    /**
1751     * Sets the session id for the loader.
1752     */
1753    protected function set_session_id() {
1754        $this->sessionid = preg_replace('#[^a-zA-Z0-9_]#', '_', session_id());
1755    }
1756
1757    /**
1758     * Returns the prefix used for all keys.
1759     * @return string
1760     */
1761    protected function get_key_prefix() {
1762        return 'u'.$this->currentuserid.'_'.$this->sessionid;
1763    }
1764
1765    /**
1766     * Parses the key turning it into a string (or array is required) suitable to be passed to the cache store.
1767     *
1768     * This function is called for every operation that uses keys. For this reason we use this function to also check
1769     * that the current user is the same as the user who last used this cache.
1770     *
1771     * On top of that if prepends the string 'sess_' to the start of all keys. The _ ensures things are easily identifiable.
1772     *
1773     * @param string|int $key As passed to get|set|delete etc.
1774     * @return string|array String unless the store supports multi-identifiers in which case an array if returned.
1775     */
1776    protected function parse_key($key) {
1777        $prefix = $this->get_key_prefix();
1778        if ($key === self::LASTACCESS) {
1779            return $key.$prefix;
1780        }
1781        return $prefix.'_'.parent::parse_key($key);
1782    }
1783
1784    /**
1785     * Check that this cache instance is tracking the current user.
1786     */
1787    protected function check_tracked_user() {
1788        if (isset($_SESSION['USER']->id) && $_SESSION['USER']->id !== null) {
1789            // Get the id of the current user.
1790            $new = $_SESSION['USER']->id;
1791        } else {
1792            // No user set up yet.
1793            $new = 0;
1794        }
1795        if ($new !== self::$loadeduserid) {
1796            // The current user doesn't match the tracked userid for this request.
1797            if (!is_null(self::$loadeduserid)) {
1798                // Purge the data we have for the old user.
1799                // This way we don't bloat the session.
1800                $this->purge();
1801            }
1802            self::$loadeduserid = $new;
1803            $this->currentuserid = $new;
1804        } else if ($new !== $this->currentuserid) {
1805            // The current user matches the loaded user but not the user last used by this cache.
1806            $this->purge_current_user();
1807            $this->currentuserid = $new;
1808        }
1809    }
1810
1811    /**
1812     * Purges the session cache of all data belonging to the current user.
1813     */
1814    public function purge_current_user() {
1815        $keys = $this->get_store()->find_by_prefix($this->get_key_prefix());
1816        $this->get_store()->delete_many($keys);
1817    }
1818
1819    /**
1820     * Retrieves the value for the given key from the cache.
1821     *
1822     * @param string|int $key The key for the data being requested.
1823     *      It can be any structure although using a scalar string or int is recommended in the interests of performance.
1824     *      In advanced cases an array may be useful such as in situations requiring the multi-key functionality.
1825     * @param int $strictness One of IGNORE_MISSING | MUST_EXIST
1826     * @return mixed|false The data from the cache or false if the key did not exist within the cache.
1827     * @throws coding_exception
1828     */
1829    public function get($key, $strictness = IGNORE_MISSING) {
1830        // Check the tracked user.
1831        $this->check_tracked_user();
1832        // 2. Parse the key.
1833        $parsedkey = $this->parse_key($key);
1834        // 3. Get it from the store.
1835        $result = $this->get_store()->get($parsedkey);
1836        if ($result !== false) {
1837            if ($result instanceof cache_ttl_wrapper) {
1838                if ($result->has_expired()) {
1839                    $this->get_store()->delete($parsedkey);
1840                    $result = false;
1841                } else {
1842                    $result = $result->data;
1843                }
1844            }
1845            if ($result instanceof cache_cached_object) {
1846                $result = $result->restore_object();
1847            }
1848        }
1849        // 4. Load if from the loader/datasource if we don't already have it.
1850        if ($result === false) {
1851            if ($this->perfdebug) {
1852                cache_helper::record_cache_miss($this->get_store(), $this->get_definition());
1853            }
1854            if ($this->get_loader() !== false) {
1855                // We must pass the original (unparsed) key to the next loader in the chain.
1856                // The next loader will parse the key as it sees fit. It may be parsed differently
1857                // depending upon the capabilities of the store associated with the loader.
1858                $result = $this->get_loader()->get($key);
1859            } else if ($this->get_datasource() !== false) {
1860                $result = $this->get_datasource()->load_for_cache($key);
1861            }
1862            // 5. Set it to the store if we got it from the loader/datasource.
1863            if ($result !== false) {
1864                $this->set($key, $result);
1865            }
1866        } else if ($this->perfdebug) {
1867            cache_helper::record_cache_hit($this->get_store(), $this->get_definition());
1868        }
1869        // 5. Validate strictness.
1870        if ($strictness === MUST_EXIST && $result === false) {
1871            throw new coding_exception('Requested key did not exist in any cache stores and could not be loaded.');
1872        }
1873        // 6. Make sure we don't pass back anything that could be a reference.
1874        //    We don't want people modifying the data in the cache.
1875        if (!$this->get_store()->supports_dereferencing_objects() && !is_scalar($result)) {
1876            // If data is an object it will be a reference.
1877            // If data is an array if may contain references.
1878            // We want to break references so that the cache cannot be modified outside of itself.
1879            // Call the function to unreference it (in the best way possible).
1880            $result = $this->unref($result);
1881        }
1882        return $result;
1883    }
1884
1885    /**
1886     * Sends a key => value pair to the cache.
1887     *
1888     * <code>
1889     * // This code will add four entries to the cache, one for each url.
1890     * $cache->set('main', 'http://moodle.org');
1891     * $cache->set('docs', 'http://docs.moodle.org');
1892     * $cache->set('tracker', 'http://tracker.moodle.org');
1893     * $cache->set('qa', 'http://qa.moodle.net');
1894     * </code>
1895     *
1896     * @param string|int $key The key for the data being requested.
1897     *      It can be any structure although using a scalar string or int is recommended in the interests of performance.
1898     *      In advanced cases an array may be useful such as in situations requiring the multi-key functionality.
1899     * @param mixed $data The data to set against the key.
1900     * @return bool True on success, false otherwise.
1901     */
1902    public function set($key, $data) {
1903        $this->check_tracked_user();
1904        $loader = $this->get_loader();
1905        if ($loader !== false) {
1906            // We have a loader available set it there as well.
1907            // We have to let the loader do its own parsing of data as it may be unique.
1908            $loader->set($key, $data);
1909        }
1910        if ($this->perfdebug) {
1911            cache_helper::record_cache_set($this->get_store(), $this->get_definition());
1912        }
1913        if (is_object($data) && $data instanceof cacheable_object) {
1914            $data = new cache_cached_object($data);
1915        } else if (!$this->get_store()->supports_dereferencing_objects() && !is_scalar($data)) {
1916            // If data is an object it will be a reference.
1917            // If data is an array if may contain references.
1918            // We want to break references so that the cache cannot be modified outside of itself.
1919            // Call the function to unreference it (in the best way possible).
1920            $data = $this->unref($data);
1921        }
1922        // We dont' support native TTL here as we consolidate data for sessions.
1923        if ($this->has_a_ttl() && !$this->store_supports_native_ttl()) {
1924            $data = new cache_ttl_wrapper($data, $this->get_definition()->get_ttl());
1925        }
1926        return $this->get_store()->set($this->parse_key($key), $data);
1927    }
1928
1929    /**
1930     * Delete the given key from the cache.
1931     *
1932     * @param string|int $key The key to delete.
1933     * @param bool $recurse When set to true the key will also be deleted from all stacked cache loaders and their stores.
1934     *     This happens by default and ensure that all the caches are consistent. It is NOT recommended to change this.
1935     * @return bool True of success, false otherwise.
1936     */
1937    public function delete($key, $recurse = true) {
1938        $parsedkey = $this->parse_key($key);
1939        if ($recurse && $this->get_loader() !== false) {
1940            // Delete from the bottom of the stack first.
1941            $this->get_loader()->delete($key, $recurse);
1942        }
1943        return $this->get_store()->delete($parsedkey);
1944    }
1945
1946    /**
1947     * Retrieves an array of values for an array of keys.
1948     *
1949     * Using this function comes with potential performance implications.
1950     * Not all cache stores will support get_many/set_many operations and in order to replicate this functionality will call
1951     * the equivalent singular method for each item provided.
1952     * This should not deter you from using this function as there is a performance benefit in situations where the cache store
1953     * does support it, but you should be aware of this fact.
1954     *
1955     * @param array $keys The keys of the data being requested.
1956     *      Each key can be any structure although using a scalar string or int is recommended in the interests of performance.
1957     *      In advanced cases an array may be useful such as in situations requiring the multi-key functionality.
1958     * @param int $strictness One of IGNORE_MISSING or MUST_EXIST.
1959     * @return array An array of key value pairs for the items that could be retrieved from the cache.
1960     *      If MUST_EXIST was used and not all keys existed within the cache then an exception will be thrown.
1961     *      Otherwise any key that did not exist will have a data value of false within the results.
1962     * @throws coding_exception
1963     */
1964    public function get_many(array $keys, $strictness = IGNORE_MISSING) {
1965        $this->check_tracked_user();
1966        $parsedkeys = array();
1967        $keymap = array();
1968        foreach ($keys as $key) {
1969            $parsedkey = $this->parse_key($key);
1970            $parsedkeys[$key] = $parsedkey;
1971            $keymap[$parsedkey] = $key;
1972        }
1973        $result = $this->get_store()->get_many($parsedkeys);
1974        $return = array();
1975        $missingkeys = array();
1976        $hasmissingkeys = false;
1977        foreach ($result as $parsedkey => $value) {
1978            $key = $keymap[$parsedkey];
1979            if ($value instanceof cache_ttl_wrapper) {
1980                /* @var cache_ttl_wrapper $value */
1981                if ($value->has_expired()) {
1982                    $this->delete($keymap[$parsedkey]);
1983                    $value = false;
1984                } else {
1985                    $value = $value->data;
1986                }
1987            }
1988            if ($value instanceof cache_cached_object) {
1989                /* @var cache_cached_object $value */
1990                $value = $value->restore_object();
1991            } else if (!$this->get_store()->supports_dereferencing_objects() && !is_scalar($value)) {
1992                // If data is an object it will be a reference.
1993                // If data is an array if may contain references.
1994                // We want to break references so that the cache cannot be modified outside of itself.
1995                // Call the function to unreference it (in the best way possible).
1996                $value = $this->unref($value);
1997            }
1998            $return[$key] = $value;
1999            if ($value === false) {
2000                $hasmissingkeys = true;
2001                $missingkeys[$parsedkey] = $key;
2002            }
2003        }
2004        if ($hasmissingkeys) {
2005            // We've got missing keys - we've got to check any loaders or data sources.
2006            $loader = $this->get_loader();
2007            $datasource = $this->get_datasource();
2008            if ($loader !== false) {
2009                foreach ($loader->get_many($missingkeys) as $key => $value) {
2010                    if ($value !== false) {
2011                        $return[$key] = $value;
2012                        unset($missingkeys[$parsedkeys[$key]]);
2013                    }
2014                }
2015            }
2016            $hasmissingkeys = count($missingkeys) > 0;
2017            if ($datasource !== false && $hasmissingkeys) {
2018                // We're still missing keys but we've got a datasource.
2019                foreach ($datasource->load_many_for_cache($missingkeys) as $key => $value) {
2020                    if ($value !== false) {
2021                        $return[$key] = $value;
2022                        unset($missingkeys[$parsedkeys[$key]]);
2023                    }
2024                }
2025                $hasmissingkeys = count($missingkeys) > 0;
2026            }
2027        }
2028        if ($hasmissingkeys && $strictness === MUST_EXIST) {
2029            throw new coding_exception('Requested key did not exist in any cache stores and could not be loaded.');
2030        }
2031        if ($this->perfdebug) {
2032            $hits = 0;
2033            $misses = 0;
2034            foreach ($return as $value) {
2035                if ($value === false) {
2036                    $misses++;
2037                } else {
2038                    $hits++;
2039                }
2040            }
2041            cache_helper::record_cache_hit($this->get_store(), $this->get_definition(), $hits);
2042            cache_helper::record_cache_miss($this->get_store(), $this->get_definition(), $misses);
2043        }
2044        return $return;
2045
2046    }
2047
2048    /**
2049     * Delete all of the given keys from the cache.
2050     *
2051     * @param array $keys The key to delete.
2052     * @param bool $recurse When set to true the key will also be deleted from all stacked cache loaders and their stores.
2053     *     This happens by default and ensure that all the caches are consistent. It is NOT recommended to change this.
2054     * @return int The number of items successfully deleted.
2055     */
2056    public function delete_many(array $keys, $recurse = true) {
2057        $parsedkeys = array_map(array($this, 'parse_key'), $keys);
2058        if ($recurse && $this->get_loader() !== false) {
2059            // Delete from the bottom of the stack first.
2060            $this->get_loader()->delete_many($keys, $recurse);
2061        }
2062        return $this->get_store()->delete_many($parsedkeys);
2063    }
2064
2065    /**
2066     * Sends several key => value pairs to the cache.
2067     *
2068     * Using this function comes with potential performance implications.
2069     * Not all cache stores will support get_many/set_many operations and in order to replicate this functionality will call
2070     * the equivalent singular method for each item provided.
2071     * This should not deter you from using this function as there is a performance benefit in situations where the cache store
2072     * does support it, but you should be aware of this fact.
2073     *
2074     * <code>
2075     * // This code will add four entries to the cache, one for each url.
2076     * $cache->set_many(array(
2077     *     'main' => 'http://moodle.org',
2078     *     'docs' => 'http://docs.moodle.org',
2079     *     'tracker' => 'http://tracker.moodle.org',
2080     *     'qa' => ''http://qa.moodle.net'
2081     * ));
2082     * </code>
2083     *
2084     * @param array $keyvaluearray An array of key => value pairs to send to the cache.
2085     * @return int The number of items successfully set. It is up to the developer to check this matches the number of items.
2086     *      ... if they care that is.
2087     */
2088    public function set_many(array $keyvaluearray) {
2089        $this->check_tracked_user();
2090        $loader = $this->get_loader();
2091        if ($loader !== false) {
2092            // We have a loader available set it there as well.
2093            // We have to let the loader do its own parsing of data as it may be unique.
2094            $loader->set_many($keyvaluearray);
2095        }
2096        $data = array();
2097        $definitionid = $this->get_definition()->get_ttl();
2098        $simulatettl = $this->has_a_ttl() && !$this->store_supports_native_ttl();
2099        foreach ($keyvaluearray as $key => $value) {
2100            if (is_object($value) && $value instanceof cacheable_object) {
2101                $value = new cache_cached_object($value);
2102            } else if (!$this->get_store()->supports_dereferencing_objects() && !is_scalar($value)) {
2103                // If data is an object it will be a reference.
2104                // If data is an array if may contain references.
2105                // We want to break references so that the cache cannot be modified outside of itself.
2106                // Call the function to unreference it (in the best way possible).
2107                $value = $this->unref($value);
2108            }
2109            if ($simulatettl) {
2110                $value = new cache_ttl_wrapper($value, $definitionid);
2111            }
2112            $data[$key] = array(
2113                'key' => $this->parse_key($key),
2114                'value' => $value
2115            );
2116        }
2117        $successfullyset = $this->get_store()->set_many($data);
2118        if ($this->perfdebug && $successfullyset) {
2119            cache_helper::record_cache_set($this->get_store(), $this->get_definition(), $successfullyset);
2120        }
2121        return $successfullyset;
2122    }
2123
2124    /**
2125     * Purges the cache store, and loader if there is one.
2126     *
2127     * @return bool True on success, false otherwise
2128     */
2129    public function purge() {
2130        $this->get_store()->purge();
2131        if ($this->get_loader()) {
2132            $this->get_loader()->purge();
2133        }
2134        return true;
2135    }
2136
2137    /**
2138     * Test is a cache has a key.
2139     *
2140     * The use of the has methods is strongly discouraged. In a high load environment the cache may well change between the
2141     * test and any subsequent action (get, set, delete etc).
2142     * Instead it is recommended to write your code in such a way they it performs the following steps:
2143     * <ol>
2144     * <li>Attempt to retrieve the information.</li>
2145     * <li>Generate the information.</li>
2146     * <li>Attempt to set the information</li>
2147     * </ol>
2148     *
2149     * Its also worth mentioning that not all stores support key tests.
2150     * For stores that don't support key tests this functionality is mimicked by using the equivalent get method.
2151     * Just one more reason you should not use these methods unless you have a very good reason to do so.
2152     *
2153     * @param string|int $key
2154     * @param bool $tryloadifpossible If set to true, the cache doesn't contain the key, and there is another cache loader or
2155     *      data source then the code will try load the key value from the next item in the chain.
2156     * @return bool True if the cache has the requested key, false otherwise.
2157     */
2158    public function has($key, $tryloadifpossible = false) {
2159        $this->check_tracked_user();
2160        $parsedkey = $this->parse_key($key);
2161        $store = $this->get_store();
2162        if ($this->has_a_ttl() && !$this->store_supports_native_ttl()) {
2163            // The data has a TTL and the store doesn't support it natively.
2164            // We must fetch the data and expect a ttl wrapper.
2165            $data = $store->get($parsedkey);
2166            $has = ($data instanceof cache_ttl_wrapper && !$data->has_expired());
2167        } else if (!$this->store_supports_key_awareness()) {
2168            // The store doesn't support key awareness, get the data and check it manually... puke.
2169            // Either no TTL is set of the store supports its handling natively.
2170            $data = $store->get($parsedkey);
2171            $has = ($data !== false);
2172        } else {
2173            // The store supports key awareness, this is easy!
2174            // Either no TTL is set of the store supports its handling natively.
2175            /* @var cache_store|cache_is_key_aware $store */
2176            $has = $store->has($parsedkey);
2177        }
2178        if (!$has && $tryloadifpossible) {
2179            $result = null;
2180            if ($this->get_loader() !== false) {
2181                $result = $this->get_loader()->get($parsedkey);
2182            } else if ($this->get_datasource() !== null) {
2183                $result = $this->get_datasource()->load_for_cache($key);
2184            }
2185            $has = ($result !== null);
2186            if ($has) {
2187                $this->set($key, $result);
2188            }
2189        }
2190        return $has;
2191    }
2192
2193    /**
2194     * Test is a cache has all of the given keys.
2195     *
2196     * It is strongly recommended to avoid the use of this function if not absolutely required.
2197     * In a high load environment the cache may well change between the test and any subsequent action (get, set, delete etc).
2198     *
2199     * Its also worth mentioning that not all stores support key tests.
2200     * For stores that don't support key tests this functionality is mimicked by using the equivalent get method.
2201     * Just one more reason you should not use these methods unless you have a very good reason to do so.
2202     *
2203     * @param array $keys
2204     * @return bool True if the cache has all of the given keys, false otherwise.
2205     */
2206    public function has_all(array $keys) {
2207        $this->check_tracked_user();
2208        if (($this->has_a_ttl() && !$this->store_supports_native_ttl()) || !$this->store_supports_key_awareness()) {
2209            foreach ($keys as $key) {
2210                if (!$this->has($key)) {
2211                    return false;
2212                }
2213            }
2214            return true;
2215        }
2216        // The cache must be key aware and if support native ttl if it a ttl is set.
2217        /* @var cache_store|cache_is_key_aware $store */
2218        $store = $this->get_store();
2219        return $store->has_all(array_map(array($this, 'parse_key'), $keys));
2220    }
2221
2222    /**
2223     * Test if a cache has at least one of the given keys.
2224     *
2225     * It is strongly recommended to avoid the use of this function if not absolutely required.
2226     * In a high load environment the cache may well change between the test and any subsequent action (get, set, delete etc).
2227     *
2228     * Its also worth mentioning that not all stores support key tests.
2229     * For stores that don't support key tests this functionality is mimicked by using the equivalent get method.
2230     * Just one more reason you should not use these methods unless you have a very good reason to do so.
2231     *
2232     * @param array $keys
2233     * @return bool True if the cache has at least one of the given keys
2234     */
2235    public function has_any(array $keys) {
2236        if (($this->has_a_ttl() && !$this->store_supports_native_ttl()) || !$this->store_supports_key_awareness()) {
2237            foreach ($keys as $key) {
2238                if ($this->has($key)) {
2239                    return true;
2240                }
2241            }
2242            return false;
2243        }
2244        /* @var cache_store|cache_is_key_aware $store */
2245        $store = $this->get_store();
2246        return $store->has_any(array_map(array($this, 'parse_key'), $keys));
2247    }
2248
2249    /**
2250     * The session loader never uses static acceleration.
2251     * Instead it stores things in the static $session variable. Shared between all session loaders.
2252     *
2253     * @return bool
2254     */
2255    protected function use_static_acceleration() {
2256        return false;
2257    }
2258}
2259
2260/**
2261 * An request cache.
2262 *
2263 * This class is used for request caches returned by the cache::make methods.
2264 *
2265 * This cache class should never be interacted with directly. Instead you should always use the cache::make methods.
2266 * It is technically possible to call those methods through this class however there is no guarantee that you will get an
2267 * instance of this class back again.
2268 *
2269 * @internal don't use me directly.
2270 *
2271 * @package    core
2272 * @category   cache
2273 * @copyright  2012 Sam Hemelryk
2274 * @license    http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
2275 */
2276class cache_request extends cache {
2277    // This comment appeases code pre-checker ;) !
2278}
2279