-
Notifications
You must be signed in to change notification settings - Fork 157
/
Copy pathDirectThrowSniff.php
110 lines (102 loc) · 3.25 KB
/
DirectThrowSniff.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
<?php
/**
* Copyright 2018 Adobe
* All Rights Reserved.
*/
declare(strict_types=1);
namespace Magento2\Sniffs\Exceptions;
use PHP_CodeSniffer\Sniffs\Sniff;
use PHP_CodeSniffer\Files\File;
/**
* Detects possible direct throws of Exceptions.
*/
class DirectThrowSniff implements Sniff
{
/**
* String representation of warning.
* phpcs:disable Generic.Files.LineLength.TooLong
* @var string
*/
protected $warningMessage = 'Direct throw of generic Exception is discouraged. Use context specific instead.';
//phpcs:enable
/**
* Warning violation code.
*
* @var string
*/
protected $warningCode = 'FoundDirectThrow';
/**
* @inheritdoc
*/
public function register()
{
return [T_THROW];
}
/**
* @inheritdoc
*/
public function process(File $phpcsFile, $stackPtr)
{
$tokens = $phpcsFile->getTokens();
$endOfStatement = $phpcsFile->findEndOfStatement($stackPtr);
$posOfException = $phpcsFile->findNext(T_STRING, $stackPtr, $endOfStatement);
$fullExceptionString = $this->getFullClassNameAndAlias($tokens, $stackPtr, $endOfStatement);
$exceptionString = 'Exception';
$customExceptionFound = false;
foreach ($tokens as $key => $token) {
if ($token['code'] !== T_USE) {
continue;
}
$endOfUse = $phpcsFile->findEndOfStatement($key);
$useStatementValue = $this->getFullClassNameAndAlias($tokens, $key, $endOfUse);
//we safely consider use statement has alias will not be a direct exception class
if (empty($useStatementValue['alias'])) {
if (substr($useStatementValue['name'], 0, strlen($exceptionString)) !== $exceptionString
&& substr($useStatementValue['name'], -strlen($exceptionString)) === $exceptionString
&& $useStatementValue['name'] !== $exceptionString
) {
$customExceptionFound = true;
break;
}
}
}
if (($tokens[$posOfException]['content'] === 'Exception' && !$customExceptionFound)
|| $fullExceptionString['name'] === '\Exception'
) {
$phpcsFile->addWarning(
$this->warningMessage,
$stackPtr,
$this->warningCode,
[$posOfException]
);
}
}
/**
* Get full class name and alias
*
* @param array $tokens
* @param int $start
* @param int $end
* @return array
*/
private function getFullClassNameAndAlias($tokens, $start, $end): array
{
$fullName = $alias = '';
$foundAlias = false;
for ($i = $start; $i <= $end; $i++) {
$type = $tokens[$i]['code'];
if ($type === T_AS) {
$foundAlias = true;
continue;
}
if ($type === T_STRING || $type === T_NS_SEPARATOR) {
if (!$foundAlias) {
$fullName .= $tokens[$i]['content'];
} else {
$alias = $tokens[$i]['content'];
}
}
}
return ['name' => $fullName, 'alias' => $alias];
}
}