file_provider = new Provider\FakeFileProvider(); } /** * @param Config $config * * @return \Psalm\Internal\Analyzer\ProjectAnalyzer */ private function getProjectAnalyzerWithConfig(Config $config) { $p = new \Psalm\Internal\Analyzer\ProjectAnalyzer( $config, new \Psalm\Internal\Provider\Providers( $this->file_provider, new Provider\FakeParserCacheProvider() ) ); $p->setPhpVersion('7.3'); return $p; } /** * @return void */ public function testBarebonesConfig() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( Config::loadFromXML( (string)getcwd(), ' ' ) ); $config = $this->project_analyzer->getConfig(); $this->assertTrue($config->isInProjectDirs(realpath('src/Psalm/Type.php'))); $this->assertFalse($config->isInProjectDirs(realpath('examples/StringAnalyzer.php'))); } /** * @return void */ public function testIgnoreProjectDirectory() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( Config::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $config = $this->project_analyzer->getConfig(); $this->assertTrue($config->isInProjectDirs(realpath('src/Psalm/Type.php'))); $this->assertFalse($config->isInProjectDirs(realpath('src/Psalm/Internal/Analyzer/FileAnalyzer.php'))); $this->assertFalse($config->isInProjectDirs(realpath('examples/StringAnalyzer.php'))); } /** * @return void */ public function testIgnoreMissingProjectDirectory() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( Config::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $config = $this->project_analyzer->getConfig(); $this->assertTrue($config->isInProjectDirs(realpath('src/Psalm/Type.php'))); $this->assertFalse($config->isInProjectDirs(realpath('does/not/exist/FileAnalyzer.php'))); $this->assertFalse($config->isInProjectDirs(realpath('examples/StringAnalyzer.php'))); } /** * @return void */ public function testIgnoreSymlinkedProjectDirectory() { @unlink(dirname(__DIR__, 1) . '/fixtures/symlinktest/ignored/b'); $no_symlinking_error = 'symlink(): Cannot create symlink, error code(1314)'; $last_error = error_get_last(); $check_symlink_error = !is_array($last_error) || !isset($last_error['message']) || $no_symlinking_error !== $last_error['message']; @symlink(dirname(__DIR__, 1) . '/fixtures/symlinktest/a', dirname(__DIR__, 1) . '/fixtures/symlinktest/ignored/b'); if ($check_symlink_error) { $last_error = error_get_last(); if (is_array($last_error) && $no_symlinking_error === $last_error['message']) { $this->markTestSkipped($no_symlinking_error); return; } } $this->project_analyzer = $this->getProjectAnalyzerWithConfig( Config::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $config = $this->project_analyzer->getConfig(); $this->assertTrue($config->isInProjectDirs(realpath('tests/AnnotationTest.php'))); $this->assertFalse($config->isInProjectDirs(realpath('tests/fixtures/symlinktest/a/ignoreme.php'))); $this->assertFalse($config->isInProjectDirs(realpath('examples/StringAnalyzer.php'))); $regex = '/^unlink\([^\)]+\): (?:Permission denied|No such file or directory)$/'; $last_error = error_get_last(); $check_unlink_error = !is_array($last_error) || !preg_match($regex, $last_error['message']); @unlink(__DIR__ . '/fixtures/symlinktest/ignored/b'); if ($check_unlink_error) { $last_error = error_get_last(); if (is_array($last_error) && !preg_match($regex, $last_error['message'])) { throw new \ErrorException( $last_error['message'], 0, $last_error['type'], $last_error['file'], $last_error['line'] ); } } } /** * @return void */ public function testIgnoreWildcardProjectDirectory() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( Config::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $config = $this->project_analyzer->getConfig(); $this->assertTrue($config->isInProjectDirs(realpath('src/Psalm/Type.php'))); $this->assertFalse($config->isInProjectDirs(realpath('src/Psalm/Internal/Analyzer/FileAnalyzer.php'))); $this->assertFalse($config->isInProjectDirs(realpath('src/Psalm/Internal/Analyzer/Statements/ReturnAnalyzer.php'))); $this->assertFalse($config->isInProjectDirs(realpath('examples/StringAnalyzer.php'))); } /** * @return void */ public function testIgnoreWildcardFiles() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( Config::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $config = $this->project_analyzer->getConfig(); $this->assertTrue($config->isInProjectDirs(realpath('src/Psalm/Type.php'))); $this->assertFalse($config->isInProjectDirs(realpath('src/Psalm/Internal/Analyzer/FileAnalyzer.php'))); $this->assertTrue($config->isInProjectDirs(realpath('src/Psalm/Internal/Analyzer/Statements/ReturnAnalyzer.php'))); $this->assertFalse($config->isInProjectDirs(realpath('examples/StringAnalyzer.php'))); } /** * @return void */ public function testIgnoreWildcardFilesInWildcardFolder() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( Config::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $config = $this->project_analyzer->getConfig(); $this->assertTrue($config->isInProjectDirs(realpath('src/Psalm/Type.php'))); $this->assertTrue($config->isInProjectDirs(realpath('src/Psalm/Internal/Visitor/ReflectorVisitor.php'))); $this->assertFalse($config->isInProjectDirs(realpath('src/Psalm/Internal/Analyzer/FileAnalyzer.php'))); $this->assertFalse($config->isInProjectDirs(realpath('src/Psalm/Internal/Analyzer/Statements/ReturnAnalyzer.php'))); $this->assertTrue($config->isInProjectDirs(realpath('examples/plugins/StringChecker.php'))); } /** * @return void */ public function testIgnoreWildcardFilesInAllPossibleWildcardFolders() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( Config::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $config = $this->project_analyzer->getConfig(); $this->assertTrue($config->isInProjectDirs(realpath('src/Psalm/Type.php'))); $this->assertTrue($config->isInProjectDirs(realpath('src/Psalm/Internal/Visitor/ReflectorVisitor.php'))); $this->assertFalse($config->isInProjectDirs(realpath('src/Psalm/Internal/Analyzer/FileAnalyzer.php'))); $this->assertFalse($config->isInProjectDirs(realpath('src/Psalm/Internal/Analyzer/Statements/ReturnAnalyzer.php'))); $this->assertFalse($config->isInProjectDirs(realpath('examples/StringAnalyzer.php'))); } /** * @return void */ public function testIssueHandler() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( Config::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $config = $this->project_analyzer->getConfig(); $this->assertFalse($config->reportIssueInFile('MissingReturnType', realpath('tests/ConfigTest.php'))); $this->assertFalse($config->reportIssueInFile('MissingReturnType', realpath('src/Psalm/Type.php'))); } /** * @return void */ public function testIssueHandlerWithCustomErrorLevels() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( Config::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $config = $this->project_analyzer->getConfig(); $this->assertSame( 'info', $config->getReportingLevelForFile( 'MissingReturnType', realpath('src/Psalm/Type.php') ) ); $this->assertSame( 'error', $config->getReportingLevelForFile( 'MissingReturnType', realpath('src/Psalm/Internal/Analyzer/FileAnalyzer.php') ) ); $this->assertSame( 'error', $config->getReportingLevelForFile( 'PossiblyInvalidArgument', realpath('src/psalm.php') ) ); $this->assertSame( 'info', $config->getReportingLevelForFile( 'PossiblyInvalidArgument', realpath('examples/TemplateChecker.php') ) ); $this->assertSame( 'suppress', $config->getReportingLevelForClass( 'UndefinedClass', 'Psalm\Badger' ) ); $this->assertSame( 'suppress', $config->getReportingLevelForClass( 'UndefinedClass', 'Psalm\BadActor' ) ); $this->assertSame( 'suppress', $config->getReportingLevelForClass( 'UndefinedClass', 'Psalm\GoodActor' ) ); $this->assertSame( 'suppress', $config->getReportingLevelForClass( 'UndefinedClass', 'Psalm\MagicFactory' ) ); $this->assertNull( $config->getReportingLevelForClass( 'UndefinedClass', 'Psalm\Bodger' ) ); $this->assertSame( 'suppress', $config->getReportingLevelForMethod( 'UndefinedMethod', 'Psalm\Bodger::find1' ) ); $this->assertSame( 'suppress', $config->getReportingLevelForMethod( 'UndefinedMethod', 'Psalm\Bodger::find2' ) ); $this->assertSame( 'suppress', $config->getReportingLevelForMethod( 'UndefinedMethod', 'Psalm\Badger::find2' ) ); $this->assertNull( $config->getReportingLevelForProperty( 'UndefinedMethod', 'Psalm\Bodger::$find3' ) ); $this->assertNull( $config->getReportingLevelForProperty( 'UndefinedMethod', 'Psalm\Bodger::$find4' ) ); $this->assertSame( 'suppress', $config->getReportingLevelForMethod( 'UndefinedFunction', 'fooBar' ) ); $this->assertSame( 'suppress', $config->getReportingLevelForMethod( 'UndefinedFunction', 'foobar' ) ); } /** * @return void */ public function testAllPossibleIssues() { $all_possible_handlers = implode( ' ', array_map( /** * @param string $issue_name * * @return string */ function ($issue_name) { return '<' . $issue_name . ' errorLevel="suppress" />' . "\n"; }, \Psalm\Config\IssueHandler::getAllIssueTypes() ) ); $this->project_analyzer = $this->getProjectAnalyzerWithConfig( Config::loadFromXML( dirname(__DIR__, 2), ' ' . $all_possible_handlers . ' ' ) ); } /** * @return void */ public function testImpossibleIssue() { $this->expectExceptionMessage('This element is not expected'); $this->expectException(\Psalm\Exception\ConfigException::class); $this->project_analyzer = $this->getProjectAnalyzerWithConfig( Config::loadFromXML( dirname(__DIR__, 2), ' ' ) ); } /** * @return void */ public function testRequireVoidReturnTypeExists() { $this->expectExceptionMessage('MissingReturnType'); $this->expectException(\Psalm\Exception\CodeException::class); $this->project_analyzer = $this->getProjectAnalyzerWithConfig( TestConfig::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $file_path = getcwd() . '/src/somefile.php'; $this->addFile( $file_path, 'analyzeFile($file_path, new Context()); } /** * @return void */ public function testDoNotRequireVoidReturnTypeExists() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( TestConfig::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $file_path = getcwd() . '/src/somefile.php'; $this->addFile( $file_path, 'analyzeFile($file_path, new Context()); } /** * @return void */ public function testMethodCallMemoize() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( TestConfig::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $file_path = getcwd() . '/src/somefile.php'; $this->addFile( $file_path, 'getFoo()) { if ($a->getFoo()->getBar()) { $a->getFoo()->getBar()->bat(); } }' ); $this->analyzeFile($file_path, new Context()); } /** * @return void */ public function testThing() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( TestConfig::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $file_path = getcwd() . '/src/somefile.php'; $this->addFile( $file_path, 'foo($b = 5); echo $b;' ); $this->analyzeFile($file_path, new Context()); } /** * @return void */ public function testExitFunctions() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( TestConfig::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $file_path = getcwd() . '/src/somefile.php'; $this->addFile( $file_path, 'analyzeFile($file_path, new Context()); } /** * @return void */ public function testAllowedEchoFunction() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( TestConfig::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $file_path = getcwd() . '/src/somefile.php'; $this->addFile( $file_path, 'analyzeFile($file_path, new Context()); } /** * @return void */ public function testForbiddenEchoFunctionViaFunctions() { $this->expectExceptionMessage('ForbiddenCode'); $this->expectException(\Psalm\Exception\CodeException::class); $this->project_analyzer = $this->getProjectAnalyzerWithConfig( TestConfig::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $file_path = getcwd() . '/src/somefile.php'; $this->addFile( $file_path, 'analyzeFile($file_path, new Context()); } /** * @return void */ public function testForbiddenEchoFunctionViaFlag() { $this->expectExceptionMessage('ForbiddenEcho'); $this->expectException(\Psalm\Exception\CodeException::class); $this->project_analyzer = $this->getProjectAnalyzerWithConfig( TestConfig::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $file_path = getcwd() . '/src/somefile.php'; $this->addFile( $file_path, 'analyzeFile($file_path, new Context()); } /** * @return void */ public function testAllowedPrintFunction() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( TestConfig::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $file_path = getcwd() . '/src/somefile.php'; $this->addFile( $file_path, 'analyzeFile($file_path, new Context()); } /** * @return void */ public function testForbiddenPrintFunction() { $this->expectExceptionMessage('ForbiddenCode'); $this->expectException(\Psalm\Exception\CodeException::class); $this->project_analyzer = $this->getProjectAnalyzerWithConfig( TestConfig::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $file_path = getcwd() . '/src/somefile.php'; $this->addFile( $file_path, 'analyzeFile($file_path, new Context()); } /** * @return void */ public function testAllowedVarExportFunction() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( TestConfig::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $file_path = getcwd() . '/src/somefile.php'; $this->addFile( $file_path, 'analyzeFile($file_path, new Context()); } /** * @return void */ public function testForbiddenVarExportFunction() { $this->expectExceptionMessage('ForbiddenCode'); $this->expectException(\Psalm\Exception\CodeException::class); $this->project_analyzer = $this->getProjectAnalyzerWithConfig( TestConfig::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $file_path = getcwd() . '/src/somefile.php'; $this->addFile( $file_path, 'analyzeFile($file_path, new Context()); } /** * @return void */ public function testValidThrowInvalidCatch() { $this->expectExceptionMessage('InvalidCatch'); $this->expectException(\Psalm\Exception\CodeException::class); $this->project_analyzer = $this->getProjectAnalyzerWithConfig( TestConfig::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $file_path = getcwd() . '/src/somefile.php'; $this->addFile( $file_path, 'analyzeFile($file_path, new Context()); } /** * @return void */ public function testInvalidThrowValidCatch() { $this->expectExceptionMessage('InvalidThrow'); $this->expectException(\Psalm\Exception\CodeException::class); $this->project_analyzer = $this->getProjectAnalyzerWithConfig( TestConfig::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $file_path = getcwd() . '/src/somefile.php'; $this->addFile( $file_path, 'analyzeFile($file_path, new Context()); } /** * @return void */ public function testValidThrowValidCatch() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( TestConfig::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $file_path = getcwd() . '/src/somefile.php'; $this->addFile( $file_path, 'analyzeFile($file_path, new Context()); } /** * @return void */ public function testTemplatedFiles() { foreach (['1.xml', '2.xml', '3.xml', '4.xml', '5.xml', '6.xml', '7.xml', '8.xml'] as $file_name) { $project_root = dirname(__DIR__, 2); $file_path = realpath($project_root . '/assets/config_levels/' . $file_name); symlink($file_path, $project_root . DIRECTORY_SEPARATOR . $file_name); Config::loadFromXMLFile( $project_root . DIRECTORY_SEPARATOR . $file_name, $project_root ); } } /** @return void */ public function testModularConfig() { $root = __DIR__ . '/../fixtures/ModularConfig'; $config = Config::loadFromXMLFile($root . '/psalm.xml', $root); $this->assertEquals( [ realpath($root . '/Bar.php'), realpath($root . '/Bat.php') ], $config->getProjectFiles() ); } public function tearDown(): void { parent::tearDown(); if ($this->getName() == 'testTemplatedFiles') { $project_root = dirname(__DIR__, 2); foreach (['1.xml', '2.xml', '3.xml', '4.xml', '5.xml', '6.xml', '7.xml', '8.xml'] as $file_name) { @unlink($project_root . DIRECTORY_SEPARATOR . $file_name); } } } /** * @return void */ public function testGlobals() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( TestConfig::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $file_path = getcwd() . '/src/somefile.php'; $this->addFile( $file_path, 'func(); ord($_GET["str"]); assert($glob4 !== null); ord($glob4); function example1(): void { global $glob1, $glob2, $glob3, $glob4; ord($glob1); ord($glob2["str"]); $glob3->func(); ord($glob4); ord($_GET["str"]); } $glob1 = 0; error_reporting($glob1); $_GET["str"] = 0; error_reporting($_GET["str"]); function example2(): void { global $glob1, $glob2, $glob3; error_reporting($glob1); ord($glob2["str"]); $glob3->func(); ord($_GET["str"]); } } namespace ns { ord($glob1); ord($glob2["str"]); $glob3->func(); ord($_GET["str"]); class Clazz { public function func(): void {} } function example3(): void { global $glob1, $glob2, $glob3; ord($glob1); ord($glob2["str"]); $glob3->func(); ord($_GET["str"]); } } namespace ns2 { /** @psalm-suppress InvalidGlobal */ global $glob1, $glob2, $glob3; ord($glob1); ord($glob2["str"]); $glob3->func(); } namespace { ord($glob1 ?: "str"); ord($_GET["str"] ?? "str"); function example4(): void { global $glob1; ord($glob1 ?: "str"); ord($_GET["str"] ?? "str"); } }' ); $this->analyzeFile($file_path, new Context()); } /** * @return void */ public function testIgnoreExceptions() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( TestConfig::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $file_path = getcwd() . '/src/somefile.php'; $this->addFile( $file_path, 'analyzeFile($file_path, new Context()); } public function testNotIgnoredException() : void { $this->expectException(\Psalm\Exception\CodeException::class); $this->expectExceptionMessage('MissingThrowsDocblock'); $this->project_analyzer = $this->getProjectAnalyzerWithConfig( TestConfig::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $file_path = getcwd() . '/src/somefile.php'; $this->addFile( $file_path, 'analyzeFile($file_path, new Context()); } /** * @return void */ public function testGetPossiblePsr4Path() { $this->project_analyzer = $this->getProjectAnalyzerWithConfig( Config::loadFromXML( dirname(__DIR__, 2), ' ' ) ); $config = $this->project_analyzer->getConfig(); $classloader = new \Composer\Autoload\ClassLoader(); $classloader->addPsr4( 'Psalm\\', [ dirname(__DIR__, 2) . DIRECTORY_SEPARATOR . 'vendor' . DIRECTORY_SEPARATOR . 'composer' . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . 'src' . DIRECTORY_SEPARATOR . 'Psalm', ] ); $classloader->addPsr4( 'Psalm\\Tests\\', [ dirname(__DIR__, 2) . DIRECTORY_SEPARATOR . 'vendor' . DIRECTORY_SEPARATOR . 'composer' . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . '..' . DIRECTORY_SEPARATOR . 'tests', ] ); $config->setComposerClassLoader($classloader); $this->assertSame( dirname(__DIR__, 2) . DIRECTORY_SEPARATOR . 'src' . DIRECTORY_SEPARATOR . 'Psalm' . DIRECTORY_SEPARATOR . 'Foo.php', $config->getPotentialComposerFilePathForClassLike('Psalm\\Foo') ); $this->assertSame( dirname(__DIR__, 2) . DIRECTORY_SEPARATOR . 'tests' . DIRECTORY_SEPARATOR . 'Foo.php', $config->getPotentialComposerFilePathForClassLike('Psalm\\Tests\\Foo') ); } /** * @return void */ public function testTakesPhpVersionFromConfigFile() { $cfg = Config::loadFromXML( dirname(__DIR__, 2), '' ); $this->assertSame('7.1', $cfg->getPhpVersion()); } /** * @return void */ public function testReadsComposerJsonForPhpVersion() { $root = __DIR__ . '/../fixtures/ComposerPhpVersion'; $cfg = Config::loadFromXML($root, ""); $this->assertSame('7.2', $cfg->getPhpVersion()); $cfg = Config::loadFromXML($root, ""); $this->assertSame('8.0', $cfg->getPhpVersion()); } }