filename = "{$path}/fc-{$context}-" . str_replace('/', '.', $cacheId) . '.php'; // If the file couldn't be opened or if a lock couldn't be acquired, quit if (!($fp = @fopen($this->filename, 'r')) || !flock($fp, LOCK_SH)) { $this->cache = null; return; } // Reasoning: When the include below fails, it returns "false" and we have no way to determine if it's an error or a valid cache value set_error_handler(static fn () => throw new Exception('Failed to include file')); try { $this->cache = include $this->filename; } catch (Exception) { $this->cache = null; } finally { restore_error_handler(); flock($fp, LOCK_UN); fclose($fp); } } /** * Flush the cache */ public function flush() { unset($this->cache); $this->cache = null; if (function_exists('opcache_invalidate')) { opcache_invalidate($this->filename, true); } @unlink($this->filename); } /** * Get an object from the cache. * * @param string $id */ public function getCache($id) { if (!isset($this->cache)) { return $this->cacheMiss; } return ($this->cache[$id] ?? null); } /** * Set an object in the cache. This function should be overridden * by subclasses. * * @param string $id */ public function setCache($id, $value) { // Flush the cache; it will be regenerated on demand. $this->flush(); } /** * Set the entire contents of the cache. */ public function setEntireCache($contents) { if (@file_put_contents( $this->filename, 'filename, FileManager::FILE_MODE_MASK & ~$umask); } } $this->cache = $contents; } /** * Get the time at which the data was cached. * If the file does not exist or an error occurs, null is returned. * * @return int|null */ public function getCacheTime() { $result = @filemtime($this->filename); if ($result === false) { return null; } return ((int) $result); } /** * Get the entire contents of the cache in an associative array. */ public function &getContents() { if (!isset($this->cache)) { // Trigger a cache miss to load the cache. $this->get(null); } return $this->cache; } }