Page Menu
Home
WickedGov Phorge
Search
Configure Global Search
Log In
Files
F1431128
DocCommentSniff.php
No One
Temporary
Actions
Download File
Edit File
Delete File
View Transforms
Subscribe
Flag For Later
Award Token
Size
11 KB
Referenced Files
None
Subscribers
None
DocCommentSniff.php
View Options
<?php
/**
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
* http://www.gnu.org/copyleft/gpl.html
*
* @file
*/
namespace
MediaWiki\Sniffs\Commenting
;
use
PHP_CodeSniffer\Files\File
;
use
PHP_CodeSniffer\Sniffs\Sniff
;
class
DocCommentSniff
implements
Sniff
{
/** Do not report very long asterisks line, there are eye-catchers for structure of the code */
private
const
COMMENT_START_ASTERISKS_MAX_LEN
=
10
;
/**
* List of annotations where the spacing before is not checked.
* For example because this annotations are used inside a long text
*
* @var string[]
*/
private
const
ANNOTATIONS_IGNORE_MULTI_SPACE_BEFORE
=
[
'@see'
,
'@deprecated'
,
];
/**
* @inheritDoc
*/
public
function
register
():
array
{
return
[
T_DOC_COMMENT_OPEN_TAG
];
}
/**
* Processes this test, when one of its tokens is encountered.
*
* @param File $phpcsFile The file being scanned.
* @param int $stackPtr The position of the current token in the stack passed in $tokens.
*
* @return void
*/
public
function
process
(
File
$phpcsFile
,
$stackPtr
)
{
$tokens
=
$phpcsFile
->
getTokens
();
$commentStart
=
$stackPtr
;
$commentEnd
=
$tokens
[
$stackPtr
][
'comment_closer'
];
$isMultiLineDoc
=
(
$tokens
[
$commentStart
][
'line'
]
!==
$tokens
[
$commentEnd
][
'line'
]
);
// Start token should exact /**
// Self-closing comments are tokenized also as open tag, but ignore them
if
(
$tokens
[
$commentStart
][
'code'
]
===
T_DOC_COMMENT_OPEN_TAG
&&
$tokens
[
$commentStart
][
'content'
]
!==
'/**'
&&
$tokens
[
$commentStart
][
'length'
]
<
self
::
COMMENT_START_ASTERISKS_MAX_LEN
&&
!
str_ends_with
(
$tokens
[
$commentStart
][
'content'
],
'*/'
)
)
{
$error
=
'Comment open tag must be
\'
/**
\'
'
;
$fix
=
$phpcsFile
->
addFixableError
(
$error
,
$commentStart
,
'SyntaxOpenTag'
);
if
(
$fix
)
{
$phpcsFile
->
fixer
->
replaceToken
(
$commentStart
,
'/**'
);
}
}
$columnDocStar
=
$this
->
getDocStarColumn
(
$phpcsFile
,
$commentStart
);
$prevLineDocStar
=
$tokens
[
$commentStart
][
'line'
];
$lastLine
=
$commentStart
;
$lineWithDocStar
=
true
;
$indent
=
$this
->
getCommentIndent
(
$phpcsFile
,
$commentStart
);
for
(
$i
=
$commentStart
;
$i
<=
$commentEnd
;
$i
++
)
{
$initialStarChars
=
0
;
if
(
$tokens
[
$lastLine
][
'line'
]
!==
$tokens
[
$i
][
'line'
]
)
{
if
(
!
$lineWithDocStar
)
{
$fix
=
$phpcsFile
->
addFixableError
(
'Expected
\'
*
\'
on each line'
,
$lastLine
,
'NoDocStar'
);
if
(
$fix
)
{
$posNonWhitespace
=
$phpcsFile
->
findFirstOnLine
(
[
T_DOC_COMMENT_WHITESPACE
],
$i
-
1
,
true
);
if
(
$posNonWhitespace
===
false
)
{
// empty line
$phpcsFile
->
fixer
->
addContentBefore
(
$lastLine
,
$indent
.
'*'
);
}
else
{
$phpcsFile
->
fixer
->
beginChangeset
();
// line with content, first remove old indent
for
(
$j
=
$lastLine
;
$j
<
$posNonWhitespace
;
$j
++
)
{
$phpcsFile
->
fixer
->
replaceToken
(
$j
,
''
);
}
// and set a new indent with the doc star and a space
$phpcsFile
->
fixer
->
addContentBefore
(
$lastLine
,
$indent
.
'* '
);
$phpcsFile
->
fixer
->
endChangeset
();
}
}
}
$lineWithDocStar
=
false
;
$lastLine
=
$i
;
}
// Star token should exact *
if
(
$tokens
[
$i
][
'code'
]
===
T_DOC_COMMENT_STAR
)
{
// Multi stars in a line are parsed as a new token
$initialStarChars
=
strspn
(
$tokens
[
$i
+
1
][
'content'
],
'*'
);
if
(
$initialStarChars
>
0
)
{
$error
=
'Comment star must be a single
\'
*
\'
'
;
$fix
=
$phpcsFile
->
addFixableError
(
$error
,
$i
,
'SyntaxMultiDocStar'
);
if
(
$fix
)
{
$phpcsFile
->
fixer
->
replaceToken
(
$i
+
1
,
substr
(
$tokens
[
$i
+
1
][
'content'
],
$initialStarChars
)
);
}
}
$lineWithDocStar
=
true
;
}
// Ensure whitespace or tab after /** or *
if
(
(
$tokens
[
$i
][
'code'
]
===
T_DOC_COMMENT_OPEN_TAG
||
$tokens
[
$i
][
'code'
]
===
T_DOC_COMMENT_STAR
)
&&
$tokens
[
$i
+
1
][
'code'
]
!==
T_DOC_COMMENT_CLOSE_TAG
&&
$tokens
[
$i
+
1
][
'length'
]
>
0
)
{
$commentStarSpacing
=
$i
+
1
;
$expectedSpaces
=
1
;
// ignore * removed by SyntaxMultiDocStar and count spaces after that
$currentSpaces
=
strspn
(
$tokens
[
$commentStarSpacing
][
'content'
],
"
\t
"
,
$initialStarChars
);
$error
=
null
;
$code
=
null
;
if
(
$isMultiLineDoc
&&
$currentSpaces
<
$expectedSpaces
)
{
// be relax for multiline docs, because some line breaks in @param can
// have more than one space after a doc star
$error
=
'Expected at least %s spaces after doc star; %s found'
;
$code
=
'SpacingDocStar'
;
}
elseif
(
!
$isMultiLineDoc
&&
$currentSpaces
!==
$expectedSpaces
)
{
$error
=
'Expected %s spaces after doc star on single line; %s found'
;
$code
=
'SpacingDocStarSingleLine'
;
}
if
(
$error
!==
null
&&
$code
!==
null
)
{
$fix
=
$phpcsFile
->
addFixableError
(
$error
,
$commentStarSpacing
,
$code
,
[
$expectedSpaces
,
$currentSpaces
]
);
if
(
$fix
)
{
if
(
$currentSpaces
>
$expectedSpaces
)
{
// Remove whitespace
$content
=
$tokens
[
$commentStarSpacing
][
'content'
];
$phpcsFile
->
fixer
->
replaceToken
(
$commentStarSpacing
,
substr
(
$content
,
0
,
$expectedSpaces
-
$currentSpaces
)
);
}
else
{
// Add whitespace
$phpcsFile
->
fixer
->
addContent
(
$i
,
str_repeat
(
' '
,
$expectedSpaces
)
);
}
}
}
}
if
(
!
$isMultiLineDoc
)
{
continue
;
}
// Ensure one whitespace before @param/@return
if
(
$tokens
[
$i
][
'code'
]
===
T_DOC_COMMENT_TAG
&&
$tokens
[
$i
][
'line'
]
===
$tokens
[
$i
-
1
][
'line'
]
)
{
$commentTagSpacing
=
$i
-
1
;
$expectedSpaces
=
1
;
$currentSpaces
=
strspn
(
strrev
(
$tokens
[
$commentTagSpacing
][
'content'
]
),
' '
);
// Relax the check for a list of annotations for multi spaces before the annotation
if
(
$currentSpaces
>
$expectedSpaces
&&
!
in_array
(
$tokens
[
$i
][
'content'
],
self
::
ANNOTATIONS_IGNORE_MULTI_SPACE_BEFORE
,
true
)
)
{
$data
=
[
$expectedSpaces
,
$tokens
[
$i
][
'content'
],
$currentSpaces
,
];
$fix
=
$phpcsFile
->
addFixableError
(
'Expected %s spaces before %s; %s found'
,
$commentTagSpacing
,
'SpacingDocTag'
,
$data
);
if
(
$fix
)
{
// Remove whitespace
$content
=
$tokens
[
$commentTagSpacing
][
'content'
];
$phpcsFile
->
fixer
->
replaceToken
(
$commentTagSpacing
,
substr
(
$content
,
0
,
$expectedSpaces
-
$currentSpaces
)
);
}
}
continue
;
}
// Ensure aligned * or */ for multiline comments
if
(
(
$tokens
[
$i
][
'code'
]
===
T_DOC_COMMENT_STAR
||
$tokens
[
$i
][
'code'
]
===
T_DOC_COMMENT_CLOSE_TAG
)
&&
$tokens
[
$i
][
'column'
]
!==
$columnDocStar
&&
$tokens
[
$i
][
'line'
]
!==
$prevLineDocStar
)
{
if
(
$tokens
[
$i
][
'code'
]
===
T_DOC_COMMENT_STAR
)
{
$error
=
'Comment star tag not aligned with open tag'
;
$code
=
'SyntaxAlignedDocStar'
;
}
else
{
$error
=
'Comment close tag not aligned with open tag'
;
$code
=
'SyntaxAlignedDocClose'
;
}
$fix
=
$phpcsFile
->
addFixableError
(
$error
,
$i
,
$code
);
if
(
$fix
)
{
$tokenBefore
=
$i
-
1
;
$columnOff
=
$columnDocStar
-
$tokens
[
$i
][
'column'
];
if
(
$columnOff
<
0
)
{
// Ensure to remove only whitespaces
if
(
$tokens
[
$tokenBefore
][
'code'
]
===
T_DOC_COMMENT_WHITESPACE
)
{
$columnOff
=
max
(
$columnOff
,
$tokens
[
$tokenBefore
][
'length'
]
*
-
1
);
// remove whitespaces
$phpcsFile
->
fixer
->
replaceToken
(
$tokenBefore
,
substr
(
$tokens
[
$tokenBefore
][
'content'
],
0
,
$columnOff
)
);
}
}
elseif
(
$tokens
[
$tokenBefore
][
'length'
]
!==
0
)
{
// Set correct indent
$phpcsFile
->
fixer
->
replaceToken
(
$tokenBefore
,
$indent
);
}
else
{
// Add correct indent
$phpcsFile
->
fixer
->
addContent
(
$tokenBefore
,
$indent
);
}
}
$prevLineDocStar
=
$tokens
[
$i
][
'line'
];
continue
;
}
}
// End token should exact */
if
(
$tokens
[
$commentEnd
][
'code'
]
===
T_DOC_COMMENT_CLOSE_TAG
&&
$tokens
[
$commentEnd
][
'length'
]
>
0
&&
$tokens
[
$commentEnd
][
'content'
]
!==
'*/'
)
{
$error
=
'Comment close tag must be
\'
*/
\'
'
;
$fix
=
$phpcsFile
->
addFixableError
(
$error
,
$commentEnd
,
'SyntaxCloseTag'
);
if
(
$fix
)
{
$phpcsFile
->
fixer
->
replaceToken
(
$commentEnd
,
'*/'
);
}
}
// For multi line comments the closing tag must have it own line
if
(
$isMultiLineDoc
)
{
$prev
=
$commentEnd
-
1
;
$prevNonWhitespace
=
$phpcsFile
->
findPrevious
(
[
T_DOC_COMMENT_WHITESPACE
],
$prev
,
null
,
true
);
if
(
$tokens
[
$prevNonWhitespace
][
'line'
]
===
$tokens
[
$commentEnd
][
'line'
]
)
{
$error
=
'Comment close tag should have own line'
;
$fix
=
$phpcsFile
->
addFixableError
(
$error
,
$commentEnd
,
'CloseTagOwnLine'
);
if
(
$fix
)
{
$phpcsFile
->
fixer
->
beginChangeset
();
$phpcsFile
->
fixer
->
addNewline
(
$prev
);
$phpcsFile
->
fixer
->
addContent
(
$prev
,
$indent
);
$phpcsFile
->
fixer
->
endChangeset
();
}
}
}
elseif
(
$tokens
[
$commentEnd
][
'length'
]
>
0
&&
$tokens
[
$commentEnd
-
1
][
'code'
]
!==
T_DOC_COMMENT_OPEN_TAG
)
{
// Ensure a whitespace before the token
$commentCloseSpacing
=
$commentEnd
-
1
;
$expectedSpaces
=
1
;
$currentSpaces
=
strspn
(
strrev
(
$tokens
[
$commentCloseSpacing
][
'content'
]
),
' '
);
if
(
$currentSpaces
!==
$expectedSpaces
)
{
$data
=
[
$expectedSpaces
,
$currentSpaces
,
];
$fix
=
$phpcsFile
->
addFixableError
(
'Expected %s spaces before close comment tag on single line; %s found'
,
$commentCloseSpacing
,
'SpacingSingleLineCloseTag'
,
$data
);
if
(
$fix
)
{
if
(
$currentSpaces
>
$expectedSpaces
)
{
// Remove whitespace
$content
=
$tokens
[
$commentCloseSpacing
][
'content'
];
$phpcsFile
->
fixer
->
replaceToken
(
$commentCloseSpacing
,
substr
(
$content
,
0
,
$expectedSpaces
-
$currentSpaces
)
);
}
else
{
// Add whitespace
$phpcsFile
->
fixer
->
addContentBefore
(
$commentEnd
,
str_repeat
(
' '
,
$expectedSpaces
)
);
}
}
}
}
}
/**
* @param File $phpcsFile
* @param int $stackPtr
* @return string
*/
private
function
getCommentIndent
(
File
$phpcsFile
,
int
$stackPtr
):
string
{
$firstLineToken
=
$phpcsFile
->
findFirstOnLine
(
[
T_WHITESPACE
],
$stackPtr
);
if
(
$firstLineToken
===
false
)
{
// no indent before the comment, but the doc star has one space indent
return
' '
;
}
return
$phpcsFile
->
getTokensAsString
(
$firstLineToken
,
$stackPtr
-
$firstLineToken
)
.
' '
;
}
/**
* @param File $phpcsFile
* @param int $stackPtr
* @return int
*/
private
function
getDocStarColumn
(
File
$phpcsFile
,
int
$stackPtr
):
int
{
$tokens
=
$phpcsFile
->
getTokens
();
// Handle special case /*****//** to look for the column of the first comment start
if
(
$tokens
[
$stackPtr
-
1
][
'code'
]
===
T_DOC_COMMENT_CLOSE_TAG
)
{
$stackPtr
=
$tokens
[
$stackPtr
-
1
][
'comment_opener'
];
}
// Calculate the column to align all doc stars. Use column of /**, add 1 to skip char /
return
$tokens
[
$stackPtr
][
'column'
]
+
1
;
}
}
File Metadata
Details
Attached
Mime Type
text/x-php
Expires
Sat, May 16, 20:06 (2 h, 56 m)
Storage Engine
local-disk
Storage Format
Raw Data
Storage Handle
13/d1/75bfdfa6f0c63349205c478d5bc7
Default Alt Text
DocCommentSniff.php (11 KB)
Attached To
Mode
rMWPROD MediaWiki Production
Attached
Detach File
Event Timeline
Log In to Comment