123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127 |
- <?php
- declare(strict_types=1);
- /**
- * SPDX-FileCopyrightText: 2023 Nextcloud GmbH and Nextcloud contributors
- * SPDX-License-Identifier: AGPL-3.0-only
- */
- use PhpParser\Node\Stmt;
- use PhpParser\Node\Stmt\ClassLike;
- use Psalm\CodeLocation;
- use Psalm\DocComment;
- use Psalm\Exception\DocblockParseException;
- use Psalm\FileSource;
- use Psalm\Issue\InvalidDocblock;
- use Psalm\IssueBuffer;
- use Psalm\Plugin\EventHandler\Event\AfterClassLikeVisitEvent;
- class OcpSinceChecker implements Psalm\Plugin\EventHandler\AfterClassLikeVisitInterface {
- public static function afterClassLikeVisit(AfterClassLikeVisitEvent $event): void {
- $classLike = $event->getStmt();
- $statementsSource = $event->getStatementsSource();
- self::checkClassComment($classLike, $statementsSource);
- foreach ($classLike->stmts as $stmt) {
- if ($stmt instanceof ClassConst) {
- self::checkStatementComment($stmt, $statementsSource, 'constant');
- }
- if ($stmt instanceof ClassMethod) {
- self::checkStatementComment($stmt, $statementsSource, 'method');
- }
- if ($stmt instanceof EnumCase) {
- self::checkStatementComment($stmt, $statementsSource, 'enum');
- }
- }
- }
- private static function checkClassComment(ClassLike $stmt, FileSource $statementsSource): void {
- $docblock = $stmt->getDocComment();
- if ($docblock === null) {
- IssueBuffer::maybeAdd(
- new InvalidDocblock(
- 'PHPDoc is required for classes/interfaces in OCP.',
- new CodeLocation($statementsSource, $stmt)
- )
- );
- return;
- }
- try {
- $parsedDocblock = DocComment::parsePreservingLength($docblock);
- } catch (DocblockParseException $e) {
- IssueBuffer::maybeAdd(
- new InvalidDocblock(
- $e->getMessage(),
- new CodeLocation($statementsSource, $stmt)
- )
- );
- return;
- }
- if (!isset($parsedDocblock->tags['since'])) {
- IssueBuffer::maybeAdd(
- new InvalidDocblock(
- '@since is required for classes/interfaces in OCP.',
- new CodeLocation($statementsSource, $stmt)
- )
- );
- }
- if (isset($parsedDocblock->tags['depreacted'])) {
- IssueBuffer::maybeAdd(
- new InvalidDocblock(
- 'Typo in @deprecated for classes/interfaces in OCP.',
- new CodeLocation($statementsSource, $stmt)
- )
- );
- }
- }
- private static function checkStatementComment(Stmt $stmt, FileSource $statementsSource, string $type): void {
- $docblock = $stmt->getDocComment();
- if ($docblock === null) {
- IssueBuffer::maybeAdd(
- new InvalidDocblock(
- 'PHPDoc is required for ' . $type . 's in OCP.',
- new CodeLocation($statementsSource, $stmt)
- ),
- );
- return;
- }
- try {
- $parsedDocblock = DocComment::parsePreservingLength($docblock);
- } catch (DocblockParseException $e) {
- IssueBuffer::maybeAdd(
- new InvalidDocblock(
- $e->getMessage(),
- new CodeLocation($statementsSource, $stmt)
- )
- );
- return;
- }
- if (!isset($parsedDocblock->tags['since'])) {
- IssueBuffer::maybeAdd(
- new InvalidDocblock(
- '@since is required for ' . $type . 's in OCP.',
- new CodeLocation($statementsSource, $stmt)
- )
- );
- }
- if (isset($parsedDocblock->tags['depreacted'])) {
- IssueBuffer::maybeAdd(
- new InvalidDocblock(
- 'Typo in @deprecated for ' . $type . ' in OCP.',
- new CodeLocation($statementsSource, $stmt)
- )
- );
- }
- }
- }
|