AutorÃa | Ultima modificación | Ver Log |
<?php// This file is part of Moodle - http://moodle.org///// Moodle is free software: you can redistribute it and/or modify// it under the terms of the GNU General Public License as published by// the Free Software Foundation, either version 3 of the License, or// (at your option) any later version.//// Moodle is distributed in the hope that it will be useful,// but WITHOUT ANY WARRANTY; without even the implied warranty of// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the// GNU General Public License for more details.//// You should have received a copy of the GNU General Public License// along with Moodle. If not, see <http://www.gnu.org/licenses/>.namespace core_cache;use core\exception\coding_exception;use core\exception\moodle_exception;/*** An application cache.** This class is used for application caches returned by the cache::make methods.* On top of the standard functionality it also allows locking to be required and or manually operated.** This cache class should never be interacted with directly. Instead you should always use the cache::make methods.* It is technically possible to call those methods through this class however there is no guarantee that you will get an* instance of this class back again.** @internal don't use me directly.** @package core_cache* @category cache* @copyright 2012 Sam Hemelryk* @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later*/class application_cache extends cache implements loader_with_locking_interface {/*** Lock identifier.* This is used to ensure the lock belongs to the cache instance + definition + user.* @var string*/protected $lockidentifier;/*** Gets set to true if the cache's primary store natively supports locking.* If it does then we use that, otherwise we need to instantiate a second store to use for locking.* @var store*/protected $nativelocking = null;/*** Gets set to true if the cache is going to be using locking.* This isn't a requirement, it doesn't need to use locking (most won't) and this bool is used to quickly check things.* If required then locking will be forced for the get|set|delete operation.* @var bool*/protected $requirelocking = false;/*** Gets set to true if the cache writes (set|delete) must have a manual lock created first* @var bool*/protected $requirelockingbeforewrite = false;/*** Gets set to a store to use for locking if the caches primary store doesn't support locking natively.* @var lockable_cache_interface*/protected $cachelockinstance;/*** Store a list of locks acquired by this process.* @var array*/protected $locks;/*** Overrides the cache construct method.** You should not call this method from your code, instead you should use the cache::make methods.** @param definition $definition* @param store $store* @param loader_interface|data_source_interface $loader*/public function __construct(definition $definition, store $store, $loader = null) {parent::__construct($definition, $store, $loader);$this->nativelocking = $this->store_supports_native_locking();if ($definition->require_locking()) {$this->requirelocking = true;$this->requirelockingbeforewrite = $definition->require_locking_before_write();}$this->handle_invalidation_events();}/*** Returns the identifier to use** @staticvar int $instances Counts the number of instances. Used as part of the lock identifier.* @return string*/public function get_identifier() {static $instances = 0;if ($this->lockidentifier === null) {$this->lockidentifier = md5($this->get_definition()->generate_definition_hash() .sesskey() .$instances++ .application_cache::class,);}return $this->lockidentifier;}/*** Fixes the instance up after a clone.*/public function __clone() {// Force a new idenfitier.$this->lockidentifier = null;}/*** Acquires a lock on the given key.** This is done automatically if the definition requires it.* It is recommended to use a definition if you want to have locking although it is possible to do locking without having* it required by the definition.* The problem with such an approach is that you cannot ensure that code will consistently use locking. You will need to* rely on the integrators review skills.** @param string|int $key The key as given to get|set|delete* @return bool Always returns true* @throws moodle_exception If the lock cannot be obtained*/public function acquire_lock($key) {$releaseparent = false;try {if ($this->get_loader() !== false) {$this->get_loader()->acquire_lock($key);// We need to release this lock later if the lock is not successful.$releaseparent = true;}$hashedkey = helper::hash_key($key, $this->get_definition());$before = microtime(true);if ($this->nativelocking) {$lock = $this->get_store()->acquire_lock($hashedkey, $this->get_identifier());} else {$this->ensure_cachelock_available();$lock = $this->cachelockinstance->lock($hashedkey, $this->get_identifier());}$after = microtime(true);if ($lock) {$this->locks[$hashedkey] = $lock;if (MDL_PERF || $this->perfdebug) {\core\lock\timing_wrapper_lock_factory::record_lock_data($after,$before,$this->get_definition()->get_id(),$hashedkey,$lock,$this->get_identifier() . $hashedkey);}$releaseparent = false;return true;} else {throw new moodle_exception('ex_unabletolock','cache','',null,'store: ' . get_class($this->get_store()) . ', lock: ' . $hashedkey);}} finally {// Release the parent lock if we acquired it, then threw an exception.if ($releaseparent) {$this->get_loader()->release_lock($key);}}}/*** Checks if this cache has a lock on the given key.** @param string|int $key The key as given to get|set|delete* @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* someone else has the lock.*/public function check_lock_state($key) {$key = helper::hash_key($key, $this->get_definition());if (!empty($this->locks[$key])) {return true; // Shortcut to save having to make a call to the cache store if the lock is held by this process.}if ($this->nativelocking) {return $this->get_store()->check_lock_state($key, $this->get_identifier());} else {$this->ensure_cachelock_available();return $this->cachelockinstance->check_state($key, $this->get_identifier());}}/*** Releases the lock this cache has on the given key** @param string|int $key* @return bool True if the operation succeeded, false otherwise.*/public function release_lock($key) {$loaderkey = $key;$key = helper::hash_key($key, $this->get_definition());if ($this->nativelocking) {$released = $this->get_store()->release_lock($key, $this->get_identifier());} else {$this->ensure_cachelock_available();$released = $this->cachelockinstance->unlock($key, $this->get_identifier());}if ($released && array_key_exists($key, $this->locks)) {unset($this->locks[$key]);if (MDL_PERF || $this->perfdebug) {\core\lock\timing_wrapper_lock_factory::record_lock_released_data($this->get_identifier() . $key);}}if ($this->get_loader() !== false) {$this->get_loader()->release_lock($loaderkey);}return $released;}/*** Ensure that the dedicated lock store is ready to go.** This should only happen if the cache store doesn't natively support it.*/protected function ensure_cachelock_available() {if ($this->cachelockinstance === null) {$this->cachelockinstance = helper::get_cachelock_for_store($this->get_store());}}/*** Sends a key => value pair to the cache.** <code>* // This code will add four entries to the cache, one for each url.* $cache->set('main', 'http://moodle.org');* $cache->set('docs', 'http://docs.moodle.org');* $cache->set('tracker', 'http://tracker.moodle.org');* $cache->set('qa', 'http://qa.moodle.net');* </code>** @param string|int $key The key for the data being requested.* @param int $version Version number* @param mixed $data The data to set against the key.* @param bool $setparents If true, sets all parent loaders, otherwise only this one* @return bool True on success, false otherwise.* @throws coding_exception If a required lock has not beeen acquired*/protected function set_implementation($key, int $version, $data, bool $setparents = true): bool {if ($this->requirelockingbeforewrite && !$this->check_lock_state($key)) {throw new coding_exception('Attempted to set cache key "' . $key . '" without a lock. '. 'Locking before writes is required for ' . $this->get_definition()->get_id());}return parent::set_implementation($key, $version, $data, $setparents);}/*** Sends several key => value pairs to the cache.** Using this function comes with potential performance implications.* Not all cache stores will support get_many/set_many operations and in order to replicate this functionality will call* the equivalent singular method for each item provided.* This should not deter you from using this function as there is a performance benefit in situations where the cache store* does support it, but you should be aware of this fact.** <code>* // This code will add four entries to the cache, one for each url.* $cache->set_many(array(* 'main' => 'http://moodle.org',* 'docs' => 'http://docs.moodle.org',* 'tracker' => 'http://tracker.moodle.org',* 'qa' => ''http://qa.moodle.net'* ));* </code>** @param array $keyvaluearray An array of key => value pairs to send to the cache.* @return int The number of items successfully set. It is up to the developer to check this matches the number of items.* ... if they care that is.* @throws coding_exception If a required lock has not beeen acquired*/public function set_many(array $keyvaluearray) {if ($this->requirelockingbeforewrite) {foreach ($keyvaluearray as $key => $value) {if (!$this->check_lock_state($key)) {throw new coding_exception('Attempted to set cache key "' . $key . '" without a lock. '. 'Locking before writes is required for ' . $this->get_definition()->get_id());}}}return parent::set_many($keyvaluearray);}/*** Delete the given key from the cache.** @param string|int $key The key to delete.* @param bool $recurse When set to true the key will also be deleted from all stacked cache loaders and their stores.* This happens by default and ensure that all the caches are consistent. It is NOT recommended to change this.* @return bool True of success, false otherwise.* @throws coding_exception If a required lock has not beeen acquired*/public function delete($key, $recurse = true) {if ($this->requirelockingbeforewrite && !$this->check_lock_state($key)) {throw new coding_exception('Attempted to delete cache key "' . $key . '" without a lock. '. 'Locking before writes is required for ' . $this->get_definition()->get_id());}return parent::delete($key, $recurse);}/*** Delete all of the given keys from the cache.** @param array $keys The key to delete.* @param bool $recurse When set to true the key will also be deleted from all stacked cache loaders and their stores.* This happens by default and ensure that all the caches are consistent. It is NOT recommended to change this.* @return int The number of items successfully deleted.* @throws coding_exception If a required lock has not beeen acquired*/public function delete_many(array $keys, $recurse = true) {if ($this->requirelockingbeforewrite) {foreach ($keys as $key) {if (!$this->check_lock_state($key)) {throw new coding_exception('Attempted to delete cache key "' . $key . '" without a lock. '. 'Locking before writes is required for ' . $this->get_definition()->get_id());}}}return parent::delete_many($keys, $recurse);}}// Alias this class to the old name.// This file will be autoloaded by the legacyclasses autoload system.// In future all uses of this class will be corrected and the legacy references will be removed.class_alias(application_cache::class, \cache_application::class);