wishthis/vendor/phpcsstandards/phpcsutils/PHPCSUtils/Utils/GetTokensAsString.php

263 lines
12 KiB
PHP
Raw Normal View History

2023-09-19 17:26:46 +00:00
<?php
/**
* PHPCSUtils, utility functions and classes for PHP_CodeSniffer sniff developers.
*
* @package PHPCSUtils
* @copyright 2019-2020 PHPCSUtils Contributors
* @license https://opensource.org/licenses/LGPL-3.0 LGPL3
* @link https://github.com/PHPCSStandards/PHPCSUtils
*/
namespace PHPCSUtils\Utils;
use PHP_CodeSniffer\Exceptions\RuntimeException;
use PHP_CodeSniffer\Files\File;
use PHP_CodeSniffer\Util\Tokens;
/**
* Utility functions to retrieve the content of a set of tokens as a string.
*
* In contrast to the PHPCS native {@see \PHP_CodeSniffer\Files\File::getTokensAsString()} method,
* which has `$length` as the third parameter, all methods in this class expect a stack pointer to
* an `$end` token (inclusive) as the third parameter.
*
* @since 1.0.0 The principle of this class is loosely based on and inspired by the
* {@see \PHP_CodeSniffer\Files\File::getTokensAsString()} method in the
* PHPCS native `File` class.
* Also see {@see \PHPCSUtils\BackCompat\BCFile::getTokensAsString()}.
*/
final class GetTokensAsString
{
/**
* Retrieve the tab-replaced content of the tokens from the specified start position in
* the token stack to the specified end position (inclusive).
*
* Alias for the {@see \PHPCSUtils\Utils\GetTokensAsString::tabReplaced()} method.
*
* @since 1.0.0
*
* @param \PHP_CodeSniffer\Files\File $phpcsFile The file being scanned.
* @param int $start The position to start from in the token stack.
* @param int $end The position to end at in the token stack (inclusive).
*
* @return string The token contents.
*
* @throws \PHP_CodeSniffer\Exceptions\RuntimeException If the specified start position does not exist.
*/
public static function normal(File $phpcsFile, $start, $end)
{
return self::tabReplaced($phpcsFile, $start, $end);
}
/**
* Retrieve the tab-replaced content of the tokens from the specified start position in
* the token stack to the specified end position (inclusive).
*
* This is the default behaviour for PHPCS.
*
* If the `tabWidth` is set, either via a (custom) ruleset, the config file or by passing it
* on the command-line, PHPCS will automatically replace tabs with spaces.
* The `'content'` index key in the `$tokens` array will contain the tab-replaced content.
* The `'orig_content'` index key in the `$tokens` array will contain the original content.
*
* @see \PHP_CodeSniffer\Files\File::getTokensAsString() Similar length-based function.
* @see \PHPCSUtils\BackCompat\BCFile::getTokensAsString() Cross-version compatible version of the original.
*
* @since 1.0.0
*
* @param \PHP_CodeSniffer\Files\File $phpcsFile The file being scanned.
* @param int $start The position to start from in the token stack.
* @param int $end The position to end at in the token stack (inclusive).
*
* @return string The token contents.
*
* @throws \PHP_CodeSniffer\Exceptions\RuntimeException If the specified start position does not exist.
*/
public static function tabReplaced(File $phpcsFile, $start, $end)
{
return self::getString($phpcsFile, $start, $end);
}
/**
* Retrieve the original content of the tokens from the specified start position in
* the token stack to the specified end position (inclusive).
*
* In contrast to the {@see GetTokensAsString::tabReplaced()} method, this method will return
* the original token content for the specified tokens.
* That means that if the original content contained tabs, the return value of this function
* will also contain tabs.
*
* The same can be achieved in PHPCS since version 3.3.0, by calling the
* {@see \PHP_CodeSniffer\Files\File::getTokensAsString()} method and passing `true` as the
* value for the `$origContent` parameter.
*
* @see \PHP_CodeSniffer\Files\File::getTokensAsString() Similar length-based function.
* @see \PHPCSUtils\BackCompat\BCFile::getTokensAsString() Cross-version compatible version of the original.
*
* @since 1.0.0
*
* @param \PHP_CodeSniffer\Files\File $phpcsFile The file being scanned.
* @param int $start The position to start from in the token stack.
* @param int $end The position to end at in the token stack (inclusive).
*
* @return string The token contents.
*
* @throws \PHP_CodeSniffer\Exceptions\RuntimeException If the specified start position does not exist.
*/
public static function origContent(File $phpcsFile, $start, $end)
{
return self::getString($phpcsFile, $start, $end, true);
}
/**
* Retrieve the content of the tokens from the specified start position in the token
* stack to the specified end position (inclusive) without comments.
*
* @see \PHP_CodeSniffer\Files\File::getTokensAsString() Loosely related function.
*
* @since 1.0.0
*
* @param \PHP_CodeSniffer\Files\File $phpcsFile The file being scanned.
* @param int $start The position to start from in the token stack.
* @param int $end The position to end at in the token stack (inclusive).
*
* @return string The token contents stripped off comments.
*
* @throws \PHP_CodeSniffer\Exceptions\RuntimeException If the specified start position does not exist.
*/
public static function noComments(File $phpcsFile, $start, $end)
{
return self::getString($phpcsFile, $start, $end, false, true);
}
/**
* Retrieve the code-tokens only content of the tokens from the specified start position
* in the token stack to the specified end position (inclusive) without whitespace or comments.
*
* This is useful, for instance, to retrieve a namespace name without stray whitespace or comments.
* Use this function selectively and with care!
*
* @see \PHP_CodeSniffer\Files\File::getTokensAsString() Loosely related function.
*
* @since 1.0.0
*
* @param \PHP_CodeSniffer\Files\File $phpcsFile The file being scanned.
* @param int $start The position to start from in the token stack.
* @param int $end The position to end at in the token stack (inclusive).
*
* @return string The token contents stripped off comments and whitespace.
*
* @throws \PHP_CodeSniffer\Exceptions\RuntimeException If the specified start position does not exist.
*/
public static function noEmpties(File $phpcsFile, $start, $end)
{
return self::getString($phpcsFile, $start, $end, false, true, true);
}
/**
* Retrieve the content of the tokens from the specified start position in the token
* stack to the specified end position (inclusive) with all consecutive whitespace tokens - tabs,
* new lines, multiple spaces - replaced by a single space and optionally without comments.
*
* @see \PHP_CodeSniffer\Files\File::getTokensAsString() Loosely related function.
*
* @since 1.0.0
*
* @param \PHP_CodeSniffer\Files\File $phpcsFile The file being scanned.
* @param int $start The position to start from in the token stack.
* @param int $end The position to end at in the token stack (inclusive).
* @param bool $stripComments Whether comments should be stripped from the contents.
* Defaults to `false`.
*
* @return string The token contents with compacted whitespace and optionally stripped off comments.
*
* @throws \PHP_CodeSniffer\Exceptions\RuntimeException If the specified start position does not exist.
*/
public static function compact(File $phpcsFile, $start, $end, $stripComments = false)
{
return self::getString($phpcsFile, $start, $end, false, $stripComments, false, true);
}
/**
* Retrieve the content of the tokens from the specified start position in the token stack
* to the specified end position (inclusive).
*
* @since 1.0.0
*
* @param \PHP_CodeSniffer\Files\File $phpcsFile The file being scanned.
* @param int $start The position to start from in the token stack.
* @param int $end The position to end at in the token stack (inclusive).
* @param bool $origContent Whether the original content or the tab replaced
* content should be used.
* Defaults to `false` (= tabs replaced with spaces).
* @param bool $stripComments Whether comments should be stripped from the contents.
* Defaults to `false`.
* @param bool $stripWhitespace Whether whitespace should be stripped from the contents.
* Defaults to `false`.
* @param bool $compact Whether all consecutive whitespace tokens should be
* replaced with a single space. Defaults to `false`.
*
* @return string The token contents.
*
* @throws \PHP_CodeSniffer\Exceptions\RuntimeException If the specified start position does not exist.
*/
protected static function getString(
File $phpcsFile,
$start,
$end,
$origContent = false,
$stripComments = false,
$stripWhitespace = false,
$compact = false
) {
$tokens = $phpcsFile->getTokens();
if (\is_int($start) === false || isset($tokens[$start]) === false) {
throw new RuntimeException(
'The $start position for GetTokensAsString methods must exist in the token stack'
);
}
if (\is_int($end) === false || $end < $start) {
return '';
}
$str = '';
if ($end >= $phpcsFile->numTokens) {
$end = ($phpcsFile->numTokens - 1);
}
$lastAdded = null;
for ($i = $start; $i <= $end; $i++) {
if ($stripComments === true && isset(Tokens::$commentTokens[$tokens[$i]['code']])) {
continue;
}
if ($stripWhitespace === true && $tokens[$i]['code'] === \T_WHITESPACE) {
continue;
}
if ($compact === true && $tokens[$i]['code'] === \T_WHITESPACE) {
if (isset($lastAdded) === false || $tokens[$lastAdded]['code'] !== \T_WHITESPACE) {
$str .= ' ';
$lastAdded = $i;
}
continue;
}
// If tabs are being converted to spaces by the tokenizer, the
// original content should be used instead of the converted content.
if ($origContent === true && isset($tokens[$i]['orig_content']) === true) {
$str .= $tokens[$i]['orig_content'];
} else {
$str .= $tokens[$i]['content'];
}
$lastAdded = $i;
}
return $str;
}
}