file_provider = new FakeFileProvider(); $this->testConfig = $this->makeConfig(); $providers = new Providers( $this->file_provider, new FakeParserCacheProvider(), ); $this->project_analyzer = new ProjectAnalyzer( $this->testConfig, $providers, ); $this->project_analyzer->setPhpVersion('7.4', 'tests'); } public function tearDown(): void { unset($this->project_analyzer, $this->file_provider, $this->testConfig); RuntimeCaches::clearAll(); } public function addFile(string $file_path, string $contents): void { $this->file_provider->registerFile($file_path, $contents); $this->project_analyzer->getCodebase()->scanner->addFileToShallowScan($file_path); } public function addStubFile(string $file_path, string $contents): void { $this->file_provider->registerFile($file_path, $contents); $this->project_analyzer->getConfig()->addStubFile($file_path); } public function analyzeFile(string $file_path, Context $context, bool $track_unused_suppressions = true, bool $taint_flow_tracking = false): void { $codebase = $this->project_analyzer->getCodebase(); if ($taint_flow_tracking) { $this->project_analyzer->trackTaintedInputs(); } $codebase->addFilesToAnalyze([$file_path => $file_path]); $codebase->scanFiles(); $codebase->config->visitStubFiles($codebase); if ($codebase->alter_code) { $this->project_analyzer->interpretRefactors(); } $this->project_analyzer->trackUnusedSuppressions(); $file_analyzer = new FileAnalyzer( $this->project_analyzer, $file_path, $codebase->config->shortenFileName($file_path), ); $file_analyzer->analyze($context); if ($codebase->taint_flow_graph) { $codebase->taint_flow_graph->connectSinksAndSources(); } if ($track_unused_suppressions) { IssueBuffer::processUnusedSuppressions($codebase->file_provider); } } protected function getTestName(bool $withDataSet = true): string { return $this->getName($withDataSet); } public static function assertArrayKeysAreStrings(array $array, string $message = ''): void { $validKeys = array_filter($array, 'is_string', ARRAY_FILTER_USE_KEY); self::assertTrue(count($array) === count($validKeys), $message); } public static function assertArrayKeysAreZeroOrString(array $array, string $message = ''): void { $isZeroOrString = /** @param mixed $key */ static fn($key): bool => $key === 0 || is_string($key); $validKeys = array_filter($array, $isZeroOrString, ARRAY_FILTER_USE_KEY); self::assertTrue(count($array) === count($validKeys), $message); } public static function assertArrayValuesAreArrays(array $array, string $message = ''): void { $validValues = array_filter($array, 'is_array'); self::assertTrue(count($array) === count($validValues), $message); } public static function assertArrayValuesAreStrings(array $array, string $message = ''): void { $validValues = array_filter($array, 'is_string'); self::assertTrue(count($array) === count($validValues), $message); } public static function assertStringIsParsableType(string $type, string $message = ''): void { if ($type === '') { // Ignore empty types for now, as these are quite common for pecl libraries self::assertTrue(true); } else { $union = null; try { $tokens = TypeTokenizer::tokenize($type); $union = TypeParser::parseTokens($tokens); } catch (Throwable $_e) { } self::assertInstanceOf(Union::class, $union, $message); } } }