2020-05-29 04:14:41 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace Psalm\Internal\Scanner;
|
|
|
|
|
2020-07-02 23:55:57 +02:00
|
|
|
use function explode;
|
|
|
|
use function implode;
|
2020-05-29 04:14:41 +02:00
|
|
|
use function min;
|
2020-07-02 23:55:57 +02:00
|
|
|
use function preg_match;
|
|
|
|
use function preg_replace;
|
|
|
|
use function rtrim;
|
|
|
|
use function str_replace;
|
|
|
|
use function strlen;
|
|
|
|
use function strpos;
|
|
|
|
use function strspn;
|
|
|
|
use function substr;
|
|
|
|
use function trim;
|
2020-05-29 04:14:41 +02:00
|
|
|
|
2021-06-08 04:55:21 +02:00
|
|
|
use const PREG_OFFSET_CAPTURE;
|
|
|
|
|
2021-08-09 21:59:29 +02:00
|
|
|
/**
|
|
|
|
* This class will parse Docblocks in order to extract known tags from them
|
|
|
|
*/
|
2020-05-29 04:14:41 +02:00
|
|
|
class DocblockParser
|
|
|
|
{
|
2021-08-09 21:59:29 +02:00
|
|
|
/**
|
|
|
|
* $offsetStart is the absolute position of the docblock in the file. It'll be used to add to the position of some
|
|
|
|
* special tags (like `psalm-suppress`) for future uses
|
|
|
|
*/
|
2021-12-05 18:51:26 +01:00
|
|
|
public static function parse(string $docblock, int $offsetStart): ParsedDocblock
|
2020-05-29 04:14:41 +02:00
|
|
|
{
|
|
|
|
// Strip off comments.
|
|
|
|
$docblock = trim($docblock);
|
|
|
|
|
2021-09-26 22:51:44 +02:00
|
|
|
if (strpos($docblock, '/**') === 0) {
|
2020-11-05 15:29:20 +01:00
|
|
|
$docblock = substr($docblock, 3);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (substr($docblock, -2) === '*/') {
|
|
|
|
$docblock = substr($docblock, 0, -2);
|
|
|
|
|
|
|
|
if (substr($docblock, -1) === '*') {
|
|
|
|
$docblock = substr($docblock, 0, -1);
|
|
|
|
}
|
|
|
|
}
|
2020-05-29 04:14:41 +02:00
|
|
|
|
|
|
|
// Normalize multi-line @specials.
|
|
|
|
$lines = explode("\n", $docblock);
|
|
|
|
|
|
|
|
$special = [];
|
|
|
|
|
2020-08-14 22:26:55 +02:00
|
|
|
$first_line_padding = null;
|
|
|
|
|
2020-05-29 04:14:41 +02:00
|
|
|
$last = false;
|
|
|
|
foreach ($lines as $k => $line) {
|
|
|
|
if (preg_match('/^[ \t]*\*?\s*@\w/i', $line)) {
|
|
|
|
$last = $k;
|
|
|
|
} elseif (preg_match('/^\s*\r?$/', $line)) {
|
|
|
|
$last = false;
|
|
|
|
} elseif ($last !== false) {
|
|
|
|
$old_last_line = $lines[$last];
|
|
|
|
$lines[$last] = $old_last_line . "\n" . $line;
|
|
|
|
|
|
|
|
unset($lines[$k]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$line_offset = 0;
|
|
|
|
|
2020-07-02 23:55:57 +02:00
|
|
|
foreach ($lines as $k => $line) {
|
2020-05-29 04:14:41 +02:00
|
|
|
$original_line_length = strlen($line);
|
|
|
|
|
|
|
|
$line = str_replace("\r", '', $line);
|
|
|
|
|
2020-08-14 22:26:55 +02:00
|
|
|
if ($first_line_padding === null) {
|
|
|
|
$asterisk_pos = strpos($line, '*');
|
|
|
|
|
|
|
|
if ($asterisk_pos) {
|
|
|
|
$first_line_padding = substr($line, 0, $asterisk_pos - 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-03 01:39:14 +02:00
|
|
|
if (preg_match('/^[ \t]*\*?\s*@([\w\-\\\:]+)[\t ]*(.*)$/sm', $line, $matches, PREG_OFFSET_CAPTURE)) {
|
2020-05-29 04:14:41 +02:00
|
|
|
/** @var array<int, array{string, int}> $matches */
|
2021-10-04 00:03:06 +02:00
|
|
|
[, $type_info, $data_info] = $matches;
|
2020-05-29 04:14:41 +02:00
|
|
|
|
2020-09-02 06:17:41 +02:00
|
|
|
[$type] = $type_info;
|
|
|
|
[$data, $data_offset] = $data_info;
|
2020-05-29 04:14:41 +02:00
|
|
|
|
|
|
|
if (strpos($data, '*')) {
|
|
|
|
$data = rtrim(preg_replace('/^[ \t]*\*\s*$/m', '', $data));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (empty($special[$type])) {
|
|
|
|
$special[$type] = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
$data_offset += $line_offset;
|
|
|
|
|
2021-08-06 22:00:37 +02:00
|
|
|
$special[$type][$data_offset + 3 + $offsetStart] = $data;
|
2020-07-02 23:55:57 +02:00
|
|
|
|
|
|
|
unset($lines[$k]);
|
|
|
|
} else {
|
|
|
|
// Strip the leading *, if present.
|
|
|
|
$lines[$k] = str_replace("\t", ' ', $line);
|
|
|
|
$lines[$k] = preg_replace('/^ *\*/', '', $line);
|
2020-05-29 04:14:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
$line_offset += $original_line_length + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Smush the whole docblock to the left edge.
|
|
|
|
$min_indent = 80;
|
2020-07-02 23:55:57 +02:00
|
|
|
foreach ($lines as $k => $line) {
|
|
|
|
$indent = strspn($line, ' ');
|
2020-09-20 00:26:51 +02:00
|
|
|
if ($indent === strlen($line)) {
|
2020-07-02 23:55:57 +02:00
|
|
|
// This line consists of only spaces. Trim it completely.
|
|
|
|
$lines[$k] = '';
|
|
|
|
continue;
|
2020-05-29 04:14:41 +02:00
|
|
|
}
|
|
|
|
$min_indent = min($indent, $min_indent);
|
|
|
|
}
|
2020-07-02 23:55:57 +02:00
|
|
|
if ($min_indent > 0) {
|
|
|
|
foreach ($lines as $k => $line) {
|
|
|
|
if (strlen($line) < $min_indent) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
$lines[$k] = substr($line, $min_indent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$docblock = implode("\n", $lines);
|
2020-05-29 04:14:41 +02:00
|
|
|
$docblock = rtrim($docblock);
|
|
|
|
|
|
|
|
// Trim any empty lines off the front, but leave the indent level if there
|
|
|
|
// is one.
|
|
|
|
$docblock = preg_replace('/^\s*\n/', '', $docblock);
|
|
|
|
|
2020-08-14 22:26:55 +02:00
|
|
|
$parsed = new ParsedDocblock($docblock, $special, $first_line_padding ?: '');
|
2020-05-29 04:14:41 +02:00
|
|
|
|
|
|
|
self::resolveTags($parsed);
|
|
|
|
|
|
|
|
return $parsed;
|
|
|
|
}
|
|
|
|
|
2021-12-05 18:51:26 +01:00
|
|
|
private static function resolveTags(ParsedDocblock $docblock): void
|
2020-05-29 04:14:41 +02:00
|
|
|
{
|
|
|
|
if (isset($docblock->tags['template'])
|
|
|
|
|| isset($docblock->tags['psalm-template'])
|
|
|
|
|| isset($docblock->tags['phpstan-template'])
|
|
|
|
) {
|
|
|
|
$docblock->combined_tags['template']
|
|
|
|
= ($docblock->tags['template'] ?? [])
|
|
|
|
+ ($docblock->tags['phpstan-template'] ?? [])
|
|
|
|
+ ($docblock->tags['psalm-template'] ?? []);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($docblock->tags['template-covariant'])
|
|
|
|
|| isset($docblock->tags['psalm-template-covariant'])
|
|
|
|
|| isset($docblock->tags['phpstan-template-covariant'])
|
|
|
|
) {
|
|
|
|
$docblock->combined_tags['template-covariant']
|
|
|
|
= ($docblock->tags['template-covariant'] ?? [])
|
|
|
|
+ ($docblock->tags['phpstan-template-covariant'] ?? [])
|
|
|
|
+ ($docblock->tags['psalm-template-covariant'] ?? []);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($docblock->tags['template-extends'])
|
|
|
|
|| isset($docblock->tags['inherits'])
|
|
|
|
|| isset($docblock->tags['extends'])
|
|
|
|
|| isset($docblock->tags['psalm-extends'])
|
|
|
|
|| isset($docblock->tags['phpstan-extends'])
|
|
|
|
) {
|
|
|
|
$docblock->combined_tags['extends']
|
|
|
|
= ($docblock->tags['template-extends'] ?? [])
|
|
|
|
+ ($docblock->tags['inherits'] ?? [])
|
|
|
|
+ ($docblock->tags['extends'] ?? [])
|
|
|
|
+ ($docblock->tags['psalm-extends'] ?? [])
|
|
|
|
+ ($docblock->tags['phpstan-extends'] ?? []);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($docblock->tags['template-implements'])
|
|
|
|
|| isset($docblock->tags['implements'])
|
|
|
|
|| isset($docblock->tags['phpstan-implements'])
|
|
|
|
|| isset($docblock->tags['psalm-implements'])
|
|
|
|
) {
|
|
|
|
$docblock->combined_tags['implements']
|
|
|
|
= ($docblock->tags['template-implements'] ?? [])
|
|
|
|
+ ($docblock->tags['implements'] ?? [])
|
|
|
|
+ ($docblock->tags['phpstan-implements'] ?? [])
|
|
|
|
+ ($docblock->tags['psalm-implements'] ?? []);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($docblock->tags['template-use'])
|
|
|
|
|| isset($docblock->tags['use'])
|
|
|
|
|| isset($docblock->tags['phpstan-use'])
|
|
|
|
|| isset($docblock->tags['psalm-use'])
|
|
|
|
) {
|
|
|
|
$docblock->combined_tags['use']
|
|
|
|
= ($docblock->tags['template-use'] ?? [])
|
|
|
|
+ ($docblock->tags['use'] ?? [])
|
|
|
|
+ ($docblock->tags['phpstan-use'] ?? [])
|
|
|
|
+ ($docblock->tags['psalm-use'] ?? []);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($docblock->tags['method'])
|
|
|
|
|| isset($docblock->tags['psalm-method'])
|
|
|
|
) {
|
|
|
|
$docblock->combined_tags['method']
|
|
|
|
= ($docblock->tags['method'] ?? [])
|
|
|
|
+ ($docblock->tags['psalm-method'] ?? []);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($docblock->tags['return'])
|
|
|
|
|| isset($docblock->tags['psalm-return'])
|
|
|
|
|| isset($docblock->tags['phpstan-return'])
|
|
|
|
) {
|
|
|
|
if (isset($docblock->tags['psalm-return'])) {
|
|
|
|
$docblock->combined_tags['return'] = $docblock->tags['psalm-return'];
|
|
|
|
} elseif (isset($docblock->tags['phpstan-return'])) {
|
|
|
|
$docblock->combined_tags['return'] = $docblock->tags['phpstan-return'];
|
|
|
|
} else {
|
|
|
|
$docblock->combined_tags['return'] = $docblock->tags['return'];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($docblock->tags['param'])
|
|
|
|
|| isset($docblock->tags['psalm-param'])
|
|
|
|
|| isset($docblock->tags['phpstan-param'])
|
|
|
|
) {
|
|
|
|
$docblock->combined_tags['param']
|
|
|
|
= ($docblock->tags['param'] ?? [])
|
|
|
|
+ ($docblock->tags['phpstan-param'] ?? [])
|
|
|
|
+ ($docblock->tags['psalm-param'] ?? []);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($docblock->tags['var'])
|
|
|
|
|| isset($docblock->tags['psalm-var'])
|
|
|
|
|| isset($docblock->tags['phpstan-var'])
|
|
|
|
) {
|
2021-03-27 02:20:23 +01:00
|
|
|
if (!isset($docblock->tags['ignore-var'])
|
|
|
|
&& !isset($docblock->tags['psalm-ignore-var'])
|
|
|
|
) {
|
|
|
|
$docblock->combined_tags['var']
|
|
|
|
= ($docblock->tags['var'] ?? [])
|
|
|
|
+ ($docblock->tags['phpstan-var'] ?? [])
|
|
|
|
+ ($docblock->tags['psalm-var'] ?? []);
|
|
|
|
}
|
2020-05-29 04:14:41 +02:00
|
|
|
}
|
2020-11-27 23:05:26 +01:00
|
|
|
|
|
|
|
if (isset($docblock->tags['param-out'])
|
|
|
|
|| isset($docblock->tags['psalm-param-out'])
|
|
|
|
) {
|
|
|
|
$docblock->combined_tags['param-out']
|
|
|
|
= ($docblock->tags['param-out'] ?? [])
|
|
|
|
+ ($docblock->tags['psalm-param-out'] ?? []);
|
|
|
|
}
|
2020-05-29 04:14:41 +02:00
|
|
|
}
|
|
|
|
}
|