1
0
mirror of https://github.com/danog/psalm.git synced 2024-12-14 02:07:37 +01:00
psalm/src/Psalm/Internal/Provider/FileReferenceCacheProvider.php

307 lines
8.2 KiB
PHP
Raw Normal View History

<?php
2018-11-06 03:57:36 +01:00
namespace Psalm\Internal\Provider;
use Psalm\Config;
/**
* @psalm-type IssueData = array{
* severity: string,
* line_from: int,
* line_to: int,
* type: string,
* message: string,
* file_name: string,
* file_path: string,
* snippet: string,
* from: int,
* to: int,
* snippet_from: int,
* snippet_to: int,
* column_from: int,
* column_to: int
* }
*
* @psalm-type TaggedCodeType = array<int, array{0: int, 1: string}>
*/
/**
* Used to determine which files reference other files, necessary for using the --diff
* option from the command line.
*/
class FileReferenceCacheProvider
{
const REFERENCE_CACHE_NAME = 'references';
2018-11-02 02:52:39 +01:00
const ANALYZED_METHODS_CACHE_NAME = 'analyzed_methods';
const CLASS_METHOD_CACHE_NAME = 'class_method_references';
const ISSUES_CACHE_NAME = 'issues';
const FILE_MAPS_CACHE_NAME = 'file_maps';
const CONFIG_HASH_CACHE_NAME = 'config';
/**
* @var Config
*/
private $config;
/**
* @var bool
*/
public $config_changed;
public function __construct(Config $config)
{
$this->config = $config;
$this->config_changed = $config->hash !== $this->getConfigHashCache();
$this->setConfigHashCache($config->hash);
}
/**
* @return ?array
*
* @psalm-suppress MixedAssignment
* @psalm-suppress MixedTypeCoercion
*/
public function getCachedFileReferences()
{
$cache_directory = $this->config->getCacheDirectory();
if (!$cache_directory) {
return null;
}
$reference_cache_location = $cache_directory . DIRECTORY_SEPARATOR . self::REFERENCE_CACHE_NAME;
if (!is_readable($reference_cache_location)) {
return null;
}
$reference_cache = unserialize((string) file_get_contents($reference_cache_location));
if (!is_array($reference_cache)) {
throw new \UnexpectedValueException('The reference cache must be an array');
}
return $reference_cache;
}
/**
* @return ?array
*
* @psalm-suppress MixedAssignment
* @psalm-suppress MixedTypeCoercion
*/
public function getCachedMethodReferences()
{
$cache_directory = $this->config->getCacheDirectory();
if (!$cache_directory) {
return null;
}
$class_method_cache_location = $cache_directory . DIRECTORY_SEPARATOR . self::CLASS_METHOD_CACHE_NAME;
if (!is_readable($class_method_cache_location)) {
return null;
}
$class_method_reference_cache = unserialize((string) file_get_contents($class_method_cache_location));
if (!is_array($class_method_reference_cache)) {
throw new \UnexpectedValueException('The reference cache must be an array');
}
return $class_method_reference_cache;
}
/**
* @return ?array
*
* @psalm-suppress MixedAssignment
* @psalm-suppress MixedTypeCoercion
*/
public function getCachedIssues()
{
$cache_directory = $this->config->getCacheDirectory();
if (!$cache_directory) {
return null;
}
$issues_cache_location = $cache_directory . DIRECTORY_SEPARATOR . self::ISSUES_CACHE_NAME;
if (!is_readable($issues_cache_location)) {
return null;
}
$issues_cache = unserialize((string) file_get_contents($issues_cache_location));
if (!is_array($issues_cache)) {
throw new \UnexpectedValueException('The reference cache must be an array');
}
return $issues_cache;
}
/**
* @return void
*/
public function setCachedFileReferences(array $file_references)
{
$cache_directory = $this->config->getCacheDirectory();
if (!$cache_directory) {
return;
}
$reference_cache_location = $cache_directory . DIRECTORY_SEPARATOR . self::REFERENCE_CACHE_NAME;
file_put_contents($reference_cache_location, serialize($file_references));
}
/**
* @return void
*/
public function setCachedMethodReferences(array $method_references)
{
$cache_directory = $this->config->getCacheDirectory();
if (!$cache_directory) {
return;
}
$method_cache_location = $cache_directory . DIRECTORY_SEPARATOR . self::CLASS_METHOD_CACHE_NAME;
file_put_contents($method_cache_location, serialize($method_references));
}
/**
* @return void
*/
public function setCachedIssues(array $issues)
{
$cache_directory = $this->config->getCacheDirectory();
if (!$cache_directory) {
return;
}
$issues_cache_location = $cache_directory . DIRECTORY_SEPARATOR . self::ISSUES_CACHE_NAME;
file_put_contents($issues_cache_location, serialize($issues));
}
/**
* @return array<string, array<string, int>>|false
*/
2018-11-02 02:52:39 +01:00
public function getAnalyzedMethodCache()
{
2018-10-07 02:11:19 +02:00
$cache_directory = $this->config->getCacheDirectory();
2018-11-02 02:52:39 +01:00
$analyzed_methods_cache_location = $cache_directory . DIRECTORY_SEPARATOR . self::ANALYZED_METHODS_CACHE_NAME;
if ($cache_directory
2018-11-02 02:52:39 +01:00
&& file_exists($analyzed_methods_cache_location)
&& !$this->config_changed
) {
/** @var array<string, array<string, int>> */
2018-11-02 02:52:39 +01:00
return unserialize(file_get_contents($analyzed_methods_cache_location));
}
2018-10-07 02:11:19 +02:00
return false;
}
/**
2018-11-02 02:52:39 +01:00
* @param array<string, array<string, int>> $analyzed_methods
* @return void
*/
2018-11-02 02:52:39 +01:00
public function setAnalyzedMethodCache(array $analyzed_methods)
{
$cache_directory = Config::getInstance()->getCacheDirectory();
if ($cache_directory) {
2018-11-02 02:52:39 +01:00
$analyzed_methods_cache_location = $cache_directory
. DIRECTORY_SEPARATOR
. self::ANALYZED_METHODS_CACHE_NAME;
file_put_contents(
2018-11-02 02:52:39 +01:00
$analyzed_methods_cache_location,
serialize($analyzed_methods)
);
}
}
/**
* @return array<string, array{0: TaggedCodeType, 1: TaggedCodeType}>|false
*/
public function getFileMapCache()
{
$cache_directory = $this->config->getCacheDirectory();
$file_maps_cache_location = $cache_directory . DIRECTORY_SEPARATOR . self::FILE_MAPS_CACHE_NAME;
if ($cache_directory
&& file_exists($file_maps_cache_location)
&& !$this->config_changed
) {
/** @var array<string, array{0: TaggedCodeType, 1: TaggedCodeType}> */
$file_maps_cache = unserialize(file_get_contents($file_maps_cache_location));
return $file_maps_cache;
}
return false;
}
/**
* @param array<string, array{0: TaggedCodeType, 1: TaggedCodeType}> $file_maps
* @return void
*/
public function setFileMapCache(array $file_maps)
{
$cache_directory = Config::getInstance()->getCacheDirectory();
if ($cache_directory) {
$file_maps_cache_location = $cache_directory . DIRECTORY_SEPARATOR . self::FILE_MAPS_CACHE_NAME;
file_put_contents(
$file_maps_cache_location,
serialize($file_maps)
);
}
}
/**
* @return string|false
*/
public function getConfigHashCache()
{
$cache_directory = $this->config->getCacheDirectory();
$config_hash_cache_location = $cache_directory . DIRECTORY_SEPARATOR . self::CONFIG_HASH_CACHE_NAME;
if ($cache_directory
&& file_exists($config_hash_cache_location)
) {
/** @var string */
$file_maps_cache = unserialize(file_get_contents($config_hash_cache_location));
return $file_maps_cache;
}
return false;
}
/**
* @return void
*/
public function setConfigHashCache(string $hash)
{
$cache_directory = Config::getInstance()->getCacheDirectory();
if ($cache_directory) {
$config_hash_cache_location = $cache_directory . DIRECTORY_SEPARATOR . self::CONFIG_HASH_CACHE_NAME;
file_put_contents(
$config_hash_cache_location,
serialize($hash)
);
}
}
}