-
-
Notifications
You must be signed in to change notification settings - Fork 63
/
Copy pathDisallowTabIndentSniff.php
201 lines (174 loc) · 7.48 KB
/
DisallowTabIndentSniff.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
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
<?php
/**
* Throws errors if tabs are used for indentation.
*
* @author Greg Sherwood <gsherwood@squiz.net>
* @copyright 2006-2015 Squiz Pty Ltd (ABN 77 084 670 600)
* @license https://github.com/PHPCSStandards/PHP_CodeSniffer/blob/master/licence.txt BSD Licence
*/
namespace PHP_CodeSniffer\Standards\Generic\Sniffs\WhiteSpace;
use PHP_CodeSniffer\Files\File;
use PHP_CodeSniffer\Sniffs\Sniff;
class DisallowTabIndentSniff implements Sniff
{
/**
* A list of tokenizers this sniff supports.
*
* @var array
*/
public $supportedTokenizers = [
'PHP',
'JS',
'CSS',
];
/**
* The --tab-width CLI value that is being used.
*
* @var integer
*/
private $tabWidth = null;
/**
* Returns an array of tokens this test wants to listen for.
*
* @return array<int|string>
*/
public function register()
{
return [
T_OPEN_TAG,
T_OPEN_TAG_WITH_ECHO,
];
}//end register()
/**
* Processes this test, when one of its tokens is encountered.
*
* @param \PHP_CodeSniffer\Files\File $phpcsFile All the tokens found in the document.
* @param int $stackPtr The position of the current token in
* the stack passed in $tokens.
*
* @return int
*/
public function process(File $phpcsFile, $stackPtr)
{
if ($this->tabWidth === null) {
if (isset($phpcsFile->config->tabWidth) === false || $phpcsFile->config->tabWidth === 0) {
// We have no idea how wide tabs are, so assume 4 spaces for metrics.
$this->tabWidth = 4;
} else {
$this->tabWidth = $phpcsFile->config->tabWidth;
}
}
$tokens = $phpcsFile->getTokens();
$checkTokens = [
T_WHITESPACE => true,
T_INLINE_HTML => true,
T_DOC_COMMENT_WHITESPACE => true,
T_DOC_COMMENT_STRING => true,
T_COMMENT => true,
T_END_HEREDOC => true,
T_END_NOWDOC => true,
T_YIELD_FROM => true,
];
for ($i = 0; $i < $phpcsFile->numTokens; $i++) {
if (isset($checkTokens[$tokens[$i]['code']]) === false) {
continue;
}
// If tabs are being converted to spaces by the tokeniser, the
// original content should be checked instead of the converted content.
if (isset($tokens[$i]['orig_content']) === true) {
$content = $tokens[$i]['orig_content'];
} else {
$content = $tokens[$i]['content'];
}
if ($content === '') {
continue;
}
// If this is an inline HTML token or a subsequent line of a multi-line comment,
// split off the indentation as that is the only part to take into account for the metrics.
$indentation = $content;
if (($tokens[$i]['code'] === T_INLINE_HTML
|| $tokens[$i]['code'] === T_COMMENT)
&& preg_match('`^(\s*)\S.*`s', $content, $matches) > 0
) {
if (isset($matches[1]) === true) {
$indentation = $matches[1];
}
}
if (($tokens[$i]['code'] === T_DOC_COMMENT_WHITESPACE
|| $tokens[$i]['code'] === T_COMMENT)
&& $indentation === ' '
) {
// Ignore all non-indented comments, especially for recording metrics.
continue;
}
$recordMetrics = true;
if ($content === $indentation
&& isset($tokens[($i + 1)]) === true
&& $tokens[$i]['line'] < $tokens[($i + 1)]['line']
) {
// Don't record metrics for empty lines.
$recordMetrics = false;
}
$foundTabs = substr_count($content, "\t");
$error = 'Spaces must be used to indent lines; tabs are not allowed';
$errorCode = 'TabsUsed';
if ($tokens[$i]['column'] === 1) {
if ($recordMetrics === true) {
$foundIndentSpaces = substr_count($indentation, ' ');
$foundIndentTabs = substr_count($indentation, "\t");
if ($foundIndentTabs > 0 && $foundIndentSpaces === 0) {
$phpcsFile->recordMetric($i, 'Line indent', 'tabs');
} else if ($foundIndentTabs === 0 && $foundIndentSpaces > 0) {
$phpcsFile->recordMetric($i, 'Line indent', 'spaces');
} else if ($foundIndentTabs > 0 && $foundIndentSpaces > 0) {
$spacePosition = strpos($indentation, ' ');
$tabAfterSpaces = strpos($indentation, "\t", $spacePosition);
if ($tabAfterSpaces !== false) {
$phpcsFile->recordMetric($i, 'Line indent', 'mixed');
} else {
// Check for use of precision spaces.
$numTabs = (int) floor($foundIndentSpaces / $this->tabWidth);
if ($numTabs === 0) {
$phpcsFile->recordMetric($i, 'Line indent', 'tabs');
} else {
$phpcsFile->recordMetric($i, 'Line indent', 'mixed');
}
}
}
}//end if
} else {
// Look for tabs so we can report and replace, but don't
// record any metrics about them because they aren't
// line indent tokens.
if ($foundTabs > 0) {
$error = 'Spaces must be used for alignment; tabs are not allowed';
$errorCode = 'NonIndentTabsUsed';
}
}//end if
if ($foundTabs === 0) {
continue;
}
// Report, but don't auto-fix tab identation for a PHP 7.3+ flexible heredoc/nowdoc closer.
// Auto-fixing this would cause parse errors as the indentation of the heredoc/nowdoc contents
// needs to use the same type of indentation. Also see: https://3v4l.org/7OF3M .
if ($tokens[$i]['code'] === T_END_HEREDOC || $tokens[$i]['code'] === T_END_NOWDOC) {
$phpcsFile->addError($error, $i, $errorCode.'HeredocCloser');
continue;
}
$fix = $phpcsFile->addFixableError($error, $i, $errorCode);
if ($fix === true) {
if (isset($tokens[$i]['orig_content']) === true) {
// Use the replacement that PHPCS has already done.
$phpcsFile->fixer->replaceToken($i, $tokens[$i]['content']);
} else {
// Replace tabs with spaces, using an indent of tabWidth spaces.
// Other sniffs can then correct the indent if they need to.
$newContent = str_replace("\t", str_repeat(' ', $this->tabWidth), $tokens[$i]['content']);
$phpcsFile->fixer->replaceToken($i, $newContent);
}
}
}//end for
// Ignore the rest of the file.
return $phpcsFile->numTokens;
}//end process()
}//end class