1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-06 13:08:40 +01:00
psalm/src/Psalm/Provider/FileStorageCacheProvider.php
2018-11-01 16:02:44 -04:00

170 lines
4.6 KiB
PHP

<?php
namespace Psalm\Provider;
use Psalm\Config;
use Psalm\Storage\FileStorage;
class FileStorageCacheProvider
{
/**
* @var string
*/
private $modified_timestamps = '';
/**
* @var Config
*/
private $config;
const FILE_STORAGE_CACHE_DIRECTORY = 'file_cache';
public function __construct(Config $config)
{
$this->config = $config;
$storage_dir = dirname(__DIR__) . DIRECTORY_SEPARATOR . 'Storage' . DIRECTORY_SEPARATOR;
$dependent_files = [
$storage_dir . 'FileStorage.php',
$storage_dir . 'FunctionLikeStorage.php',
$storage_dir . 'ClassLikeStorage.php',
$storage_dir . 'MethodStorage.php',
$storage_dir . 'FunctionLikeParameter.php',
];
if ($config->after_visit_classlikes) {
$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 .= PSALM_VERSION . $this->config->modified_time;
}
/**
* @param string $file_path
* @param string $file_contents
*
* @return void
*/
public function writeToCache(FileStorage $storage, $file_contents)
{
$file_path = strtolower($storage->file_path);
$cache_location = $this->getCacheLocationForPath($file_path, true);
$storage->hash = $this->getCacheHash($file_path, $file_contents);
if ($this->config->use_igbinary) {
file_put_contents($cache_location, igbinary_serialize($storage));
} else {
file_put_contents($cache_location, serialize($storage));
}
}
/**
* @param string $file_path
* @param string $file_contents
*
* @return FileStorage|null
*/
public function getLatestFromCache($file_path, $file_contents)
{
$file_path = strtolower($file_path);
$cached_value = $this->loadFromCache($file_path);
if (!$cached_value) {
return null;
}
$cache_hash = $this->getCacheHash($file_path, $file_contents);
/** @psalm-suppress TypeDoesNotContainType */
if (@get_class($cached_value) === '__PHP_Incomplete_Class'
|| $cache_hash !== $cached_value->hash
) {
$this->removeCacheForFile($file_path);
return null;
}
return $cached_value;
}
/**
* @param string $file_path
*
* @return void
*/
public function removeCacheForFile($file_path)
{
$cache_path = $this->getCacheLocationForPath($file_path);
if (file_exists($cache_path)) {
unlink($cache_path);
}
}
/**
* @param string $file_path
* @param string $file_contents
*
* @return string
*/
private function getCacheHash($file_path, $file_contents)
{
return sha1(strtolower($file_path) . ' ' . $file_contents . $this->modified_timestamps);
}
/**
* @param string $file_path
*
* @return FileStorage|null
*/
private function loadFromCache($file_path)
{
$cache_location = $this->getCacheLocationForPath($file_path);
if (file_exists($cache_location)) {
if ($this->config->use_igbinary) {
/** @var FileStorage */
return igbinary_unserialize((string)file_get_contents($cache_location)) ?: null;
}
/** @var FileStorage */
return unserialize((string)file_get_contents($cache_location)) ?: null;
}
return null;
}
/**
* @param string $file_path
* @param bool $create_directory
*
* @return string
*/
private function getCacheLocationForPath($file_path, $create_directory = false)
{
$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::FILE_STORAGE_CACHE_DIRECTORY;
if ($create_directory && !is_dir($parser_cache_directory)) {
mkdir($parser_cache_directory, 0777, true);
}
return $parser_cache_directory
. DIRECTORY_SEPARATOR
. sha1($file_path)
. ($this->config->use_igbinary ? '-igbinary' : '');
}
}