2021-02-24 16:14:04 +01:00
|
|
|
<?php
|
2021-12-15 04:58:32 +01:00
|
|
|
|
2021-02-24 16:14:04 +01:00
|
|
|
namespace Psalm\Tests;
|
|
|
|
|
|
|
|
use PHPUnit\Framework\TestCase as BaseTestCase;
|
2021-12-03 20:11:20 +01:00
|
|
|
use PhpParser\Comment\Doc;
|
2022-06-11 15:07:24 +02:00
|
|
|
use PhpParser\Node\Scalar\String_;
|
|
|
|
use Psalm\CodeLocation;
|
2021-05-04 05:28:17 +03:00
|
|
|
use Psalm\Exception\IncorrectDocblockException;
|
2022-06-11 15:07:24 +02:00
|
|
|
use Psalm\Internal\Analyzer\FileAnalyzer;
|
|
|
|
use Psalm\Internal\Analyzer\ProjectAnalyzer;
|
2021-02-24 16:14:04 +01:00
|
|
|
use Psalm\Internal\PhpVisitor\Reflector\FunctionLikeDocblockParser;
|
2022-06-11 15:07:24 +02:00
|
|
|
use Psalm\Internal\Provider\FakeFileProvider;
|
|
|
|
use Psalm\Internal\Provider\Providers;
|
2021-06-08 05:55:21 +03:00
|
|
|
use Psalm\Internal\RuntimeCaches;
|
2022-06-11 15:07:24 +02:00
|
|
|
use Psalm\Tests\Internal\Provider\FakeParserCacheProvider;
|
2021-02-24 16:14:04 +01:00
|
|
|
|
|
|
|
class FunctionLikeDocblockParserTest extends BaseTestCase
|
|
|
|
{
|
2022-06-11 15:07:24 +02:00
|
|
|
/** @var string */
|
|
|
|
public $test_cased_function_id = 'hello_world';
|
|
|
|
|
|
|
|
/** @var CodeLocation */
|
|
|
|
public $test_code_location;
|
|
|
|
|
2021-02-24 16:14:04 +01:00
|
|
|
public function setUp(): void
|
|
|
|
{
|
|
|
|
RuntimeCaches::clearAll();
|
2022-06-11 15:07:24 +02:00
|
|
|
|
|
|
|
$file_provider = new FakeFileProvider();
|
|
|
|
|
|
|
|
$providers = new Providers(
|
|
|
|
$file_provider,
|
|
|
|
new FakeParserCacheProvider()
|
|
|
|
);
|
|
|
|
|
|
|
|
$test_config = new TestConfig();
|
|
|
|
|
|
|
|
$project_analyzer = new ProjectAnalyzer(
|
|
|
|
$test_config,
|
|
|
|
$providers
|
|
|
|
);
|
|
|
|
|
|
|
|
$file_analyzer = new FileAnalyzer($project_analyzer, 'none/none.php', 'none.php');
|
|
|
|
|
|
|
|
$stmt = new String_('randomString');
|
|
|
|
$this->test_code_location = new CodeLocation($file_analyzer, $stmt);
|
2021-02-24 16:14:04 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public function testDocblockDescription(): void
|
|
|
|
{
|
|
|
|
$doc = '/**
|
|
|
|
* Some Description
|
|
|
|
*
|
|
|
|
* @param string $bli
|
|
|
|
* @param int $bla
|
|
|
|
*
|
|
|
|
* @throws \Exception
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
';
|
2021-12-03 20:11:20 +01:00
|
|
|
$php_parser_doc = new Doc($doc);
|
2022-06-11 15:07:24 +02:00
|
|
|
$function_docblock = FunctionLikeDocblockParser::parse(
|
|
|
|
$php_parser_doc,
|
|
|
|
$this->test_code_location,
|
|
|
|
$this->test_cased_function_id
|
|
|
|
);
|
2021-02-24 16:14:04 +01:00
|
|
|
|
|
|
|
$this->assertSame('Some Description', $function_docblock->description);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testDocblockParamDescription(): void
|
|
|
|
{
|
|
|
|
$doc = '/**
|
|
|
|
* Some Description
|
|
|
|
*
|
|
|
|
* @param string $bli The BLI tag to iterate over.
|
|
|
|
* @param int $bla The blah tags
|
|
|
|
* that has a very long multiline description.
|
|
|
|
*
|
|
|
|
* @throws \Exception
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
';
|
2021-12-03 20:11:20 +01:00
|
|
|
$php_parser_doc = new Doc($doc);
|
2022-06-11 15:07:24 +02:00
|
|
|
$function_docblock = FunctionLikeDocblockParser::parse(
|
|
|
|
$php_parser_doc,
|
|
|
|
$this->test_code_location,
|
|
|
|
$this->test_cased_function_id
|
|
|
|
);
|
2021-02-24 16:14:04 +01:00
|
|
|
|
|
|
|
$this->assertTrue(isset($function_docblock->params[0]['description']));
|
|
|
|
$this->assertSame('The BLI tag to iterate over.', $function_docblock->params[0]['description']);
|
|
|
|
|
|
|
|
$this->assertTrue(isset($function_docblock->params[1]['description']));
|
|
|
|
$this->assertSame('The blah tags that has a very long multiline description.', $function_docblock->params[1]['description']);
|
|
|
|
}
|
2021-05-04 05:28:17 +03:00
|
|
|
|
|
|
|
public function testMisplacedVariableOnNextLine(): void
|
|
|
|
{
|
|
|
|
$doc = '/**
|
|
|
|
* @param
|
|
|
|
* $p
|
|
|
|
*/';
|
2021-12-03 20:11:20 +01:00
|
|
|
$php_parser_doc = new Doc($doc);
|
2021-05-04 05:28:17 +03:00
|
|
|
$this->expectException(IncorrectDocblockException::class);
|
|
|
|
$this->expectExceptionMessage('Misplaced variable');
|
2022-06-11 15:07:24 +02:00
|
|
|
FunctionLikeDocblockParser::parse(
|
|
|
|
$php_parser_doc,
|
|
|
|
$this->test_code_location,
|
|
|
|
$this->test_cased_function_id
|
|
|
|
);
|
2021-05-04 05:28:17 +03:00
|
|
|
}
|
2021-05-06 04:47:01 +03:00
|
|
|
|
|
|
|
public function testPreferPsalmPrefixedAnnotationsOverPhpstanOnes(): void
|
|
|
|
{
|
|
|
|
$doc = '/**
|
|
|
|
* @psalm-template T of string
|
|
|
|
* @phpstan-template T of int
|
|
|
|
*/
|
|
|
|
';
|
2021-12-03 20:11:20 +01:00
|
|
|
$php_parser_doc = new Doc($doc);
|
2022-06-11 15:07:24 +02:00
|
|
|
$function_docblock = FunctionLikeDocblockParser::parse(
|
|
|
|
$php_parser_doc,
|
|
|
|
$this->test_code_location,
|
|
|
|
$this->test_cased_function_id
|
|
|
|
);
|
2021-05-06 04:47:01 +03:00
|
|
|
$this->assertSame([['T', 'of', 'string', false]], $function_docblock->templates);
|
|
|
|
}
|
2021-05-28 16:47:39 +03:00
|
|
|
|
|
|
|
public function testReturnsUnexpectedTags(): void
|
|
|
|
{
|
|
|
|
$doc = '/**
|
|
|
|
* @psalm-import-type abcd
|
|
|
|
* @var int $p
|
|
|
|
*/
|
|
|
|
';
|
2021-12-03 20:11:20 +01:00
|
|
|
$php_parser_doc = new Doc($doc, 0);
|
2022-06-11 15:07:24 +02:00
|
|
|
$function_docblock = FunctionLikeDocblockParser::parse(
|
|
|
|
$php_parser_doc,
|
|
|
|
$this->test_code_location,
|
|
|
|
$this->test_cased_function_id
|
|
|
|
);
|
2021-05-28 16:47:39 +03:00
|
|
|
$this->assertEquals(
|
|
|
|
[
|
|
|
|
'psalm-import-type' => ['lines' => [1]],
|
|
|
|
'var' => ['lines' => [2], 'suggested_replacement' => 'param'],
|
|
|
|
],
|
|
|
|
$function_docblock->unexpected_tags
|
|
|
|
);
|
|
|
|
}
|
2021-02-24 16:14:04 +01:00
|
|
|
}
|