You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
568 lines
18 KiB
568 lines
18 KiB
<?php |
|
|
|
declare(strict_types=1); |
|
|
|
/* |
|
* The MIT License (MIT) |
|
* |
|
* Copyright (c) 2013 Jonathan Vollebregt (jnvsor@gmail.com), Rokas Šleinius (raveren@gmail.com) |
|
* |
|
* Permission is hereby granted, free of charge, to any person obtaining a copy of |
|
* this software and associated documentation files (the "Software"), to deal in |
|
* the Software without restriction, including without limitation the rights to |
|
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of |
|
* the Software, and to permit persons to whom the Software is furnished to do so, |
|
* subject to the following conditions: |
|
* |
|
* The above copyright notice and this permission notice shall be included in all |
|
* copies or substantial portions of the Software. |
|
* |
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS |
|
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR |
|
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER |
|
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN |
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. |
|
*/ |
|
|
|
namespace Kint; |
|
|
|
/** |
|
* @psalm-type PhpTokenArray = array{int, string, int} |
|
* @psalm-type PhpToken = string|PhpTokenArray |
|
*/ |
|
class CallFinder |
|
{ |
|
private static $ignore = [ |
|
T_CLOSE_TAG => true, |
|
T_COMMENT => true, |
|
T_DOC_COMMENT => true, |
|
T_INLINE_HTML => true, |
|
T_OPEN_TAG => true, |
|
T_OPEN_TAG_WITH_ECHO => true, |
|
T_WHITESPACE => true, |
|
]; |
|
|
|
/** |
|
* Things we need to do specially for operator tokens: |
|
* - Refuse to strip spaces around them |
|
* - Wrap the access path in parentheses if there |
|
* are any of these in the final short parameter. |
|
*/ |
|
private static $operator = [ |
|
T_AND_EQUAL => true, |
|
T_BOOLEAN_AND => true, |
|
T_BOOLEAN_OR => true, |
|
T_ARRAY_CAST => true, |
|
T_BOOL_CAST => true, |
|
T_CLASS => true, |
|
T_CLONE => true, |
|
T_CONCAT_EQUAL => true, |
|
T_DEC => true, |
|
T_DIV_EQUAL => true, |
|
T_DOUBLE_CAST => true, |
|
T_FUNCTION => true, |
|
T_INC => true, |
|
T_INCLUDE => true, |
|
T_INCLUDE_ONCE => true, |
|
T_INSTANCEOF => true, |
|
T_INT_CAST => true, |
|
T_IS_EQUAL => true, |
|
T_IS_GREATER_OR_EQUAL => true, |
|
T_IS_IDENTICAL => true, |
|
T_IS_NOT_EQUAL => true, |
|
T_IS_NOT_IDENTICAL => true, |
|
T_IS_SMALLER_OR_EQUAL => true, |
|
T_LOGICAL_AND => true, |
|
T_LOGICAL_OR => true, |
|
T_LOGICAL_XOR => true, |
|
T_MINUS_EQUAL => true, |
|
T_MOD_EQUAL => true, |
|
T_MUL_EQUAL => true, |
|
T_NEW => true, |
|
T_OBJECT_CAST => true, |
|
T_OR_EQUAL => true, |
|
T_PLUS_EQUAL => true, |
|
T_REQUIRE => true, |
|
T_REQUIRE_ONCE => true, |
|
T_SL => true, |
|
T_SL_EQUAL => true, |
|
T_SR => true, |
|
T_SR_EQUAL => true, |
|
T_STRING_CAST => true, |
|
T_UNSET_CAST => true, |
|
T_XOR_EQUAL => true, |
|
T_POW => true, |
|
T_POW_EQUAL => true, |
|
T_SPACESHIP => true, |
|
T_DOUBLE_ARROW => true, |
|
'!' => true, |
|
'%' => true, |
|
'&' => true, |
|
'*' => true, |
|
'+' => true, |
|
'-' => true, |
|
'.' => true, |
|
'/' => true, |
|
':' => true, |
|
'<' => true, |
|
'=' => true, |
|
'>' => true, |
|
'?' => true, |
|
'^' => true, |
|
'|' => true, |
|
'~' => true, |
|
]; |
|
|
|
private static $strip = [ |
|
'(' => true, |
|
')' => true, |
|
'[' => true, |
|
']' => true, |
|
'{' => true, |
|
'}' => true, |
|
T_OBJECT_OPERATOR => true, |
|
T_DOUBLE_COLON => true, |
|
T_NS_SEPARATOR => true, |
|
]; |
|
|
|
private static $classcalls = [ |
|
T_DOUBLE_COLON => true, |
|
T_OBJECT_OPERATOR => true, |
|
]; |
|
|
|
private static $namespace = [ |
|
T_STRING => true, |
|
]; |
|
|
|
/** |
|
* @psalm-param callable-array|callable-string $function |
|
* |
|
* @psalm-return list<array{parameters: list, modifiers: list<PhpToken>}> |
|
* |
|
* @return array List of matching calls on the relevant line |
|
*/ |
|
public static function getFunctionCalls(string $source, int $line, $function): array |
|
{ |
|
static $up = [ |
|
'(' => true, |
|
'[' => true, |
|
'{' => true, |
|
T_CURLY_OPEN => true, |
|
T_DOLLAR_OPEN_CURLY_BRACES => true, |
|
]; |
|
static $down = [ |
|
')' => true, |
|
']' => true, |
|
'}' => true, |
|
]; |
|
static $modifiers = [ |
|
'!' => true, |
|
'@' => true, |
|
'~' => true, |
|
'+' => true, |
|
'-' => true, |
|
]; |
|
static $identifier = [ |
|
T_DOUBLE_COLON => true, |
|
T_STRING => true, |
|
T_NS_SEPARATOR => true, |
|
]; |
|
|
|
if (KINT_PHP74) { |
|
self::$operator[T_FN] = true; |
|
self::$operator[T_COALESCE_EQUAL] = true; |
|
} |
|
|
|
if (KINT_PHP80) { |
|
$up[T_ATTRIBUTE] = true; |
|
self::$operator[T_MATCH] = true; |
|
self::$strip[T_NULLSAFE_OBJECT_OPERATOR] = true; |
|
self::$classcalls[T_NULLSAFE_OBJECT_OPERATOR] = true; |
|
self::$namespace[T_NAME_FULLY_QUALIFIED] = true; |
|
self::$namespace[T_NAME_QUALIFIED] = true; |
|
self::$namespace[T_NAME_RELATIVE] = true; |
|
$identifier[T_NAME_FULLY_QUALIFIED] = true; |
|
$identifier[T_NAME_QUALIFIED] = true; |
|
$identifier[T_NAME_RELATIVE] = true; |
|
} |
|
|
|
/** @psalm-var list<PhpToken> */ |
|
$tokens = \token_get_all($source); |
|
$cursor = 1; |
|
$function_calls = []; |
|
|
|
// Performance optimization preventing backwards loops |
|
/** @psalm-var array<PhpToken|null> */ |
|
$prev_tokens = [null, null, null]; |
|
|
|
if (\is_array($function)) { |
|
$class = \explode('\\', $function[0]); |
|
$class = \strtolower(\end($class)); |
|
$function = \strtolower($function[1]); |
|
} else { |
|
$class = null; |
|
/** |
|
* @psalm-suppress RedundantFunctionCallGivenDocblockType |
|
*/ |
|
$function = \strtolower($function); |
|
} |
|
|
|
// Loop through tokens |
|
foreach ($tokens as $index => $token) { |
|
if (!\is_array($token)) { |
|
continue; |
|
} |
|
|
|
// Count newlines for line number instead of using $token[2] |
|
// since certain situations (String tokens after whitespace) may |
|
// not have the correct line number unless you do this manually |
|
$cursor += \substr_count($token[1], "\n"); |
|
if ($cursor > $line) { |
|
break; |
|
} |
|
|
|
// Store the last real tokens for later |
|
if (isset(self::$ignore[$token[0]])) { |
|
continue; |
|
} |
|
|
|
$prev_tokens = [$prev_tokens[1], $prev_tokens[2], $token]; |
|
|
|
// Check if it's the right type to be the function we're looking for |
|
if (!isset(self::$namespace[$token[0]])) { |
|
continue; |
|
} |
|
|
|
$ns = \explode('\\', \strtolower($token[1])); |
|
|
|
if (\end($ns) !== $function) { |
|
continue; |
|
} |
|
|
|
// Check if it's a function call |
|
$nextReal = self::realTokenIndex($tokens, $index); |
|
if (!isset($nextReal, $tokens[$nextReal]) || '(' !== $tokens[$nextReal]) { |
|
continue; |
|
} |
|
|
|
// Check if it matches the signature |
|
if (null === $class) { |
|
if ($prev_tokens[1] && isset(self::$classcalls[$prev_tokens[1][0]])) { |
|
continue; |
|
} |
|
} else { |
|
if (!$prev_tokens[1] || T_DOUBLE_COLON !== $prev_tokens[1][0]) { |
|
continue; |
|
} |
|
|
|
if (!$prev_tokens[0] || !isset(self::$namespace[$prev_tokens[0][0]])) { |
|
continue; |
|
} |
|
|
|
// All self::$namespace tokens are T_ constants |
|
/** @psalm-var PhpTokenArray $prev_tokens[0] */ |
|
$ns = \explode('\\', \strtolower($prev_tokens[0][1])); |
|
|
|
if (\end($ns) !== $class) { |
|
continue; |
|
} |
|
} |
|
|
|
$inner_cursor = $cursor; |
|
$depth = 1; // The depth respective to the function call |
|
$offset = $nextReal + 1; // The start of the function call |
|
$instring = false; // Whether we're in a string or not |
|
$realtokens = false; // Whether the current scope contains anything meaningful or not |
|
$paramrealtokens = false; // Whether the current parameter contains anything meaningful |
|
$params = []; // All our collected parameters |
|
$shortparam = []; // The short version of the parameter |
|
$param_start = $offset; // The distance to the start of the parameter |
|
|
|
// Loop through the following tokens until the function call ends |
|
while (isset($tokens[$offset])) { |
|
$token = $tokens[$offset]; |
|
|
|
// Ensure that the $inner_cursor is correct and |
|
// that $token is either a T_ constant or a string |
|
if (\is_array($token)) { |
|
$inner_cursor += \substr_count($token[1], "\n"); |
|
} |
|
|
|
if (!isset(self::$ignore[$token[0]]) && !isset($down[$token[0]])) { |
|
$paramrealtokens = $realtokens = true; |
|
} |
|
|
|
// If it's a token that makes us to up a level, increase the depth |
|
if (isset($up[$token[0]])) { |
|
if (1 === $depth) { |
|
$shortparam[] = $token; |
|
$realtokens = false; |
|
} |
|
|
|
++$depth; |
|
} elseif (isset($down[$token[0]])) { |
|
--$depth; |
|
|
|
// If this brings us down to the parameter level, and we've had |
|
// real tokens since going up, fill the $shortparam with an ellipsis |
|
if (1 === $depth) { |
|
if ($realtokens) { |
|
$shortparam[] = '...'; |
|
} |
|
$shortparam[] = $token; |
|
} |
|
} elseif ('"' === $token[0]) { |
|
// Strings use the same symbol for up and down, but we can |
|
// only ever be inside one string, so just use a bool for that |
|
if ($instring) { |
|
--$depth; |
|
if (1 === $depth) { |
|
$shortparam[] = '...'; |
|
} |
|
} else { |
|
++$depth; |
|
} |
|
|
|
$instring = !$instring; |
|
|
|
$shortparam[] = '"'; |
|
} elseif (1 === $depth) { |
|
if (',' === $token[0]) { |
|
$params[] = [ |
|
'full' => \array_slice($tokens, $param_start, $offset - $param_start), |
|
'short' => $shortparam, |
|
]; |
|
$shortparam = []; |
|
$paramrealtokens = false; |
|
$param_start = $offset + 1; |
|
} elseif (T_CONSTANT_ENCAPSED_STRING === $token[0] && \strlen($token[1]) > 2) { |
|
$shortparam[] = $token[1][0].'...'.$token[1][0]; |
|
} else { |
|
$shortparam[] = $token; |
|
} |
|
} |
|
|
|
// Depth has dropped to 0 (So we've hit the closing paren) |
|
if ($depth <= 0) { |
|
if ($paramrealtokens) { |
|
$params[] = [ |
|
'full' => \array_slice($tokens, $param_start, $offset - $param_start), |
|
'short' => $shortparam, |
|
]; |
|
} |
|
|
|
break; |
|
} |
|
|
|
++$offset; |
|
} |
|
|
|
// If we're not passed (or at) the line at the end |
|
// of the function call, we're too early so skip it |
|
if ($inner_cursor < $line) { |
|
continue; |
|
} |
|
|
|
// Format the final output parameters |
|
foreach ($params as &$param) { |
|
$name = self::tokensFormatted($param['short']); |
|
|
|
$expression = false; |
|
foreach ($name as $token) { |
|
if (self::tokenIsOperator($token)) { |
|
$expression = true; |
|
break; |
|
} |
|
} |
|
|
|
$param = [ |
|
'name' => self::tokensToString($name), |
|
'path' => self::tokensToString(self::tokensTrim($param['full'])), |
|
'expression' => $expression, |
|
]; |
|
} |
|
|
|
// Skip first-class callables |
|
/** @psalm-var list<array{name: string, path: string, expression: bool}> $params */ |
|
if (KINT_PHP81 && 1 === \count($params) && '...' === \reset($params)['path']) { |
|
continue; |
|
} |
|
|
|
// Get the modifiers |
|
--$index; |
|
|
|
while (isset($tokens[$index])) { |
|
if (!isset(self::$ignore[$tokens[$index][0]]) && !isset($identifier[$tokens[$index][0]])) { |
|
break; |
|
} |
|
|
|
--$index; |
|
} |
|
|
|
$mods = []; |
|
|
|
while (isset($tokens[$index])) { |
|
if (isset(self::$ignore[$tokens[$index][0]])) { |
|
--$index; |
|
continue; |
|
} |
|
|
|
if (isset($modifiers[$tokens[$index][0]])) { |
|
$mods[] = $tokens[$index]; |
|
--$index; |
|
continue; |
|
} |
|
|
|
break; |
|
} |
|
|
|
$function_calls[] = [ |
|
'parameters' => $params, |
|
'modifiers' => $mods, |
|
]; |
|
} |
|
|
|
return $function_calls; |
|
} |
|
|
|
/** |
|
* @psalm-param PhpToken[] $tokens |
|
*/ |
|
private static function realTokenIndex(array $tokens, int $index): ?int |
|
{ |
|
++$index; |
|
|
|
while (isset($tokens[$index])) { |
|
if (!isset(self::$ignore[$tokens[$index][0]])) { |
|
return $index; |
|
} |
|
|
|
++$index; |
|
} |
|
|
|
return null; |
|
} |
|
|
|
/** |
|
* We need a separate method to check if tokens are operators because we |
|
* occasionally add "..." to short parameter versions. If we simply check |
|
* for `$token[0]` then "..." will incorrectly match the "." operator. |
|
* |
|
* @psalm-param PhpToken $token The token to check |
|
*/ |
|
private static function tokenIsOperator($token): bool |
|
{ |
|
return '...' !== $token && isset(self::$operator[$token[0]]); |
|
} |
|
|
|
/** |
|
* @psalm-param PhpToken[] $tokens |
|
*/ |
|
private static function tokensToString(array $tokens): string |
|
{ |
|
$out = ''; |
|
|
|
foreach ($tokens as $token) { |
|
if (\is_string($token)) { |
|
$out .= $token; |
|
} else { |
|
$out .= $token[1]; |
|
} |
|
} |
|
|
|
return $out; |
|
} |
|
|
|
/** |
|
* @psalm-param PhpToken[] $tokens |
|
*/ |
|
private static function tokensTrim(array $tokens): array |
|
{ |
|
foreach ($tokens as $index => $token) { |
|
if (isset(self::$ignore[$token[0]])) { |
|
unset($tokens[$index]); |
|
} else { |
|
break; |
|
} |
|
} |
|
|
|
$tokens = \array_reverse($tokens); |
|
|
|
foreach ($tokens as $index => $token) { |
|
if (isset(self::$ignore[$token[0]])) { |
|
unset($tokens[$index]); |
|
} else { |
|
break; |
|
} |
|
} |
|
|
|
return \array_reverse($tokens); |
|
} |
|
|
|
/** |
|
* @psalm-param PhpToken[] $tokens |
|
* |
|
* @psalm-return PhpToken[] |
|
*/ |
|
private static function tokensFormatted(array $tokens): array |
|
{ |
|
$tokens = self::tokensTrim($tokens); |
|
|
|
$space = false; |
|
$attribute = false; |
|
// Keep space between "strip" symbols for different behavior for matches or closures |
|
// Normally we want to strip spaces between strip tokens: $x{...}[...] |
|
// However with closures and matches we don't: function (...) {...} |
|
$ignorestrip = false; |
|
$output = []; |
|
$last = null; |
|
|
|
if (T_FUNCTION === $tokens[0][0] || |
|
(KINT_PHP74 && T_FN === $tokens[0][0]) || |
|
(KINT_PHP80 && T_MATCH === $tokens[0][0]) |
|
) { |
|
$ignorestrip = true; |
|
} |
|
|
|
foreach ($tokens as $index => $token) { |
|
if (isset(self::$ignore[$token[0]])) { |
|
if ($space) { |
|
continue; |
|
} |
|
|
|
$next = self::realTokenIndex($tokens, $index); |
|
if (null === $next) { |
|
// This should be impossible, since we always call tokensTrim first |
|
break; // @codeCoverageIgnore |
|
} |
|
$next = $tokens[$next]; |
|
|
|
/** @psalm-var PhpToken $last */ |
|
if ($attribute && ']' === $last[0]) { |
|
$attribute = false; |
|
} elseif (!$ignorestrip && isset(self::$strip[$last[0]]) && !self::tokenIsOperator($next)) { |
|
continue; |
|
} |
|
|
|
if (!$ignorestrip && isset(self::$strip[$next[0]]) && $last && !self::tokenIsOperator($last)) { |
|
continue; |
|
} |
|
|
|
$token = ' '; |
|
$space = true; |
|
} else { |
|
if (KINT_PHP80 && $last && T_ATTRIBUTE == $last[0]) { |
|
$attribute = true; |
|
} |
|
|
|
$space = false; |
|
$last = $token; |
|
} |
|
|
|
$output[] = $token; |
|
} |
|
|
|
return $output; |
|
} |
|
}
|
|
|