diff --git a/.gitignore b/.gitignore index 4d66ffe..dcb42db 100644 --- a/.gitignore +++ b/.gitignore @@ -1,3 +1,4 @@ /vendor/ /composer.lock /test.php +/.idea/ diff --git a/src/TodoByRule.php b/src/TodoByRule.php index b662649..72c5824 100644 --- a/src/TodoByRule.php +++ b/src/TodoByRule.php @@ -3,16 +3,32 @@ namespace staabm\PHPStanTodoBy; use PhpParser\Node; +use PHPStan\Analyser\Scope; use PHPStan\Node\VirtualNode; use PHPStan\Rules\Rule; use PHPStan\Rules\RuleErrorBuilder; +use function preg_match_all; +use function strtotime; +use function substr_count; +use function time; +use function trim; +use const PREG_OFFSET_CAPTURE; +use const PREG_SET_ORDER; /** * @implements Rule */ final class TodoByRule implements Rule { - private const PATTERN = '/^TODO:?\s*([0-9]{4}-[0-9]{2}-[0-9]{2}):?(.*)$/'; + private const PATTERN = <<<'REGEXP' +/ +@?TODO # possible leading @ +\s*[:-]?\s* # optional colon or hyphen +(?P\d{4}-\d{2}-\d{2}) # date consisting of YYYY-MM-DD format +\s*[:-]?\s* # optional colon or hyphen +(?P.*) # rest of line as comment text +/ix +REGEXP; private int $now; @@ -26,7 +42,7 @@ public function getNodeType(): string return Node::class; } - public function processNode(\PhpParser\Node $node, \PHPStan\Analyser\Scope $scope): array + public function processNode(Node $node, Scope $scope): array { if ( $node instanceof VirtualNode @@ -36,35 +52,58 @@ public function processNode(\PhpParser\Node $node, \PHPStan\Analyser\Scope $scop return []; } - $comments = $node->getComments(); - if (count($comments) === 0) { - return []; - } - $errors = []; - foreach($comments as $comment) { - $text = ltrim($comment->getText(), "\t /"); - if (!str_starts_with($text, 'TODO')) { - continue; - } - if (preg_match(self::PATTERN, $text, $matches) !== 1) { - continue; - } + foreach ($node->getComments() as $comment) { - $date = $matches[1]; - $todoText = trim($matches[2]); - if (strtotime($date) > $this->now) { + $text = $comment->getText(); + + /** + * PHP doc comments have the entire multi-line comment as the text. + * Since this could potentially contain multiple "todo" comments, we need to check all lines. + * This works for single line comments as well. + * + * PREG_OFFSET_CAPTURE: Track where each "todo" comment starts within the whole comment text. + * PREG_SET_ORDER: Make each value of $matches be structured the same as if from preg_match(). + */ + if (preg_match_all(self::PATTERN, $text, $matches, PREG_OFFSET_CAPTURE | PREG_SET_ORDER) === FALSE) { continue; } - if ($todoText === '') { - $errorMessage = 'comment expired on '. $date .'.'; - } else { - $errorMessage = "comment '$todoText' expired on ". $date .'.'; - } + /** @var array> $matches */ + foreach ($matches as $match) { - $errors[] = RuleErrorBuilder::message($errorMessage)->line($comment->getStartLine())->build(); + $date = $match['date'][0]; + $todoText = trim($match['comment'][0]); + + /** + * strtotime() will parse date-only values with time set to 00:00:00. + * This is fine, because this will count any expiration matching + * the current date as expired, except when ran exactly at 00:00:00. + */ + if (strtotime($date) > $this->now) { + continue; + } + + // Have always present date at the start of the message. + $errorMessage = "Expired on {$date}"; + + // If there is further text, append it. + if ($todoText !== '') { + + $errorMessage .= ": {$todoText}"; + } + + $whole_match_start_offset = $match[0][1]; + + // Count the number of newlines between the start of the whole comment, and the start of the match. + $new_lines = substr_count($text, "\n", 0, $whole_match_start_offset); + + // Set the message line to match the line the comment actually starts on. + $message_line = $comment->getStartLine() + $new_lines; + + $errors[] = RuleErrorBuilder::message($errorMessage)->line($message_line)->build(); + } } return $errors; diff --git a/tests/TodoByRuleTest.php b/tests/TodoByRuleTest.php index 85ed543..850a343 100644 --- a/tests/TodoByRuleTest.php +++ b/tests/TodoByRuleTest.php @@ -20,44 +20,52 @@ public function testRule(): void { $this->analyse([__DIR__ . '/data/example.php'], [ [ - "comment 'Expired comment1' expired on 2023-12-14.", + 'Expired on 2023-12-14: Expired comment1', 9, ], [ - "comment 'Expired comment2' expired on 2023-12-14.", + 'Expired on 2023-12-14: Expired comment2', 10, ], [ - "comment 'Expired comment3' expired on 2023-12-14.", + 'Expired on 2023-12-14: Expired comment3', 11, ], [ - "comment 'Expired comment4' expired on 2023-12-14.", + 'Expired on 2023-12-14: Expired comment4', 12, ], [ - "comment 'Expired comment5' expired on 2023-12-14.", + 'Expired on 2023-12-14: Expired comment5', 13, ], [ - "comment 'Expired commentX' expired on 2023-12-14.", + 'Expired on 2023-12-14: Expired commentX', 19, ], [ - "comment expired on 2023-12-14.", + 'Expired on 2023-12-14', 21, ], [ - "comment 'method comment' expired on 2023-12-14.", - 27, + 'Expired on 2023-12-14: class comment', + 29, ], [ - "comment 'in method comment1' expired on 2023-12-14.", - 29, + 'Expired on 2023-12-13: class comment', + 30, + ], + [ + "Expired on 2023-12-14: method comment", + 34, + ], + [ + 'Expired on 2023-12-14: in method comment1', + 36, ], [ - "comment 'in method comment2' expired on 2023-12-14.", - 31, + 'Expired on 2023-12-14: in method comment2', + 38, ], ]); } diff --git a/tests/data/example.php b/tests/data/example.php index 21f3e7c..b792586 100644 --- a/tests/data/example.php +++ b/tests/data/example.php @@ -23,6 +23,13 @@ function doFooBar():void { } +/** + * other text + * + * @todo 2023-12-14 class comment + * @TODO 2023-12-13 - class comment + * more comment data + */ class Z { // TODO: 2023-12-14 method comment public function XY():void {