file_provider = new FakeFileProvider(); $config = new TestConfig(); $config->throw_exception = false; $providers = new Providers( $this->file_provider, new ParserInstanceCacheProvider(), null, null, new FakeFileReferenceCacheProvider(), new ProjectCacheProvider(), ); $this->codebase = new Codebase($config, $providers); $this->project_analyzer = new ProjectAnalyzer( $config, $providers, null, [], 1, null, $this->codebase, ); $this->project_analyzer->setPhpVersion('7.3', 'tests'); } /** * @dataProvider providerTestErrorFix * @param array> $file_stages * @param array> $error_positions * @param array $ignored_issues */ public function testErrorFix( array $file_stages, array $error_positions, array $ignored_issues = [], bool $test_save = true, bool $check_unused_code = false ): void { $codebase = $this->codebase; $codebase->diff_methods = true; if ($check_unused_code) { $codebase->reportUnusedCode(); } $config = $codebase->config; foreach ($ignored_issues as $error_type => $error_level) { $config->setCustomErrorLevel($error_type, $error_level); } if (!$file_stages) { throw new UnexpectedValueException('$file_stages should not be empty'); } $start_files = array_shift($file_stages); // first batch foreach ($start_files as $file_path => $contents) { $this->file_provider->registerFile($file_path, $contents); $codebase->addFilesToAnalyze([$file_path => $file_path]); } $codebase->scanFiles(); $codebase->analyzer->analyzeFiles($this->project_analyzer, 1, false, true); $data = IssueBuffer::clear(); $found_positions = []; foreach ($data as $file_issues) { foreach ($file_issues as $issue_data) { $found_positions[] = $issue_data->from; } } $this->assertSame($error_positions[0], $found_positions); $i = 0; foreach ($file_stages as $i => $file_stage) { foreach ($file_stage as $file_path => $contents) { $codebase->addTemporaryFileChanges( $file_path, $contents, ); } $codebase->reloadFiles($this->project_analyzer, array_keys($file_stage)); $codebase->analyzer->analyzeFiles($this->project_analyzer, 1, false, true); $data = IssueBuffer::clear(); $found_positions = []; foreach ($data as $file_issues) { foreach ($file_issues as $issue_data) { $found_positions[] = $issue_data->from; } } $this->assertSame($error_positions[$i + 1], $found_positions, 'stage ' . ($i + 2)); } if ($test_save && $file_stages) { $last_file_stage = end($file_stages); foreach ($last_file_stage as $file_path => $_) { $codebase->removeTemporaryFileChanges($file_path); } foreach ($last_file_stage as $file_path => $contents) { $this->file_provider->registerFile($file_path, $contents); } $codebase->reloadFiles($this->project_analyzer, array_keys($last_file_stage)); $codebase->analyzer->analyzeFiles($this->project_analyzer, 1, false, true); $data = IssueBuffer::clear(); $found_positions = []; foreach ($data as $file_issues) { foreach ($file_issues as $issue_data) { $found_positions[] = $issue_data->from; } } $this->assertSame($error_positions[count($file_stages)], $found_positions, 'stage ' . ($i + 2)); } } /** * @return array>,error_positions:array>, ignored_issues?:array, test_save?:bool, check_unused_code?: bool}> */ public function providerTestErrorFix(): array { return [ 'fixMissingColonSyntaxError' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' ' [[], [230], []], ], 'addReturnTypesToSingleMethod' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'foo(); } }', ], [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'foo(); } }', ], [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'foo(); } }', ], ], 'error_positions' => [[136, 317, 273], [323, 279], [329]], ], 'addSpaceAffectingOffsets' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'foo(); } }', ], [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'foo(); } }', ], [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'foo(); } }', ], ], 'error_positions' => [[373], [374], [375]], 'ignored_issues' => [ 'MixedAssignment' => Config::REPORT_INFO, ], ], 'fixReturnType' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' ' [[196, 144, 339, 290], [345, 296], []], 'ignored_issues' => [ 'MissingReturnType' => Config::REPORT_INFO, ], ], 'resolveNamesInDifferentFunction' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[333], []], 'ignored_issues' => [ 'InvalidDocblock' => Config::REPORT_INFO, ], ], 'bridgeStatements' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' ' [[136, 273], [279], [193, 144]], 'ignored_issues' => [ 'MissingReturnType' => Config::REPORT_INFO, ], ], 'colonReturnType' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[136, 273], [144, 136, 275]], 'ignored_issues' => [ 'MissingReturnType' => Config::REPORT_INFO, ], 'test_save' => false, ], 'noChangeJustWeirdDocblocks' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[127], [127]], ], 'removeUseShouldInvalidate' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[], [197]], ], 'removeGroupUseShouldInvalidate' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[], [197]], ], 'removeUseWithAliasShouldInvalidate' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[], [197]], ], 'removeGroupUseWithAliasShouldInvalidate' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[], [197]], ], 'removeUseShouldInvalidateNoNamespace' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[], [147]], ], 'removeGroupUseShouldInvalidateNoNamespace' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[], [197]], ], 'removeUseWithAliasShouldInvalidateNoNamespace' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[], [153]], ], 'removeGroupUseWithAliasShouldInvalidateNoNamespace' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[], [197]], ], 'addUseShouldValidate' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' ' ' [[197], []], ], 'changeUseShouldInvalidateBadNew' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[247], []], ], 'changeUseShouldInvalidateBadReturn' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[196], []], ], 'changeUseShouldInvalidateBadDocblockReturn' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[184], []], ], 'changeUseShouldInvalidateBadParam' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[192], []], ], 'changeUseShouldInvalidateBadDocblockParam' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[183], []], ], 'changeUseShouldInvalidateBadExtends' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[142], []], ], 'fixMissingProperty' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'bar; } }', ], [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'bar; } }', ], ], 'error_positions' => [[192, 192], []], ], 'traitMethodRenameDifferentFiles' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'bar(); } }', (string) getcwd() . DIRECTORY_SEPARATOR . 'T.php' => ' 'bar(); } }', (string) getcwd() . DIRECTORY_SEPARATOR . 'T.php' => ' 'bat(); } }', (string) getcwd() . DIRECTORY_SEPARATOR . 'T.php' => ' 'bat(); } }', (string) getcwd() . DIRECTORY_SEPARATOR . 'T.php' => ' 'bar(); } }', (string) getcwd() . DIRECTORY_SEPARATOR . 'T.php' => ' [[], [238, 231], [], [238, 231], []], ], 'traitMethodRenameSameFile' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'bar(); } } trait T { public function bar() : string { return "hello"; } }', ], [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'bar(); } } trait T { public function bat() : string { return "hello"; } }', ], [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'bat(); } } trait T { public function bat() : string { return "hello"; } }', ], [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'bat(); } } trait T { public function bar() : string { return "hello"; } }', ], [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'bar(); } } trait T { public function bar() : string { return "hello"; } }', ], ], 'error_positions' => [[], [238, 231], [], [238, 231], []], ], 'duplicateMethodThenRemove' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' ' [[], [381], []], ], 'classCopiesUse' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' ' [[], [122], []], ], 'addMissingArgs' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[79, 238, 280, 238], []], ], 'fixClassRef' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[306], [306, 452, 452]], ], 'addPropertyDocblock' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' */ private $bar = []; private $baz = []; public static function get() : A { return new A(); } }', ], ], 'error_positions' => [[152, 203], [337]], ], 'fixNotNullProperty' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[230], []], ], 'dontFixNotNullProperty' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[230], [230]], ], 'requiredFileWithConstructorInitialisation' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' 'foo = "hello"; } }', ], [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' 'foo = "hello"; } }', ], ], 'error_positions' => [[230], []], ], 'updatePropertyInitialization' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'foo = $foo; $this->bar = $bar; } }', ], [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'foo = $foo; $this->bar = $bar; } }', ], [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'foo = $foo; $this->bar = $bar; } }', ], ], 'error_positions' => [[], [202], []], ], 'addPartialMethodWithSyntaxError' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' ' [[], [381], []], ], 'reformat' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[], []], ], 'dontForgetErrorInTraitMethod' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'foo();', (string) getcwd() . DIRECTORY_SEPARATOR . 'T.php' => ' 'foo();', (string) getcwd() . DIRECTORY_SEPARATOR . 'T.php' => ' [[192, 192], [192, 192]], ], 'stillUnusedClass' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'src' . DIRECTORY_SEPARATOR . 'A.php' => ' ' ' ' ' ' [[84], [84]], 'ignored_issues' => [], 'test_save' => false, 'check_unused_code' => true, ], 'stillUnusedMethod' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'src' . DIRECTORY_SEPARATOR . 'A.php' => ' 'foo();', ], [ (string) getcwd() . DIRECTORY_SEPARATOR . 'src' . DIRECTORY_SEPARATOR . 'A.php' => ' 'foo();', ], ], 'error_positions' => [[201], [234]], 'ignored_issues' => [], 'test_save' => false, 'check_unused_code' => true, ], 'usedMethodWithNoAffectedConstantChanges' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'src' . DIRECTORY_SEPARATOR . 'A.php' => ' ' 'doFoo();', ], [ (string) getcwd() . DIRECTORY_SEPARATOR . 'src' . DIRECTORY_SEPARATOR . 'A.php' => ' ' 'doFoo();', ], ], 'error_positions' => [[], []], 'ignored_issues' => [], 'test_save' => false, 'check_unused_code' => true, ], 'syntaxErrorFixed' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' [[347, 452, 538], [500, 500]], ], 'updateExampleWithSyntaxErrorThen' => [ [ [ (string) getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' ' [[500, 500], [347, 452, 538], [500, 500]], ], ]; } }