Skip to main content
Drupal API
User account menu
  • Log in

Breadcrumb

  1. Drupal Core 11.1.x

MultipleStatementAlignmentSniff.php

Same filename in this branch
  1. 11.1.x vendor/squizlabs/php_codesniffer/src/Standards/Generic/Sniffs/Formatting/MultipleStatementAlignmentSniff.php

Namespace

Drupal\Sniffs\Formatting

File

vendor/drupal/coder/coder_sniffer/Drupal/Sniffs/Formatting/MultipleStatementAlignmentSniff.php

View source
<?php


/**
 * \Drupal\Sniffs\Formatting\MultipleStatementAlignmentSniff.
 *
 * @category PHP
 * @package  PHP_CodeSniffer
 * @link     http://pear.php.net/package/PHP_CodeSniffer
 */
namespace Drupal\Sniffs\Formatting;

use PHP_CodeSniffer\Standards\Generic\Sniffs\Formatting\MultipleStatementAlignmentSniff as GenericMultipleStatementAlignmentSniff;
use PHP_CodeSniffer\Util\Tokens;

/**
 * Checks alignment of multiple assignments.Largely copied from
 * \PHP_CodeSniffer\Standards\Generic\Sniffs\Formatting\MultipleStatementAlignmentSniff
 * but also allows multiple single space assignments.
 *
 * @category PHP
 * @package  PHP_CodeSniffer
 * @link     http://pear.php.net/package/PHP_CodeSniffer
 */
class MultipleStatementAlignmentSniff extends GenericMultipleStatementAlignmentSniff {
    
    /**
     * If true, an error will be thrown; otherwise a warning.
     *
     * @var boolean
     */
    public $error = true;
    
    /**
     * Processes this test, when one of its tokens is encountered.
     *
     * @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.
     * @param int                         $end       The token where checking should end.
     *                                               If NULL, the entire file will be checked.
     *
     * @return int
     */
    public function checkAlignment($phpcsFile, $stackPtr, $end = null) {
        $tokens = $phpcsFile->getTokens();
        $assignments = [];
        $prevAssign = null;
        $lastLine = $tokens[$stackPtr]['line'];
        $maxPadding = null;
        $stopped = null;
        $lastCode = $stackPtr;
        $lastSemi = null;
        $arrayEnd = null;
        if ($end === null) {
            $end = $phpcsFile->numTokens;
        }
        $find = Tokens::$assignmentTokens;
        unset($find[T_DOUBLE_ARROW]);
        $scopes = Tokens::$scopeOpeners;
        unset($scopes[T_CLOSURE]);
        unset($scopes[T_ANON_CLASS]);
        unset($scopes[T_OBJECT]);
        for ($assign = $stackPtr; $assign < $end; $assign++) {
            if ($tokens[$assign]['level'] < $tokens[$stackPtr]['level']) {
                // Statement is in a different context, so the block is over.
                break;
            }
            if (isset($scopes[$tokens[$assign]['code']]) === true && isset($tokens[$assign]['scope_opener']) === true && $tokens[$assign]['level'] === $tokens[$stackPtr]['level']) {
                break;
            }
            if ($assign === $arrayEnd) {
                $arrayEnd = null;
            }
            if (isset($find[$tokens[$assign]['code']]) === false) {
                // A blank line indicates that the assignment block has ended.
                if (isset(Tokens::$emptyTokens[$tokens[$assign]['code']]) === false && $tokens[$assign]['line'] - $tokens[$lastCode]['line'] > 1 && $tokens[$assign]['level'] === $tokens[$stackPtr]['level'] && $arrayEnd === null) {
                    break;
                }
                if ($tokens[$assign]['code'] === T_CLOSE_TAG) {
                    // Breaking out of PHP ends the assignment block.
                    break;
                }
                if ($tokens[$assign]['code'] === T_OPEN_SHORT_ARRAY && isset($tokens[$assign]['bracket_closer']) === true) {
                    $arrayEnd = $tokens[$assign]['bracket_closer'];
                }
                if ($tokens[$assign]['code'] === T_ARRAY && isset($tokens[$assign]['parenthesis_opener']) === true && isset($tokens[$tokens[$assign]['parenthesis_opener']]['parenthesis_closer']) === true) {
                    $arrayEnd = $tokens[$tokens[$assign]['parenthesis_opener']]['parenthesis_closer'];
                }
                if (isset(Tokens::$emptyTokens[$tokens[$assign]['code']]) === false) {
                    $lastCode = $assign;
                    if ($tokens[$assign]['code'] === T_SEMICOLON) {
                        if ($tokens[$assign]['conditions'] === $tokens[$stackPtr]['conditions']) {
                            if ($lastSemi !== null && $prevAssign !== null && $lastSemi > $prevAssign) {
                                // This statement did not have an assignment operator in it.
                                break;
                            }
                            else {
                                $lastSemi = $assign;
                            }
                        }
                        else {
                            if ($tokens[$assign]['level'] < $tokens[$stackPtr]['level']) {
                                // Statement is in a different context, so the block is over.
                                break;
                            }
                        }
                    }
                }
                
                //end if
                continue;
            }
            else {
                if ($assign !== $stackPtr && $tokens[$assign]['line'] === $lastLine) {
                    // Skip multiple assignments on the same line. We only need to
                    // try and align the first assignment.
                    continue;
                }
            }
            
            //end if
            if ($assign !== $stackPtr) {
                if ($tokens[$assign]['level'] > $tokens[$stackPtr]['level']) {
                    // Has to be nested inside the same conditions as the first assignment.
                    // We've gone one level down, so process this new block.
                    $assign = $this->checkAlignment($phpcsFile, $assign);
                    $lastCode = $assign;
                    continue;
                }
                else {
                    if ($tokens[$assign]['level'] < $tokens[$stackPtr]['level']) {
                        // We've gone one level up, so the block we are processing is done.
                        break;
                    }
                    else {
                        if ($arrayEnd !== null) {
                            // Assignments inside arrays are not part of
                            // the original block, so process this new block.
                            $assign = $this->checkAlignment($phpcsFile, $assign, $arrayEnd) - 1;
                            $arrayEnd = null;
                            $lastCode = $assign;
                            continue;
                        }
                    }
                }
                // Make sure it is not assigned inside a condition (eg. IF, FOR).
                if (isset($tokens[$assign]['nested_parenthesis']) === true) {
                    foreach ($tokens[$assign]['nested_parenthesis'] as $start => $end) {
                        if (isset($tokens[$start]['parenthesis_owner']) === true) {
                            break 2;
                        }
                    }
                }
            }
            
            //end if
            $var = $phpcsFile->findPrevious(Tokens::$emptyTokens, $assign - 1, null, true);
            // Make sure we wouldn't break our max padding length if we
            // aligned with this statement, or they wouldn't break the max
            // padding length if they aligned with us.
            $varEnd = $tokens[$var + 1]['column'];
            $assignLen = $tokens[$assign]['length'];
            if ($assign !== $stackPtr) {
                if ($prevAssign === null) {
                    // Processing an inner block but no assignments found.
                    break;
                }
                if ($varEnd + 1 > $assignments[$prevAssign]['assign_col']) {
                    $padding = 1;
                    $assignColumn = $varEnd + 1;
                }
                else {
                    $padding = $assignments[$prevAssign]['assign_col'] - $varEnd + $assignments[$prevAssign]['assign_len'] - $assignLen;
                    if ($padding <= 0) {
                        $padding = 1;
                    }
                    if ($padding > $this->maxPadding) {
                        $stopped = $assign;
                        break;
                    }
                    $assignColumn = $varEnd + $padding;
                }
                
                //end if
                if ($assignColumn + $assignLen > $assignments[$maxPadding]['assign_col'] + $assignments[$maxPadding]['assign_len']) {
                    $newPadding = $varEnd - $assignments[$maxPadding]['var_end'] + $assignLen - $assignments[$maxPadding]['assign_len'] + 1;
                    if ($newPadding > $this->maxPadding) {
                        $stopped = $assign;
                        break;
                    }
                    else {
                        // New alignment settings for previous assignments.
                        foreach ($assignments as $i => $data) {
                            if ($i === $assign) {
                                break;
                            }
                            $newPadding = $varEnd - $data['var_end'] + $assignLen - $data['assign_len'] + 1;
                            $assignments[$i]['expected'] = $newPadding;
                            $assignments[$i]['assign_col'] = $data['var_end'] + $newPadding;
                        }
                        $padding = 1;
                        $assignColumn = $varEnd + 1;
                    }
                }
                else {
                    if ($padding > $assignments[$maxPadding]['expected']) {
                        $maxPadding = $assign;
                    }
                }
                
                //end if
            }
            else {
                $padding = 1;
                $assignColumn = $varEnd + 1;
                $maxPadding = $assign;
            }
            
            //end if
            $found = 0;
            if ($tokens[$var + 1]['code'] === T_WHITESPACE) {
                $found = $tokens[$var + 1]['length'];
                if ($found === 0) {
                    // This means a newline was found.
                    $found = 1;
                }
            }
            $assignments[$assign] = [
                'var_end' => $varEnd,
                'assign_len' => $assignLen,
                'assign_col' => $assignColumn,
                'expected' => $padding,
                'found' => $found,
            ];
            $lastLine = $tokens[$assign]['line'];
            $prevAssign = $assign;
        }
        
        //end for
        if (empty($assignments) === true) {
            return $stackPtr;
        }
        // If there is at least one assignment that uses more than two spaces then it
        // appears that the assignments should all be aligned right.
        $alignRight = false;
        foreach ($assignments as $assignment => $data) {
            if ($data['found'] > 2) {
                $alignRight = true;
                break;
            }
        }
        $numAssignments = count($assignments);
        $errorGenerated = false;
        foreach ($assignments as $assignment => $data) {
            // Missing space is already covered by
            // Drupal.WhiteSpace.OperatorSpacing.NoSpaceBefore.
            if ($data['found'] === 0) {
                continue;
            }
            if ($alignRight === false && $data['found'] !== $data['expected']) {
                $data['expected'] = 1;
            }
            if ($data['found'] === $data['expected']) {
                continue;
            }
            $expectedText = $data['expected'] . ' space';
            if ($data['expected'] !== 1) {
                $expectedText .= 's';
            }
            if ($data['found'] === null) {
                $foundText = 'a new line';
            }
            else {
                $foundText = $data['found'] . ' space';
                if ($data['found'] !== 1) {
                    $foundText .= 's';
                }
            }
            if ($numAssignments === 1) {
                $type = 'Incorrect';
                $error = 'Equals sign not aligned correctly; expected %s but found %s';
            }
            else {
                $type = 'NotSame';
                $error = 'Equals sign not aligned with surrounding assignments; expected %s but found %s';
            }
            $errorData = [
                $expectedText,
                $foundText,
            ];
            if ($this->error === true) {
                $fix = $phpcsFile->addFixableError($error, $assignment, $type, $errorData);
            }
            else {
                $fix = $phpcsFile->addFixableWarning($error, $assignment, $type . 'Warning', $errorData);
            }
            $errorGenerated = true;
            if ($fix === true && $data['found'] !== null) {
                $newContent = str_repeat(' ', $data['expected']);
                if ($data['found'] === 0) {
                    $phpcsFile->fixer
                        ->addContentBefore($assignment, $newContent);
                }
                else {
                    $phpcsFile->fixer
                        ->replaceToken($assignment - 1, $newContent);
                }
            }
        }
        
        //end foreach
        if ($numAssignments > 1) {
            if ($errorGenerated === true) {
                $phpcsFile->recordMetric($stackPtr, 'Adjacent assignments aligned', 'no');
            }
            else {
                $phpcsFile->recordMetric($stackPtr, 'Adjacent assignments aligned', 'yes');
            }
        }
        if ($stopped !== null) {
            return $this->checkAlignment($phpcsFile, $stopped);
        }
        else {
            return $assign;
        }
    }
    
    //end checkAlignment()

}

//end class

Classes

Title Deprecated Summary
MultipleStatementAlignmentSniff Checks alignment of multiple assignments.Largely copied from \PHP_CodeSniffer\Standards\Generic\Sniffs\Formatting\MultipleStatementAlignmentSniff but also allows multiple single space assignments.
RSS feed
Powered by Drupal