file_provider = new \Psalm\Tests\Internal\Provider\FakeFileProvider(); $config = new TestConfig(); $providers = new Providers( $this->file_provider, new \Psalm\Tests\Internal\Provider\ParserInstanceCacheProvider(), null, null, new Provider\FakeFileReferenceCacheProvider() ); $this->project_analyzer = new ProjectAnalyzer( $config, $providers, false, true, ProjectAnalyzer::TYPE_CONSOLE, 1, false ); $this->project_analyzer->getCodebase()->infer_types_from_usage = true; } /** * @dataProvider providerTestInvalidUpdates * * @param array> $file_stages * @param array $error_levels * * @return void */ public function testErrorAfterUpdate( array $file_stages, string $error_message, array $error_levels = [] ) { $this->project_analyzer->getCodebase()->diff_methods = true; $codebase = $this->project_analyzer->getCodebase(); $config = $codebase->config; foreach ($error_levels as $error_type => $error_level) { $config->setCustomErrorLevel($error_type, $error_level); } if (!$file_stages) { throw new \UnexpectedValueException('$file_stages should not be empty'); } $end_files = array_pop($file_stages); foreach ($file_stages as $files) { foreach ($files as $file_path => $contents) { $this->file_provider->registerFile($file_path, $contents); } $codebase->reloadFiles($this->project_analyzer, array_keys($files)); $codebase->analyzer->analyzeFiles($this->project_analyzer, 1, false); } foreach ($end_files as $file_path => $contents) { $this->file_provider->registerFile($file_path, $contents); } $this->expectException('\Psalm\Exception\CodeException'); $this->expectExceptionMessageRegExp('/\b' . preg_quote($error_message, '/') . '\b/'); $codebase->reloadFiles($this->project_analyzer, array_keys($end_files)); foreach ($end_files as $file_path => $_) { $codebase->addFilesToAnalyze([$file_path => $file_path]); } $codebase->analyzer->analyzeFiles($this->project_analyzer, 1, false); } /** * @return array */ public function providerTestInvalidUpdates() { return [ 'invalidateParentCaller' => [ 'file_stages' => [ [ getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' 'foo(); } }', ], [ getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' 'foo(); } }', ], ], 'error_message' => 'UndefinedMethod', ], 'invalidateAfterPropertyTypeChange' => [ 'file_stages' => [ [ getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' 'foo; } }', ], [ getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' 'foo; } }', ], ], 'error_message' => 'InvalidReturnStatement' ], 'invalidateAfterConstantChange' => [ 'file_stages' => [ [ getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' ' ' 'InvalidReturnStatement' ], 'invalidateAfterSkippedAnalysis' => [ 'file_stages' => [ [ getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' 'getB()->getString(); } }', ], [ getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' 'getB()->getString(); } public function bat() : void {} }', ], [ getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' 'getB()->getString(); } }', ], ], 'error_message' => 'NullableReturnStatement' ], 'invalidateMissingConstructorAfterPropertyChange' => [ 'file_stages' => [ [ getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' 'MissingConstructor' ], 'invalidateEmptyConstructorAfterPropertyChange' => [ 'file_stages' => [ [ getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' 'PropertyNotSetInConstructor' ], 'invalidateEmptyTraitConstructorAfterPropertyChange' => [ 'file_stages' => [ [ getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' ' ' 'PropertyNotSetInConstructor' ], 'invalidateEmptyTraitConstructorAfterTraitPropertyChange' => [ 'file_stages' => [ [ getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' ' ' 'PropertyNotSetInConstructor' ], 'invalidateSetInPrivateMethodConstructorCheck' => [ 'file_stages' => [ [ getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'setFoo(); } private function setFoo() : void { $this->foo = "bar"; } }', ], [ getcwd() . DIRECTORY_SEPARATOR . 'A.php' => 'setFoo(); } private function setFoo() : void { } }', ], ], 'error_message' => 'PropertyNotSetInConstructor' ], 'invalidateMissingConstructorAfterParentPropertyChange' => [ 'file_stages' => [ [ getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' ' ' 'MissingConstructor' ], 'invalidateNotSetInConstructorAfterParentPropertyChange' => [ 'file_stages' => [ [ getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' ' ' 'PropertyNotSetInConstructor' ], 'duplicateClass' => [ 'file_stages' => [ [ getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' 'DuplicateClass' ], 'duplicateMethod' => [ 'file_stages' => [ [ getcwd() . DIRECTORY_SEPARATOR . 'A.php' => ' ' 'DuplicateMethod' ], ]; } }