2021-02-24 16:14:04 +01:00
|
|
|
<?php
|
2021-12-15 04:58:32 +01:00
|
|
|
|
2023-10-19 13:12:06 +02:00
|
|
|
declare(strict_types=1);
|
|
|
|
|
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;
|
2021-06-08 05:55:21 +03:00
|
|
|
use Psalm\Aliases;
|
2021-02-24 16:14:04 +01:00
|
|
|
use Psalm\Internal\Analyzer\CommentAnalyzer;
|
2021-06-08 05:55:21 +03:00
|
|
|
use Psalm\Internal\RuntimeCaches;
|
2021-02-24 16:14:04 +01:00
|
|
|
use Psalm\Internal\Scanner\FileScanner;
|
|
|
|
|
|
|
|
class CommentAnalyzerTest extends BaseTestCase
|
|
|
|
{
|
|
|
|
public function setUp(): void
|
|
|
|
{
|
|
|
|
RuntimeCaches::clearAll();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testDocblockVarDescription(): void
|
|
|
|
{
|
|
|
|
$doc = '/**
|
|
|
|
* @var string Some Description
|
|
|
|
*/
|
|
|
|
';
|
2021-12-03 20:11:20 +01:00
|
|
|
$php_parser_doc = new Doc($doc);
|
2021-02-24 16:14:04 +01:00
|
|
|
$comment_docblock = CommentAnalyzer::getTypeFromComment($php_parser_doc, new FileScanner('somefile.php', 'somefile.php', false), new Aliases);
|
|
|
|
$this->assertSame('Some Description', $comment_docblock[0]->description);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testDocblockVarDescriptionWithVarId(): void
|
|
|
|
{
|
|
|
|
$doc = '/**
|
|
|
|
* @var string $foo Some Description
|
|
|
|
*/
|
|
|
|
';
|
2021-12-03 20:11:20 +01:00
|
|
|
$php_parser_doc = new Doc($doc);
|
2021-02-24 16:14:04 +01:00
|
|
|
$comment_docblock = CommentAnalyzer::getTypeFromComment($php_parser_doc, new FileScanner('somefile.php', 'somefile.php', false), new Aliases);
|
|
|
|
$this->assertSame('Some Description', $comment_docblock[0]->description);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testDocblockVarDescriptionMultiline(): void
|
|
|
|
{
|
|
|
|
$doc = '/**
|
|
|
|
* @var string $foo Some Description
|
|
|
|
* with a long description.
|
|
|
|
*/
|
|
|
|
';
|
2021-12-03 20:11:20 +01:00
|
|
|
$php_parser_doc = new Doc($doc);
|
2021-02-24 16:14:04 +01:00
|
|
|
$comment_docblock = CommentAnalyzer::getTypeFromComment($php_parser_doc, new FileScanner('somefile.php', 'somefile.php', false), new Aliases);
|
|
|
|
$this->assertSame('Some Description with a long description.', $comment_docblock[0]->description);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testDocblockDescription(): void
|
|
|
|
{
|
|
|
|
$doc = '/**
|
|
|
|
* Some Description
|
|
|
|
*
|
|
|
|
* @var string
|
|
|
|
*/
|
|
|
|
';
|
2021-12-03 20:11:20 +01:00
|
|
|
$php_parser_doc = new Doc($doc);
|
2021-02-24 16:14:04 +01:00
|
|
|
$comment_docblock = CommentAnalyzer::getTypeFromComment($php_parser_doc, new FileScanner('somefile.php', 'somefile.php', false), new Aliases);
|
|
|
|
$this->assertSame('Some Description', $comment_docblock[0]->description);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testDocblockDescriptionWithVarDescription(): void
|
|
|
|
{
|
|
|
|
$doc = '/**
|
|
|
|
* Some Description
|
|
|
|
*
|
|
|
|
* @var string Use a string
|
|
|
|
*/
|
|
|
|
';
|
2021-12-03 20:11:20 +01:00
|
|
|
$php_parser_doc = new Doc($doc);
|
2021-02-24 16:14:04 +01:00
|
|
|
$comment_docblock = CommentAnalyzer::getTypeFromComment($php_parser_doc, new FileScanner('somefile.php', 'somefile.php', false), new Aliases);
|
|
|
|
$this->assertSame('Use a string', $comment_docblock[0]->description);
|
|
|
|
}
|
2023-07-06 23:40:13 +02:00
|
|
|
|
2023-07-06 00:01:00 +02:00
|
|
|
/**
|
|
|
|
* @dataProvider providerSplitDocLine
|
|
|
|
* @param string[] $expected
|
|
|
|
*/
|
|
|
|
public function testSplitDocLine(string $doc_line, array $expected): void
|
|
|
|
{
|
|
|
|
$this->assertSame($expected, CommentAnalyzer::splitDocLine($doc_line));
|
|
|
|
}
|
|
|
|
|
2023-07-07 16:25:10 +02:00
|
|
|
/**
|
2023-07-09 14:26:58 +02:00
|
|
|
* @return iterable<array-key, array{doc_line: string, expected: string[]}>
|
2023-07-07 16:25:10 +02:00
|
|
|
*/
|
2023-07-06 00:01:00 +02:00
|
|
|
public function providerSplitDocLine(): iterable
|
|
|
|
{
|
|
|
|
return [
|
|
|
|
'typeWithVar' => [
|
|
|
|
'doc_line' =>
|
|
|
|
'TArray $array',
|
|
|
|
'expected' => [
|
|
|
|
'TArray',
|
|
|
|
'$array',
|
|
|
|
],
|
|
|
|
],
|
|
|
|
'arrayShape' => [
|
|
|
|
'doc_line' =>
|
|
|
|
'array{
|
|
|
|
* a: int,
|
|
|
|
* b: string,
|
|
|
|
* }',
|
|
|
|
'expected' => [
|
|
|
|
'array{
|
|
|
|
* a: int,
|
|
|
|
* b: string,
|
|
|
|
* }',
|
2023-07-06 18:31:26 +02:00
|
|
|
],
|
2023-07-06 00:01:00 +02:00
|
|
|
],
|
|
|
|
'arrayShapeWithSpace' => [
|
|
|
|
'doc_line' =>
|
|
|
|
'array {
|
|
|
|
* a: int,
|
|
|
|
* b: string,
|
|
|
|
* }',
|
|
|
|
'expected' => [
|
|
|
|
'array {
|
|
|
|
* a: int,
|
|
|
|
* b: string,
|
|
|
|
* }',
|
2023-07-06 18:31:26 +02:00
|
|
|
],
|
2023-07-06 00:01:00 +02:00
|
|
|
],
|
2023-07-15 18:43:47 +02:00
|
|
|
'arrayShapeWithComments' => [
|
|
|
|
'doc_line' =>
|
|
|
|
'array { // Comment
|
|
|
|
* // Comment
|
|
|
|
* a: int, // Comment
|
|
|
|
* // Comment
|
|
|
|
* b: string, // Comment
|
|
|
|
* // Comment
|
|
|
|
* }',
|
|
|
|
'expected' => [
|
|
|
|
"array {
|
|
|
|
*
|
|
|
|
* a: int,
|
|
|
|
*
|
|
|
|
* b: string,
|
|
|
|
*
|
|
|
|
* }",
|
|
|
|
],
|
|
|
|
],
|
|
|
|
'arrayShapeWithSlashesInKeys' => [
|
|
|
|
'doc_line' =>
|
|
|
|
<<<EOT
|
|
|
|
array {
|
|
|
|
* // Single quote keys
|
|
|
|
* array {
|
|
|
|
* 'single_quote_key//1': int, // Comment with ' in it
|
|
|
|
* 'single_quote_key//2': int, // Comment with ' in it
|
|
|
|
* 'single_quote_key\'//\'3': int, // Comment with ' in it
|
|
|
|
* 'single_quote_key"//"4': int, // Comment with ' in it
|
|
|
|
* },
|
|
|
|
* // Double quote keys
|
|
|
|
* array {
|
|
|
|
* "double_quote_key//1": int, // Comment with " in it
|
|
|
|
* "double_quote_key//2": int, // Comment with " in it
|
|
|
|
* "double_quote_key\"//\"3": int, // Comment with " in it
|
|
|
|
* "double_quote_key'//'4": int, // Comment with " in it
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
EOT,
|
|
|
|
'expected' => [
|
|
|
|
<<<EOT
|
|
|
|
array {
|
|
|
|
*
|
|
|
|
* array {
|
|
|
|
* 'single_quote_key//1': int,
|
|
|
|
* 'single_quote_key//2': int,
|
|
|
|
* 'single_quote_key\'//\'3': int,
|
|
|
|
* 'single_quote_key"//"4': int,
|
|
|
|
* },
|
|
|
|
*
|
|
|
|
* array {
|
|
|
|
* "double_quote_key//1": int,
|
|
|
|
* "double_quote_key//2": int,
|
|
|
|
* "double_quote_key\"//\"3": int,
|
|
|
|
* "double_quote_key'//'4": int,
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
EOT,
|
|
|
|
],
|
|
|
|
],
|
2023-07-06 00:01:00 +02:00
|
|
|
'func_num_args' => [
|
|
|
|
'doc_line' =>
|
|
|
|
'(
|
|
|
|
* func_num_args() is 1
|
|
|
|
* ? array{dirname: string, basename: string, extension?: string, filename: string}
|
|
|
|
* : string
|
|
|
|
* )',
|
|
|
|
'expected' => [
|
|
|
|
'(
|
|
|
|
* func_num_args() is 1
|
|
|
|
* ? array{dirname: string, basename: string, extension?: string, filename: string}
|
|
|
|
* : string
|
2023-07-06 18:31:26 +02:00
|
|
|
* )',
|
|
|
|
],
|
2023-07-06 00:01:00 +02:00
|
|
|
],
|
|
|
|
];
|
|
|
|
}
|
2021-02-24 16:14:04 +01:00
|
|
|
}
|