2017-07-25 22:11:02 +02:00
|
|
|
<?php
|
|
|
|
namespace Psalm\Provider;
|
|
|
|
|
|
|
|
use PhpParser;
|
|
|
|
|
|
|
|
class StatementsProvider
|
|
|
|
{
|
2018-01-21 19:38:51 +01:00
|
|
|
/**
|
|
|
|
* @var FileProvider
|
|
|
|
*/
|
|
|
|
private $file_provider;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var ParserCacheProvider
|
|
|
|
*/
|
|
|
|
private $cache_provider;
|
|
|
|
|
2018-03-12 05:01:21 +01:00
|
|
|
/**
|
|
|
|
* @var int
|
|
|
|
*/
|
|
|
|
private $this_modified_time;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @var FileStorageCacheProvider
|
|
|
|
*/
|
|
|
|
private $file_storage_cache_provider;
|
|
|
|
|
2018-01-21 19:38:51 +01:00
|
|
|
/**
|
|
|
|
* @var PhpParser\Parser|null
|
|
|
|
*/
|
2017-07-25 22:11:02 +02:00
|
|
|
protected static $parser;
|
|
|
|
|
2018-03-12 05:01:21 +01:00
|
|
|
public function __construct(
|
|
|
|
FileProvider $file_provider,
|
|
|
|
ParserCacheProvider $cache_provider,
|
|
|
|
FileStorageCacheProvider $file_storage_cache_provider
|
|
|
|
) {
|
2018-01-21 19:38:51 +01:00
|
|
|
$this->file_provider = $file_provider;
|
|
|
|
$this->cache_provider = $cache_provider;
|
2018-03-12 05:01:21 +01:00
|
|
|
$this->this_modified_time = filemtime(__FILE__);
|
|
|
|
$this->file_storage_cache_provider = $file_storage_cache_provider;
|
2018-01-21 19:38:51 +01:00
|
|
|
}
|
|
|
|
|
2017-07-25 22:11:02 +02:00
|
|
|
/**
|
|
|
|
* @param string $file_path
|
|
|
|
* @param bool $debug_output
|
|
|
|
*
|
|
|
|
* @return array<int, \PhpParser\Node\Stmt>
|
|
|
|
*/
|
2018-01-21 19:38:51 +01:00
|
|
|
public function getStatementsForFile($file_path, $debug_output = false)
|
|
|
|
{
|
2017-07-25 22:11:02 +02:00
|
|
|
$from_cache = false;
|
|
|
|
|
2018-04-24 13:07:04 +02:00
|
|
|
$version = (string) PHP_PARSER_VERSION . $this->this_modified_time;
|
2017-07-25 22:11:02 +02:00
|
|
|
|
2018-01-21 19:38:51 +01:00
|
|
|
$file_contents = $this->file_provider->getContents($file_path);
|
|
|
|
$modified_time = $this->file_provider->getModifiedTime($file_path);
|
2017-07-25 22:11:02 +02:00
|
|
|
|
|
|
|
$file_content_hash = md5($version . $file_contents);
|
2018-02-19 17:14:07 +01:00
|
|
|
$file_cache_key = $this->cache_provider->getParserCacheKey($file_path, $this->cache_provider->use_igbinary);
|
2017-07-25 22:11:02 +02:00
|
|
|
|
2018-01-21 19:38:51 +01:00
|
|
|
$stmts = $this->cache_provider->loadStatementsFromCache(
|
2017-07-25 22:11:02 +02:00
|
|
|
$modified_time,
|
|
|
|
$file_content_hash,
|
|
|
|
$file_cache_key
|
|
|
|
);
|
|
|
|
|
|
|
|
if ($stmts === null) {
|
|
|
|
if ($debug_output) {
|
2018-04-13 01:42:24 +02:00
|
|
|
echo 'Parsing ' . $file_path . "\n";
|
2017-07-25 22:11:02 +02:00
|
|
|
}
|
|
|
|
|
2018-04-22 04:13:10 +02:00
|
|
|
$stmts = self::parseStatements($file_contents);
|
2018-03-12 05:01:21 +01:00
|
|
|
$this->file_storage_cache_provider->removeCacheForFile($file_path);
|
2017-07-25 22:11:02 +02:00
|
|
|
} else {
|
|
|
|
$from_cache = true;
|
|
|
|
}
|
|
|
|
|
2018-07-13 03:25:06 +02:00
|
|
|
$nameResolver = new \Psalm\Visitor\SimpleNameResolver;
|
|
|
|
$nodeTraverser = new PhpParser\NodeTraverser;
|
|
|
|
$nodeTraverser->addVisitor($nameResolver);
|
|
|
|
|
|
|
|
/** @var array<int, \PhpParser\Node\Stmt> */
|
|
|
|
$stmts = $nodeTraverser->traverse($stmts);
|
|
|
|
|
2018-01-21 19:38:51 +01:00
|
|
|
$this->cache_provider->saveStatementsToCache($file_cache_key, $file_content_hash, $stmts, $from_cache);
|
2017-07-25 22:11:02 +02:00
|
|
|
|
|
|
|
if (!$stmts) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
return $stmts;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $file_contents
|
|
|
|
*
|
|
|
|
* @return array<int, \PhpParser\Node\Stmt>
|
|
|
|
*/
|
2018-04-22 04:13:10 +02:00
|
|
|
public static function parseStatements($file_contents)
|
2017-07-25 22:11:02 +02:00
|
|
|
{
|
|
|
|
if (!self::$parser) {
|
2018-04-24 13:08:14 +02:00
|
|
|
$lexer = new PhpParser\Lexer([
|
|
|
|
'usedAttributes' => [
|
|
|
|
'comments', 'startLine', 'startFilePos', 'endFilePos',
|
|
|
|
],
|
|
|
|
]);
|
2017-07-25 22:11:02 +02:00
|
|
|
|
|
|
|
self::$parser = (new PhpParser\ParserFactory())->create(PhpParser\ParserFactory::PREFER_PHP7, $lexer);
|
|
|
|
}
|
|
|
|
|
|
|
|
$error_handler = new \PhpParser\ErrorHandler\Collecting();
|
|
|
|
|
|
|
|
/** @var array<int, \PhpParser\Node\Stmt> */
|
|
|
|
$stmts = self::$parser->parse($file_contents, $error_handler);
|
|
|
|
|
|
|
|
if (!$stmts && $error_handler->hasErrors()) {
|
|
|
|
foreach ($error_handler->getErrors() as $error) {
|
|
|
|
throw $error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $stmts;
|
|
|
|
}
|
|
|
|
}
|