> */ private $unchanged_members = []; /** * @var array> */ private $unchanged_signature_members = []; /** * @var array> */ private $changed_members = []; /** * @var array> */ private $diff_map = []; /** * @var PhpParser\Parser|null */ protected static $parser; /** * @var PhpParser\NodeTraverser|null */ protected static $node_traverser; public function __construct( FileProvider $file_provider, ParserCacheProvider $parser_cache_provider = null, FileStorageCacheProvider $file_storage_cache_provider = null ) { $this->file_provider = $file_provider; $this->parser_cache_provider = $parser_cache_provider; $this->this_modified_time = filemtime(__FILE__); $this->file_storage_cache_provider = $file_storage_cache_provider; } /** * @param string $file_path * @param bool $debug_output * * @return array */ public function getStatementsForFile($file_path, $debug_output = false) { $from_cache = false; $version = (string) PHP_PARSER_VERSION . $this->this_modified_time; $file_contents = $this->file_provider->getContents($file_path); $modified_time = $this->file_provider->getModifiedTime($file_path); if (!$this->parser_cache_provider) { if ($debug_output) { echo 'Parsing ' . $file_path . "\n"; } return self::parseStatements($file_contents, $file_path) ?: []; } $file_content_hash = md5($version . $file_contents); $stmts = $this->parser_cache_provider->loadStatementsFromCache( $file_path, $modified_time, $file_content_hash ); if ($stmts === null) { if ($debug_output) { echo 'Parsing ' . $file_path . "\n"; } $stmts = self::parseStatements($file_contents, $file_path); $existing_file_contents = $this->parser_cache_provider->loadExistingFileContentsFromCache($file_path); if ($existing_file_contents) { $existing_statements = $this->parser_cache_provider->loadExistingStatementsFromCache($file_path); if ($existing_statements) { list($unchanged_members, $unchanged_signature_members, $changed_members, $diff_map) = \Psalm\Diff\FileStatementsDiffer::diff( $existing_statements, $stmts, $existing_file_contents, $file_contents ); $unchanged_members = array_map( /** * @param int $_ * @return bool */ function ($_) { return true; }, array_flip($unchanged_members) ); $unchanged_signature_members = array_map( /** * @param int $_ * @return bool */ function ($_) { return true; }, array_flip($unchanged_signature_members) ); $changed_members = array_map( /** * @param int $_ * @return bool */ function ($_) { return true; }, array_flip($changed_members) ); if (isset($this->unchanged_members[$file_path])) { $this->unchanged_members[$file_path] = array_intersect_key( $this->unchanged_members[$file_path], $unchanged_members ); } else { $this->unchanged_members[$file_path] = $unchanged_members; } if (isset($this->unchanged_signature_members[$file_path])) { $this->unchanged_signature_members[$file_path] = array_intersect_key( $this->unchanged_signature_members[$file_path], $unchanged_signature_members ); } else { $this->unchanged_signature_members[$file_path] = $unchanged_signature_members; } if (isset($this->changed_members[$file_path])) { $this->changed_members[$file_path] = array_merge( $this->changed_members[$file_path], $changed_members ); } else { $this->changed_members[$file_path] = $changed_members; } $this->diff_map[$file_path] = $diff_map; } } if ($this->file_storage_cache_provider) { $this->file_storage_cache_provider->removeCacheForFile($file_path); } $this->parser_cache_provider->cacheFileContents($file_path, $file_contents); } else { $from_cache = true; $this->diff_map[$file_path] = []; } $this->parser_cache_provider->saveStatementsToCache($file_path, $file_content_hash, $stmts, $from_cache); if (!$stmts) { return []; } return $stmts; } /** * @return array> */ public function getChangedMembers() { return $this->changed_members; } /** * @param array> $more_changed_members * @return void */ public function addChangedMembers(array $more_changed_members) { $this->changed_members = array_merge($more_changed_members, $this->changed_members); } /** * @return array> */ public function getUnchangedSignatureMembers() { return $this->unchanged_signature_members; } /** * @param array> $more_unchanged_members * @return void */ public function addUnchangedSignatureMembers(array $more_unchanged_members) { $this->unchanged_signature_members = array_merge($more_unchanged_members, $this->unchanged_signature_members); } /** * @param string $file_path * @return void */ public function setUnchangedFile($file_path) { if (!isset($this->diff_map[$file_path])) { $this->diff_map[$file_path] = []; } } /** * @return array> */ public function getDiffMap() { return $this->diff_map; } /** * @param array> $diff_map * @return void */ public function addDiffMap(array $diff_map) { $this->diff_map = array_merge($diff_map, $this->diff_map); } /** * @return void */ public function resetDiffs() { $this->changed_members = []; $this->unchanged_members = []; $this->unchanged_signature_members = []; $this->diff_map = []; } /** * @param string $file_contents * @param string $file_path * * @return array */ public static function parseStatements($file_contents, $file_path = null) { if (!self::$parser) { $lexer = new PhpParser\Lexer([ 'usedAttributes' => [ 'comments', 'startLine', 'startFilePos', 'endFilePos', ], ]); self::$parser = (new PhpParser\ParserFactory())->create(PhpParser\ParserFactory::PREFER_PHP7, $lexer); } if (!self::$node_traverser) { self::$node_traverser = new PhpParser\NodeTraverser; $name_resolver = new \Psalm\Visitor\SimpleNameResolver; self::$node_traverser->addVisitor($name_resolver); } $error_handler = new \PhpParser\ErrorHandler\Collecting(); /** @var array */ $stmts = self::$parser->parse($file_contents, $error_handler) ?: []; if ($error_handler->hasErrors() && $file_path) { $config = \Psalm\Config::getInstance(); foreach ($error_handler->getErrors() as $error) { if ($error->hasColumnInfo()) { \Psalm\IssueBuffer::add( new \Psalm\Issue\ParseError( $error->getMessage(), new \Psalm\CodeLocation\ParseErrorLocation( $error, $file_contents, $file_path, $config->shortenFileName($file_path) ) ) ); } } } /** @var array */ self::$node_traverser->traverse($stmts); return $stmts; } }