class FullyQualifiedNamespaceSniff in Coder 8.3.x
Same name and namespace in other branches
- 8.3 coder_sniffer/Drupal/Sniffs/Classes/FullyQualifiedNamespaceSniff.php \Drupal\Sniffs\Classes\FullyQualifiedNamespaceSniff
- 8.2 coder_sniffer/Drupal/Sniffs/Classes/FullyQualifiedNamespaceSniff.php \Drupal\Sniffs\Classes\FullyQualifiedNamespaceSniff
Checks that class references do not use FQN but use statements.
@category PHP @package PHP_CodeSniffer @link http://pear.php.net/package/PHP_CodeSniffer
Hierarchy
- class \Drupal\Sniffs\Classes\FullyQualifiedNamespaceSniff implements \PHP_CodeSniffer\Sniffs\Sniff
Expanded class hierarchy of FullyQualifiedNamespaceSniff
File
- coder_sniffer/
Drupal/ Sniffs/ Classes/ FullyQualifiedNamespaceSniff.php, line 22
Namespace
Drupal\Sniffs\ClassesView source
class FullyQualifiedNamespaceSniff implements Sniff {
/**
* Returns an array of tokens this test wants to listen for.
*
* @return array<int|string>
*/
public function register() {
return [
T_NS_SEPARATOR,
];
}
//end register()
/**
* Processes this test, when one of its tokens is encountered.
*
* @param \PHP_CodeSniffer\Files\File $phpcsFile The PHP_CodeSniffer file where the
* token was found.
* @param int $stackPtr The position in the PHP_CodeSniffer
* file's token stack where the token
* was found.
*
* @return void|int Optionally returns a stack pointer. The sniff will not be
* called again on the current file until the returned stack
* pointer is reached. Return $phpcsFile->numTokens + 1 to skip
* the rest of the file.
*/
public function process(File $phpcsFile, $stackPtr) {
$tokens = $phpcsFile
->getTokens();
// Skip this sniff in *api.php files because they want to have fully
// qualified names for documentation purposes.
if (substr($phpcsFile
->getFilename(), -8) === '.api.php') {
return $phpcsFile->numTokens + 1;
}
// We are only interested in a backslash embedded between strings, which
// means this is a class reference with more than once namespace part.
if ($tokens[$stackPtr - 1]['code'] !== T_STRING || $tokens[$stackPtr + 1]['code'] !== T_STRING) {
return;
}
// Check if this is a use statement and ignore those.
$before = $phpcsFile
->findPrevious([
T_STRING,
T_NS_SEPARATOR,
T_WHITESPACE,
T_COMMA,
T_AS,
], $stackPtr, null, true);
if ($tokens[$before]['code'] === T_USE || $tokens[$before]['code'] === T_NAMESPACE) {
return $phpcsFile
->findNext([
T_STRING,
T_NS_SEPARATOR,
T_WHITESPACE,
T_COMMA,
T_AS,
], $stackPtr + 1, null, true);
}
else {
$before = $phpcsFile
->findPrevious([
T_STRING,
T_NS_SEPARATOR,
T_WHITESPACE,
], $stackPtr, null, true);
}
// If this is a namespaced function call then ignore this because use
// statements for functions are not possible in PHP 5.5 and lower.
$after = $phpcsFile
->findNext([
T_STRING,
T_NS_SEPARATOR,
T_WHITESPACE,
], $stackPtr, null, true);
if ($tokens[$after]['code'] === T_OPEN_PARENTHESIS && $tokens[$before]['code'] !== T_NEW) {
return $after + 1;
}
$fullName = $phpcsFile
->getTokensAsString($before + 1, $after - 1 - $before);
$fullName = trim($fullName, "\\ \n");
$parts = explode('\\', $fullName);
$className = end($parts);
// Check if there is a use statement already for this class and
// namespace.
$conflict = false;
$alreadyUsed = false;
$aliasName = false;
$useStatement = $phpcsFile
->findNext(T_USE, 0);
while ($useStatement !== false && empty($tokens[$useStatement]['conditions']) === true) {
$endPtr = $phpcsFile
->findEndOfStatement($useStatement);
$useEnd = $phpcsFile
->findNext([
T_STRING,
T_NS_SEPARATOR,
T_WHITESPACE,
], $useStatement + 1, null, true) - 1;
$useFullName = trim($phpcsFile
->getTokensAsString($useStatement + 1, $useEnd - $useStatement));
// Check if use statement contains an alias.
$asPtr = $phpcsFile
->findNext(T_AS, $useEnd + 1, $endPtr);
if ($asPtr !== false) {
$aliasName = trim($phpcsFile
->getTokensAsString($asPtr + 1, $endPtr - 1 - $asPtr));
}
if (strcasecmp($useFullName, $fullName) === 0) {
$alreadyUsed = true;
break;
}
$parts = explode('\\', $useFullName);
$useClassName = end($parts);
// Check if the resulting classname would conflict with another
// use statement.
if ($aliasName === $className || $useClassName === $className) {
$conflict = true;
break;
}
$aliasName = false;
// Check if we're currently in a multi-use statement.
if ($tokens[$endPtr]['code'] === T_COMMA) {
$useStatement = $endPtr;
continue;
}
$useStatement = $phpcsFile
->findNext(T_USE, $endPtr + 1);
}
//end while
if ($conflict === false) {
$classStatement = $phpcsFile
->findNext(T_CLASS, 0);
while ($classStatement !== false) {
$afterClassStatement = $phpcsFile
->findNext(T_WHITESPACE, $classStatement + 1, null, true);
// Check for 'class ClassName' declarations.
if ($tokens[$afterClassStatement]['code'] === T_STRING) {
$declaredName = $tokens[$afterClassStatement]['content'];
if ($declaredName === $className) {
$conflict = true;
break;
}
}
$classStatement = $phpcsFile
->findNext(T_CLASS, $classStatement + 1);
}
}
$error = 'Namespaced classes/interfaces/traits should be referenced with use statements';
if ($conflict === true) {
$fix = false;
$phpcsFile
->addError($error, $stackPtr, 'UseStatementMissing');
}
else {
$fix = $phpcsFile
->addFixableError($error, $stackPtr, 'UseStatementMissing');
}
if ($fix === true) {
$phpcsFile->fixer
->beginChangeset();
// Replace the fully qualified name with the local name.
for ($i = $before + 1; $i < $after; $i++) {
if ($tokens[$i]['code'] !== T_WHITESPACE) {
$phpcsFile->fixer
->replaceToken($i, '');
}
}
// Use alias name if available.
if ($aliasName !== false) {
$phpcsFile->fixer
->addContentBefore($after - 1, $aliasName);
}
else {
$phpcsFile->fixer
->addContentBefore($after - 1, $className);
}
// Insert use statement at the beginning of the file if it is not there
// already. Also check if another sniff (for example
// UnusedUseStatementSniff) has already deleted the use statement, then
// we need to add it back.
if ($alreadyUsed === false || $phpcsFile->fixer
->getTokenContent($useStatement) !== $tokens[$useStatement]['content']) {
if ($aliasName !== false) {
$use = "use {$fullName} as {$aliasName};";
}
else {
$use = "use {$fullName};";
}
// Check if there is a group of use statements and add it there.
$useStatement = $phpcsFile
->findNext(T_USE, 0);
if ($useStatement !== false && empty($tokens[$useStatement]['conditions']) === true) {
$phpcsFile->fixer
->addContentBefore($useStatement, "{$use}\n");
}
else {
// Check if there is an @file comment.
$beginning = 0;
$fileComment = $phpcsFile
->findNext(T_WHITESPACE, $beginning + 1, null, true);
if ($tokens[$fileComment]['code'] === T_DOC_COMMENT_OPEN_TAG) {
$beginning = $tokens[$fileComment]['comment_closer'];
$phpcsFile->fixer
->addContent($beginning, "\n\n{$use}\n");
}
else {
$phpcsFile->fixer
->addContent($beginning, "{$use}\n");
}
}
}
//end if
$phpcsFile->fixer
->endChangeset();
}
//end if
// Continue after this class reference so that errors for this are not
// flagged multiple times.
return $phpcsFile
->findNext([
T_STRING,
T_NS_SEPARATOR,
], $stackPtr + 1, null, true);
}
}
Members
Name | Modifiers | Type | Description | Overrides |
---|---|---|---|---|
FullyQualifiedNamespaceSniff:: |
public | function | Processes this test, when one of its tokens is encountered. | |
FullyQualifiedNamespaceSniff:: |
public | function | Returns an array of tokens this test wants to listen for. |