config = $config; $storage_dir = dirname(__DIR__, 2) . DIRECTORY_SEPARATOR . 'Storage' . DIRECTORY_SEPARATOR; $dependent_files = [ $storage_dir . 'FileStorage.php', $storage_dir . 'FunctionLikeStorage.php', $storage_dir . 'ClassLikeStorage.php', $storage_dir . 'MethodStorage.php', ]; if ($config->eventDispatcher->hasAfterClassLikeVisitHandlers()) { $dependent_files = array_merge($dependent_files, $config->plugin_paths); } foreach ($dependent_files as $dependent_file_path) { if (!file_exists($dependent_file_path)) { throw new UnexpectedValueException($dependent_file_path . ' must exist'); } $this->modified_timestamps .= ' ' . filemtime($dependent_file_path); } $this->modified_timestamps .= $this->config->computeHash(); } public function writeToCache(ClassLikeStorage $storage, string $file_path, string $file_contents): void { $fq_classlike_name_lc = strtolower($storage->name); $storage->hash = $this->getCacheHash($file_path, $file_contents); // check if we have it in cache already $cached_value = $this->loadFromCache($fq_classlike_name_lc, $file_path); if (!is_null($cached_value) && $cached_value->hash === $storage->hash) { return; } $cache_location = $this->getCacheLocationForClass($fq_classlike_name_lc, $file_path, true); if ($this->config->use_igbinary) { file_put_contents($cache_location, igbinary_serialize($storage), LOCK_EX); } else { file_put_contents($cache_location, serialize($storage), LOCK_EX); } } public function getLatestFromCache( string $fq_classlike_name_lc, ?string $file_path, ?string $file_contents ): ClassLikeStorage { $cached_value = $this->loadFromCache($fq_classlike_name_lc, $file_path); if (!$cached_value) { throw new UnexpectedValueException($fq_classlike_name_lc . ' should be in cache'); } $cache_hash = $this->getCacheHash($file_path, $file_contents); /** @psalm-suppress TypeDoesNotContainType */ if (@get_class($cached_value) === '__PHP_Incomplete_Class' || $cache_hash !== $cached_value->hash ) { unlink($this->getCacheLocationForClass($fq_classlike_name_lc, $file_path)); throw new UnexpectedValueException($fq_classlike_name_lc . ' should not be outdated'); } return $cached_value; } private function getCacheHash(?string $_unused_file_path, ?string $file_contents): string { $data = $file_contents ? $file_contents : $this->modified_timestamps; return PHP_VERSION_ID >= 8_01_00 ? hash('xxh128', $data) : hash('md4', $data); } /** * @psalm-suppress MixedAssignment */ private function loadFromCache(string $fq_classlike_name_lc, ?string $file_path): ?ClassLikeStorage { $cache_location = $this->getCacheLocationForClass($fq_classlike_name_lc, $file_path); if (file_exists($cache_location)) { if ($this->config->use_igbinary) { $storage = igbinary_unserialize(Providers::safeFileGetContents($cache_location)); if ($storage instanceof ClassLikeStorage) { return $storage; } return null; } $storage = unserialize(Providers::safeFileGetContents($cache_location)); if ($storage instanceof ClassLikeStorage) { return $storage; } return null; } return null; } private function getCacheLocationForClass( string $fq_classlike_name_lc, ?string $file_path, bool $create_directory = false ): string { $root_cache_directory = $this->config->getCacheDirectory(); if (!$root_cache_directory) { throw new UnexpectedValueException('No cache directory defined'); } $parser_cache_directory = $root_cache_directory . DIRECTORY_SEPARATOR . self::CLASS_CACHE_DIRECTORY; if ($create_directory && !is_dir($parser_cache_directory)) { try { if (mkdir($parser_cache_directory, 0777, true) === false) { // any other error than directory already exists/permissions issue throw new RuntimeException( 'Failed to create ' . $parser_cache_directory . ' cache directory for unknown reasons' ); } } catch (RuntimeException $e) { // Race condition (#4483) if (!is_dir($parser_cache_directory)) { // rethrow the error with default message // it contains the reason why creation failed throw $e; } } } $data = $file_path ? strtolower($file_path) . ' ' : ''; $data .= $fq_classlike_name_lc; $file_path_sha = PHP_VERSION_ID >= 8_01_00 ? hash('xxh128', $data) : hash('md4', $data); return $parser_cache_directory . DIRECTORY_SEPARATOR . $file_path_sha . ($this->config->use_igbinary ? '-igbinary' : ''); } }