*/ public function register() { return [\T_VARIABLE]; } /** * Processes this test, when one of its tokens is encountered. * * @since 7.1.2 * * @param \PHP_CodeSniffer\Files\File $phpcsFile The file being scanned. * @param int $stackPtr The position of the current token * in the stack passed in $tokens. * * @return void */ public function process(File $phpcsFile, $stackPtr) { if (ScannedCode::shouldRunOnOrAbove('7.0') === false) { return; } $tokens = $phpcsFile->getTokens(); // Verify that the next token is a square open bracket. If not, bow out. $nextToken = $phpcsFile->findNext(Tokens::$emptyTokens, ($stackPtr + 1), null, true); if ($nextToken === false || $tokens[$nextToken]['code'] !== \T_OPEN_SQUARE_BRACKET || isset($tokens[$nextToken]['bracket_closer']) === false ) { return; } // The previous non-empty token has to be a $, -> or ::. $prevToken = $phpcsFile->findPrevious(Tokens::$emptyTokens, ($stackPtr - 1), null, true); if (isset(Collections::objectOperators()[$tokens[$prevToken]['code']]) === false && $tokens[$prevToken]['code'] !== \T_DOLLAR ) { return; } // For static object calls, it only applies when this is a function call. if ($tokens[$prevToken]['code'] === \T_DOUBLE_COLON) { $hasBrackets = $tokens[$nextToken]['bracket_closer']; while (($hasBrackets = $phpcsFile->findNext(Tokens::$emptyTokens, ($hasBrackets + 1), null, true)) !== false) { if ($tokens[$hasBrackets]['code'] === \T_OPEN_SQUARE_BRACKET) { if (isset($tokens[$hasBrackets]['bracket_closer'])) { $hasBrackets = $tokens[$hasBrackets]['bracket_closer']; continue; } else { // Live coding. return; } } elseif ($tokens[$hasBrackets]['code'] === \T_OPEN_PARENTHESIS) { // Caught! break; } else { // Not a function call, so bow out. return; } } } $phpcsFile->addError( 'Indirect access to variables, properties and methods will be evaluated strictly in left-to-right order since PHP 7.0. Use curly braces to remove ambiguity.', $stackPtr, 'Found' ); } }