1
0
mirror of https://github.com/danog/psalm.git synced 2025-01-09 14:38:37 +01:00
psalm/src/Psalm/Internal/Codebase/Scanner.php
Daniil Gentili 1986c8b4a8
Add support for strict arrays, fix type alias intersection, fix array_is_list assertion on non-lists (#8395)
* Immutable CodeLocation

* Remove excess clones

* Remove external clones

* Remove leftover clones

* Fix final clone issue

* Immutable storages

* Refactoring

* Fixes

* Fixes

* Fix

* Fix

* Fixes

* Simplify

* Fixes

* Fix

* Fixes

* Update

* Fix

* Cache global types

* Fix

* Update

* Update

* Fixes

* Fixes

* Refactor

* Fixes

* Fix

* Fix

* More caching

* Fix

* Fix

* Update

* Update

* Fix

* Fixes

* Update

* Refactor

* Update

* Fixes

* Break one more test

* Fix

* FIx

* Fix

* Fix

* Fix

* Fix

* Improve performance and readability

* Equivalent logic

* Fixes

* Revert

* Revert "Revert"

This reverts commit f9175100c8452c80559234200663fd4c4f4dd889.

* Fix

* Fix reference bug

* Make default TypeVisitor immutable

* Bugfix

* Remove clones

* Partial refactoring

* Refactoring

* Fixes

* Fix

* Fixes

* Fixes

* cs-fix

* Fix final bugs

* Add test

* Misc fixes

* Update

* Fixes

* Experiment with removing different property

* revert "Experiment with removing different property"

This reverts commit ac1156e077fc4ea633530d51096d27b6e88bfdf9.

* Uniform naming

* Uniform naming

* Hack hotfix

* Clean up $_FILES ref #8621

* Undo hack, try fixing properly

* Helper method

* Remove redundant call

* Partially fix bugs

* Cleanup

* Change defaults

* Fix bug

* Fix (?, hope this doesn't break anything else)

* cs-fix

* Review fixes

* Bugfix

* Bugfix

* Improve logic

* Add support for list{} and callable-list{} types, properly implement array_is_list assertions (fixes #8389)

* Default to sealed arrays

* Fix array_merge bug

* Fixes

* Fix

* Sealed type checks

* Properly infer properties-of and get_object_vars on final classes

* Fix array_map zipping

* Fix tests

* Fixes

* Fixes

* Fix more stuff

* Recursively resolve type aliases

* Fix typo

* Fixes

* Fix array_is_list assertion on keyed array

* Add BC docs

* Fixes

* fix

* Update

* Update

* Update

* Update

* Seal arrays with count assertions

* Fix #8528

* Fix

* Update

* Improve sealed array foreach logic

* get_object_vars on template properties

* Fix sealed array assertion reconciler logic

* Improved reconciler

* Add tests

* Single source of truth for test types

* Fix tests

* Fixup tests

* Fixup tests

* Fixup tests

* Update

* Fix tests

* Fix tests

* Final fixes

* Fixes

* Use list syntax only when needed

* Fix tests

* Cs-fix

* Update docs

* Update docs

* Update docs

* Update docs

* Update docs

* Document missing types

* Update docs

* Improve class-string-map docs

* Update

* Update

* I love working on psalm :)

* Keep arrays unsealed by default

* Fixup tests

* Fix syntax mistake

* cs-fix

* Fix typo

* Re-import missing types

* Keep strict types only in return types

* argc/argv fixes

* argc/argv fixes

* Fix test

* Comment-out valinor code, pinging @romm pls merge https://github.com/CuyZ/Valinor/pull/246 so we can add valinor to the psalm docs :)
2022-11-05 22:34:42 +01:00

815 lines
27 KiB
PHP

<?php
namespace Psalm\Internal\Codebase;
use Closure;
use Psalm\Codebase;
use Psalm\CodeLocation;
use Psalm\Config;
use Psalm\Internal\Analyzer\IssueData;
use Psalm\Internal\Analyzer\ProjectAnalyzer;
use Psalm\Internal\ErrorHandler;
use Psalm\Internal\Fork\Pool;
use Psalm\Internal\Provider\ClassLikeStorageProvider;
use Psalm\Internal\Provider\FileProvider;
use Psalm\Internal\Provider\FileReferenceProvider;
use Psalm\Internal\Provider\FileStorageProvider;
use Psalm\Internal\Scanner\FileScanner;
use Psalm\IssueBuffer;
use Psalm\Progress\Progress;
use Psalm\Storage\ClassLikeStorage;
use Psalm\Storage\FileStorage;
use Psalm\Type;
use ReflectionClass;
use Throwable;
use UnexpectedValueException;
use function array_filter;
use function array_merge;
use function array_pop;
use function ceil;
use function count;
use function error_reporting;
use function explode;
use function file_exists;
use function min;
use function realpath;
use function strtolower;
use function substr;
use const DIRECTORY_SEPARATOR;
use const PHP_EOL;
/**
* @psalm-type ThreadData = strict-array{
* array<string, string>,
* array<string, string>,
* array<string, string>,
* array<string, bool>,
* array<string, bool>,
* array<string, string>,
* array<string, bool>,
* array<string, bool>,
* array<string, bool>
* }
*
* @psalm-type PoolData = strict-array{
* classlikes_data:strict-array{
* array<lowercase-string, bool>,
* array<lowercase-string, bool>,
* array<lowercase-string, bool>,
* array<string, bool>,
* array<lowercase-string, bool>,
* array<string, bool>,
* array<lowercase-string, bool>,
* array<string, bool>,
* array<string, bool>
* },
* scanner_data: ThreadData,
* issues:array<string, list<IssueData>>,
* changed_members:array<string, array<string, bool>>,
* unchanged_signature_members:array<string, array<string, bool>>,
* diff_map:array<string, array<int, strict-array{int, int, int, int}>>,
* deletion_ranges:array<string, array<int, strict-array{int, int}>>,
* errors:array<string, bool>,
* classlike_storage:array<string, ClassLikeStorage>,
* file_storage:array<lowercase-string, FileStorage>,
* new_file_content_hashes: array<string, string>,
* taint_data: ?TaintFlowGraph
* }
*/
/**
* @internal
*
* Contains methods that aid in the scanning of Psalm's codebase
*/
class Scanner
{
/**
* @var Codebase
*/
private $codebase;
/**
* @var array<string, string>
*/
private $classlike_files = [];
/**
* @var array<string, bool>
*/
private $deep_scanned_classlike_files = [];
/**
* @var array<string, string>
*/
private $files_to_scan = [];
/**
* @var array<string, string>
*/
private $classes_to_scan = [];
/**
* @var array<string, bool>
*/
private $classes_to_deep_scan = [];
/**
* @var array<string, string>
*/
private $files_to_deep_scan = [];
/**
* @var array<string, bool>
*/
private $scanned_files = [];
/**
* @var array<string, bool>
*/
private $store_scan_failure = [];
/**
* @var array<string, bool>
*/
private $reflected_classlikes_lc = [];
/**
* @var Reflection
*/
private $reflection;
/**
* @var Config
*/
private $config;
/**
* @var Progress
*/
private $progress;
/**
* @var FileStorageProvider
*/
private $file_storage_provider;
/**
* @var FileProvider
*/
private $file_provider;
/**
* @var FileReferenceProvider
*/
private $file_reference_provider;
/**
* @var bool
*/
private $is_forked = false;
public function __construct(
Codebase $codebase,
Config $config,
FileStorageProvider $file_storage_provider,
FileProvider $file_provider,
Reflection $reflection,
FileReferenceProvider $file_reference_provider,
Progress $progress
) {
$this->codebase = $codebase;
$this->reflection = $reflection;
$this->file_provider = $file_provider;
$this->progress = $progress;
$this->file_storage_provider = $file_storage_provider;
$this->config = $config;
$this->file_reference_provider = $file_reference_provider;
}
/**
* @param array<string, string> $files_to_scan
*
*/
public function addFilesToShallowScan(array $files_to_scan): void
{
$this->files_to_scan += $files_to_scan;
}
/**
* @param array<string, string> $files_to_scan
*/
public function addFilesToDeepScan(array $files_to_scan): void
{
$this->files_to_scan += $files_to_scan;
$this->files_to_deep_scan += $files_to_scan;
}
public function addFileToShallowScan(string $file_path): void
{
$this->files_to_scan[$file_path] = $file_path;
}
public function addFileToDeepScan(string $file_path): void
{
$this->files_to_scan[$file_path] = $file_path;
$this->files_to_deep_scan[$file_path] = $file_path;
}
public function removeFile(string $file_path): void
{
unset($this->scanned_files[$file_path]);
}
public function removeClassLike(string $fq_classlike_name_lc): void
{
unset(
$this->classlike_files[$fq_classlike_name_lc],
$this->deep_scanned_classlike_files[$fq_classlike_name_lc]
);
}
public function setClassLikeFilePath(string $fq_classlike_name_lc, string $file_path): void
{
$this->classlike_files[$fq_classlike_name_lc] = $file_path;
}
public function getClassLikeFilePath(string $fq_classlike_name_lc): string
{
if (!isset($this->classlike_files[$fq_classlike_name_lc])) {
throw new UnexpectedValueException('Could not find file for ' . $fq_classlike_name_lc);
}
return $this->classlike_files[$fq_classlike_name_lc];
}
/**
* @param array<string, mixed> $phantom_classes
*/
public function queueClassLikeForScanning(
string $fq_classlike_name,
bool $analyze_too = false,
bool $store_failure = true,
array $phantom_classes = []
): void {
if ($fq_classlike_name[0] === '\\') {
$fq_classlike_name = substr($fq_classlike_name, 1);
}
$fq_classlike_name_lc = strtolower($fq_classlike_name);
if ($fq_classlike_name_lc === 'static') {
return;
}
// avoid checking classes that we know will just end in failure
if ($fq_classlike_name_lc === 'null' || substr($fq_classlike_name_lc, -5) === '\null') {
return;
}
if (!isset($this->classlike_files[$fq_classlike_name_lc])
|| ($analyze_too && !isset($this->deep_scanned_classlike_files[$fq_classlike_name_lc]))
) {
if (!isset($this->classes_to_scan[$fq_classlike_name_lc]) || $store_failure) {
$this->classes_to_scan[$fq_classlike_name_lc] = $fq_classlike_name;
}
if ($analyze_too) {
$this->classes_to_deep_scan[$fq_classlike_name_lc] = true;
}
$this->store_scan_failure[$fq_classlike_name] = $store_failure;
if (PropertyMap::inPropertyMap($fq_classlike_name_lc)) {
$public_mapped_properties = PropertyMap::getPropertyMap()[$fq_classlike_name_lc];
foreach ($public_mapped_properties as $public_mapped_property) {
$property_type = Type::parseString($public_mapped_property);
/** @psalm-suppress UnusedMethodCall */
$property_type->queueClassLikesForScanning(
$this->codebase,
null,
$phantom_classes + [$fq_classlike_name_lc => true]
);
}
}
}
}
public function scanFiles(ClassLikes $classlikes, int $pool_size = 1): bool
{
$has_changes = false;
while ($this->files_to_scan || $this->classes_to_scan) {
if ($this->files_to_scan) {
if ($this->scanFilePaths($pool_size)) {
$has_changes = true;
}
} else {
$this->convertClassesToFilePaths($classlikes);
}
}
return $has_changes;
}
private function shouldScan(string $file_path): bool
{
return $this->file_provider->fileExists($file_path)
&& (!isset($this->scanned_files[$file_path])
|| (isset($this->files_to_deep_scan[$file_path]) && !$this->scanned_files[$file_path]));
}
private function scanFilePaths(int $pool_size): bool
{
$files_to_scan = array_filter(
$this->files_to_scan,
[$this, 'shouldScan']
);
$this->files_to_scan = [];
if (!$files_to_scan) {
return false;
}
if (!$this->is_forked && $pool_size > 1 && count($files_to_scan) > 512) {
$pool_size = ceil(min($pool_size, count($files_to_scan) / 256));
} else {
$pool_size = 1;
}
if ($pool_size > 1) {
$process_file_paths = [];
$i = 0;
foreach ($files_to_scan as $file_path) {
$process_file_paths[$i % $pool_size][] = $file_path;
++$i;
}
$this->progress->debug('Forking process for scanning' . PHP_EOL);
// Run scanning one file at a time, splitting the set of
// files up among a given number of child processes.
$pool = new Pool(
$this->config,
$process_file_paths,
function (): void {
$this->progress->debug('Initialising forked process for scanning' . PHP_EOL);
$project_analyzer = ProjectAnalyzer::getInstance();
$codebase = $project_analyzer->getCodebase();
$statements_provider = $codebase->statements_provider;
$codebase->scanner->isForked();
FileStorageProvider::deleteAll();
ClassLikeStorageProvider::deleteAll();
$statements_provider->resetDiffs();
$this->progress->debug('Have initialised forked process for scanning' . PHP_EOL);
},
Closure::fromCallable([$this, 'scanAPath']),
/**
* @return PoolData
*/
function () {
$this->progress->debug('Collecting data from forked scanner process' . PHP_EOL);
$project_analyzer = ProjectAnalyzer::getInstance();
$codebase = $project_analyzer->getCodebase();
$statements_provider = $codebase->statements_provider;
return [
'classlikes_data' => $codebase->classlikes->getThreadData(),
'scanner_data' => $codebase->scanner->getThreadData(),
'issues' => IssueBuffer::getIssuesData(),
'changed_members' => $statements_provider->getChangedMembers(),
'unchanged_signature_members' => $statements_provider->getUnchangedSignatureMembers(),
'diff_map' => $statements_provider->getDiffMap(),
'deletion_ranges' => $statements_provider->getDeletionRanges(),
'errors' => $statements_provider->getErrors(),
'classlike_storage' => $codebase->classlike_storage_provider->getAll(),
'file_storage' => $codebase->file_storage_provider->getAll(),
'new_file_content_hashes' => $statements_provider->parser_cache_provider
? $statements_provider->parser_cache_provider->getNewFileContentHashes()
: [],
'taint_data' => $codebase->taint_flow_graph,
];
}
);
// Wait for all tasks to complete and collect the results.
/**
* @var array<int, PoolData>
*/
$forked_pool_data = $pool->wait();
foreach ($forked_pool_data as $pool_data) {
IssueBuffer::addIssues($pool_data['issues']);
$this->codebase->statements_provider->addChangedMembers(
$pool_data['changed_members']
);
$this->codebase->statements_provider->addUnchangedSignatureMembers(
$pool_data['unchanged_signature_members']
);
$this->codebase->statements_provider->addDiffMap(
$pool_data['diff_map']
);
$this->codebase->statements_provider->addDeletionRanges(
$pool_data['deletion_ranges']
);
$this->codebase->statements_provider->addErrors($pool_data['errors']);
if ($this->codebase->taint_flow_graph && $pool_data['taint_data']) {
$this->codebase->taint_flow_graph->addGraph($pool_data['taint_data']);
}
$this->codebase->file_storage_provider->addMore($pool_data['file_storage']);
$this->codebase->classlike_storage_provider->addMore($pool_data['classlike_storage']);
$this->codebase->classlikes->addThreadData($pool_data['classlikes_data']);
$this->addThreadData($pool_data['scanner_data']);
if ($this->codebase->statements_provider->parser_cache_provider) {
$this->codebase->statements_provider->parser_cache_provider->addNewFileContentHashes(
$pool_data['new_file_content_hashes']
);
}
}
if ($pool->didHaveError()) {
exit(1);
}
} else {
$i = 0;
foreach ($files_to_scan as $file_path => $_) {
$this->scanAPath($i, $file_path);
++$i;
}
}
if ($this->codebase->statements_provider->parser_cache_provider) {
$this->codebase->statements_provider->parser_cache_provider->saveFileContentHashes();
}
foreach ($files_to_scan as $scanned_file) {
if ($this->config->hasStubFile($scanned_file)) {
$file_storage = $this->file_storage_provider->get($scanned_file);
foreach ($file_storage->functions as $function_storage) {
if ($function_storage->cased_name
&& !$this->codebase->functions->hasStubbedFunction($function_storage->cased_name)
) {
$this->codebase->functions->addGlobalFunction(
$function_storage->cased_name,
$function_storage
);
}
}
foreach ($file_storage->constants as $name => $type) {
$this->codebase->addGlobalConstantType($name, $type);
}
}
}
$this->file_reference_provider->addClassLikeFiles($this->classlike_files);
return true;
}
private function convertClassesToFilePaths(ClassLikes $classlikes): void
{
$classes_to_scan = $this->classes_to_scan;
$this->classes_to_scan = [];
foreach ($classes_to_scan as $fq_classlike_name) {
$fq_classlike_name_lc = strtolower($fq_classlike_name);
if (isset($this->reflected_classlikes_lc[$fq_classlike_name_lc])) {
continue;
}
if ($classlikes->isMissingClassLike($fq_classlike_name_lc)) {
continue;
}
if (!isset($this->classlike_files[$fq_classlike_name_lc])) {
if ($classlikes->doesClassLikeExist($fq_classlike_name_lc)) {
if ($fq_classlike_name_lc === 'self') {
continue;
}
$this->progress->debug('Using reflection to get metadata for ' . $fq_classlike_name . "\n");
/** @psalm-suppress ArgumentTypeCoercion */
$reflected_class = new ReflectionClass($fq_classlike_name);
$this->reflection->registerClass($reflected_class);
$this->reflected_classlikes_lc[$fq_classlike_name_lc] = true;
} elseif ($this->fileExistsForClassLike($classlikes, $fq_classlike_name)) {
$fq_classlike_name_lc = strtolower($classlikes->getUnAliasedName(
$fq_classlike_name_lc
));
// even though we've checked this above, calling the method invalidates it
if (isset($this->classlike_files[$fq_classlike_name_lc])) {
$file_path = $this->classlike_files[$fq_classlike_name_lc];
$this->files_to_scan[$file_path] = $file_path;
if (isset($this->classes_to_deep_scan[$fq_classlike_name_lc])) {
unset($this->classes_to_deep_scan[$fq_classlike_name_lc]);
$this->files_to_deep_scan[$file_path] = $file_path;
}
}
} elseif ($this->store_scan_failure[$fq_classlike_name]) {
$classlikes->registerMissingClassLike($fq_classlike_name_lc);
}
} elseif (isset($this->classes_to_deep_scan[$fq_classlike_name_lc])
&& !isset($this->deep_scanned_classlike_files[$fq_classlike_name_lc])
) {
$file_path = $this->classlike_files[$fq_classlike_name_lc];
$this->files_to_scan[$file_path] = $file_path;
unset($this->classes_to_deep_scan[$fq_classlike_name_lc]);
$this->files_to_deep_scan[$file_path] = $file_path;
$this->deep_scanned_classlike_files[$fq_classlike_name_lc] = true;
}
}
}
/**
* @param array<string, class-string<FileScanner>> $filetype_scanners
*/
private function scanFile(
string $file_path,
array $filetype_scanners,
bool $will_analyze = false
): void {
$file_scanner = $this->getScannerForPath($file_path, $filetype_scanners, $will_analyze);
if (isset($this->scanned_files[$file_path])
&& (!$will_analyze || $this->scanned_files[$file_path])
) {
throw new UnexpectedValueException('Should not be rescanning ' . $file_path);
}
if (!$this->file_provider->fileExists($file_path) && $this->config->mustBeIgnored($file_path)) {
// this should not happen, but might if the file was temporary
return;
}
$file_contents = $this->file_provider->getContents($file_path);
$from_cache = $this->file_storage_provider->has($file_path, $file_contents);
if (!$from_cache) {
$this->file_storage_provider->create($file_path);
}
$this->scanned_files[$file_path] = $will_analyze;
$file_storage = $this->file_storage_provider->get($file_path);
$file_scanner->scan(
$this->codebase,
$file_storage,
$from_cache,
$this->progress
);
if (!$from_cache) {
if (!$file_storage->has_visitor_issues && $this->file_storage_provider->cache) {
$this->file_storage_provider->cache->writeToCache($file_storage, $file_contents);
}
} else {
$this->codebase->statements_provider->setUnchangedFile($file_path);
foreach ($file_storage->required_file_paths as $required_file_path) {
if ($will_analyze) {
$this->addFileToDeepScan($required_file_path);
} else {
$this->addFileToShallowScan($required_file_path);
}
}
foreach ($file_storage->classlikes_in_file as $fq_classlike_name) {
$this->codebase->exhumeClassLikeStorage($fq_classlike_name, $file_path);
}
foreach ($file_storage->required_classes as $fq_classlike_name) {
$this->queueClassLikeForScanning($fq_classlike_name, $will_analyze, false);
}
foreach ($file_storage->required_interfaces as $fq_classlike_name) {
$this->queueClassLikeForScanning($fq_classlike_name, false, false);
}
foreach ($file_storage->referenced_classlikes as $fq_classlike_name) {
$this->queueClassLikeForScanning($fq_classlike_name, false, false);
}
if ($this->codebase->register_autoload_files) {
foreach ($file_storage->functions as $function_storage) {
if ($function_storage->cased_name
&& !$this->codebase->functions->hasStubbedFunction($function_storage->cased_name)
) {
$this->codebase->functions->addGlobalFunction(
$function_storage->cased_name,
$function_storage
);
}
}
foreach ($file_storage->constants as $name => $type) {
$this->codebase->addGlobalConstantType($name, $type);
}
}
foreach ($file_storage->classlike_aliases as $aliased_name => $unaliased_name) {
$this->codebase->classlikes->addClassAlias($unaliased_name, $aliased_name);
}
}
}
/**
* @param array<string, class-string<FileScanner>> $filetype_scanners
*/
private function getScannerForPath(
string $file_path,
array $filetype_scanners,
bool $will_analyze = false
): FileScanner {
$path_parts = explode(DIRECTORY_SEPARATOR, $file_path);
$file_name_parts = explode('.', array_pop($path_parts));
$extension = count($file_name_parts) > 1 ? array_pop($file_name_parts) : null;
$file_name = $this->config->shortenFileName($file_path);
if (isset($filetype_scanners[$extension])) {
return new $filetype_scanners[$extension]($file_path, $file_name, $will_analyze);
}
return new FileScanner($file_path, $file_name, $will_analyze);
}
/**
* @return array<string, bool>
*/
public function getScannedFiles(): array
{
return $this->scanned_files;
}
/**
* Checks whether a class exists, and if it does then records what file it's in
* for later checking
*/
private function fileExistsForClassLike(ClassLikes $classlikes, string $fq_class_name): bool
{
$fq_class_name_lc = strtolower($fq_class_name);
if (isset($this->classlike_files[$fq_class_name_lc])) {
return true;
}
if ($fq_class_name === 'self') {
return false;
}
$composer_file_path = $this->config->getComposerFilePathForClassLike($fq_class_name);
if ($composer_file_path && file_exists($composer_file_path)) {
$this->progress->debug('Using composer to locate file for ' . $fq_class_name . "\n");
$classlikes->addFullyQualifiedClassLikeName(
$fq_class_name_lc,
realpath($composer_file_path)
);
return true;
}
$reflected_class = ErrorHandler::runWithExceptionsSuppressed(
function () use ($fq_class_name): ?ReflectionClass {
$old_level = error_reporting();
$this->progress->setErrorReporting();
try {
$this->progress->debug('Using reflection to locate file for ' . $fq_class_name . "\n");
/** @psalm-suppress ArgumentTypeCoercion */
return new ReflectionClass($fq_class_name);
} catch (Throwable $e) {
// do not cache any results here (as case-sensitive filenames can screw things up)
return null;
} finally {
error_reporting($old_level);
}
}
);
if (null === $reflected_class) {
return false;
}
$file_path = (string)$reflected_class->getFileName();
// if the file was autoloaded but exists in evaled code only, return false
if (!file_exists($file_path)) {
return false;
}
$new_fq_class_name = $reflected_class->getName();
$new_fq_class_name_lc = strtolower($new_fq_class_name);
if ($new_fq_class_name_lc !== $fq_class_name_lc) {
$classlikes->addClassAlias($new_fq_class_name, $fq_class_name);
$fq_class_name_lc = $new_fq_class_name_lc;
}
$fq_class_name = $new_fq_class_name;
$classlikes->addFullyQualifiedClassLikeName($fq_class_name_lc);
if ($reflected_class->isInterface()) {
$classlikes->addFullyQualifiedInterfaceName($fq_class_name, $file_path);
} elseif ($reflected_class->isTrait()) {
$classlikes->addFullyQualifiedTraitName($fq_class_name, $file_path);
} else {
$classlikes->addFullyQualifiedClassName($fq_class_name, $file_path);
}
return true;
}
/**
* @return ThreadData
*/
public function getThreadData(): array
{
return [
$this->files_to_scan,
$this->files_to_deep_scan,
$this->classes_to_scan,
$this->classes_to_deep_scan,
$this->store_scan_failure,
$this->classlike_files,
$this->deep_scanned_classlike_files,
$this->scanned_files,
$this->reflected_classlikes_lc,
];
}
/**
* @param ThreadData $thread_data
*
*/
public function addThreadData(array $thread_data): void
{
[
$files_to_scan,
$files_to_deep_scan,
$classes_to_scan,
$classes_to_deep_scan,
$store_scan_failure,
$classlike_files,
$deep_scanned_classlike_files,
$scanned_files,
$reflected_classlikes_lc
] = $thread_data;
$this->files_to_scan = array_merge($files_to_scan, $this->files_to_scan);
$this->files_to_deep_scan = array_merge($files_to_deep_scan, $this->files_to_deep_scan);
$this->classes_to_scan = array_merge($classes_to_scan, $this->classes_to_scan);
$this->classes_to_deep_scan = array_merge($classes_to_deep_scan, $this->classes_to_deep_scan);
$this->store_scan_failure = array_merge($store_scan_failure, $this->store_scan_failure);
$this->classlike_files = array_merge($classlike_files, $this->classlike_files);
$this->deep_scanned_classlike_files = array_merge(
$deep_scanned_classlike_files,
$this->deep_scanned_classlike_files
);
$this->scanned_files = array_merge($scanned_files, $this->scanned_files);
$this->reflected_classlikes_lc = array_merge($reflected_classlikes_lc, $this->reflected_classlikes_lc);
}
public function isForked(): void
{
$this->is_forked = true;
}
private function scanAPath(int $_, string $file_path): void
{
$this->scanFile(
$file_path,
$this->config->getFiletypeScanners(),
isset($this->files_to_deep_scan[$file_path])
);
}
}