You are here

class GlobalFunctionSniff in Coder 8.3.x

Same name and namespace in other branches
  1. 8.3 coder_sniffer/DrupalPractice/Sniffs/Objects/GlobalFunctionSniff.php \DrupalPractice\Sniffs\Objects\GlobalFunctionSniff
  2. 8.2 coder_sniffer/DrupalPractice/Sniffs/Objects/GlobalFunctionSniff.php \DrupalPractice\Sniffs\Objects\GlobalFunctionSniff

Checks that global functions like t() are not used in forms or controllers.

@category PHP @package PHP_CodeSniffer @link http://pear.php.net/package/PHP_CodeSniffer

Hierarchy

Expanded class hierarchy of GlobalFunctionSniff

File

coder_sniffer/DrupalPractice/Sniffs/Objects/GlobalFunctionSniff.php, line 23

Namespace

DrupalPractice\Sniffs\Objects
View source
class GlobalFunctionSniff extends FunctionCall {

  /**
   * List of global functions that should not be called.
   *
   * @var string[]
   */
  protected $functions = [
    'drupal_get_destination' => 'the "redirect.destination" service',
    'drupal_render' => 'the "renderer" service',
    'entity_load' => 'the "entity_type.manager" service',
    'file_load' => 'the "entity_type.manager" service',
    'format_date' => 'the "date.formatter" service',
    'node_load' => 'the "entity_type.manager" service',
    'node_load_multiple' => 'the "entity_type.manager" service',
    'node_type_load' => 'the "entity_type.manager" service',
    't' => '$this->t()',
    'taxonomy_term_load' => 'the "entity_type.manager" service',
    'taxonomy_vocabulary_load' => 'the "entity_type.manager" service',
    'user_load' => 'the "entity_type.manager" service',
    'user_role_load' => 'the "entity_type.manager" service',
  ];

  /**
   * List of global functions that are covered by traits.
   *
   * This is a subset of the global functions list. These functions can be
   * replaced by methods that are provided by the listed trait.
   *
   * @var string[]
   */
  protected $traitFunctions = [
    't' => '\\Drupal\\Core\\StringTranslation\\StringTranslationTrait',
  ];

  /**
   * 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.
   *
   * @return void|int
   */
  public function process(File $phpcsFile, $stackPtr) {
    $tokens = $phpcsFile
      ->getTokens();

    // Only run this sniff on Drupal 8+.
    if (Project::getCoreVersion($phpcsFile) < 8) {

      // No need to check this file again, mark it as done.
      return $phpcsFile->numTokens + 1;
    }

    // We just want to listen on function calls, nothing else.
    if ($this
      ->isFunctionCall($phpcsFile, $stackPtr) === false || empty($tokens[$stackPtr]['conditions']) === true || isset($this->functions[$tokens[$stackPtr]['content']]) === false) {
      return;
    }

    // Check that this statement is not in a static function.
    foreach ($tokens[$stackPtr]['conditions'] as $conditionPtr => $conditionCode) {
      if ($conditionCode === T_FUNCTION && $phpcsFile
        ->getMethodProperties($conditionPtr)['is_static'] === true) {
        return;
      }
    }

    // Check if the class extends another class and get the name of the class
    // that is extended.
    $classPtr = key($tokens[$stackPtr]['conditions']);
    if ($tokens[$classPtr]['code'] !== T_CLASS) {
      return;
    }
    if (isset($this->traitFunctions[$tokens[$stackPtr]['content']]) === false) {
      $extendsName = $phpcsFile
        ->findExtendedClassName($classPtr);

      // Check if the class implements ContainerInjectionInterface.
      $implementedInterfaceNames = $phpcsFile
        ->findImplementedInterfaceNames($classPtr);
      $canAccessContainer = !empty($implementedInterfaceNames) && in_array('ContainerInjectionInterface', $implementedInterfaceNames);
      if (($extendsName === false || in_array($extendsName, GlobalDrupalSniff::$baseClasses) === false) && Project::isServiceClass($phpcsFile, $classPtr) === false && $canAccessContainer === false) {
        return;
      }
      $warning = '%s() calls should be avoided in classes, use dependency injection and %s instead';
      $data = [
        $tokens[$stackPtr]['content'],
        $this->functions[$tokens[$stackPtr]['content']],
      ];
    }
    else {
      $warning = '%s() calls should be avoided in classes, use %s and %s instead';
      $data = [
        $tokens[$stackPtr]['content'],
        $this->traitFunctions[$tokens[$stackPtr]['content']],
        $this->functions[$tokens[$stackPtr]['content']],
      ];
    }

    //end if
    $phpcsFile
      ->addWarning($warning, $stackPtr, 'GlobalFunction', $data);
  }

}

Members

Namesort descending Modifiers Type Description Overrides
FunctionCall::$arguments protected property Internal cache to save the calculated arguments of the function call.
FunctionCall::$closeBracket protected property The token position of the closing bracket of the function call.
FunctionCall::$functionCall protected property The token position of the function call.
FunctionCall::$includeMethodCalls protected property Whether method invocations with the same function name should be processed, too. 1
FunctionCall::$openBracket protected property The token position of the opening bracket of the function call.
FunctionCall::$phpcsFile protected property The currently processed file.
FunctionCall::getArgument public function Returns start and end token for a given argument number.
FunctionCall::isFunctionCall protected function Checks if this is a function call.
FunctionCall::register public function Returns an array of tokens this test wants to listen for.
GlobalFunctionSniff::$functions protected property List of global functions that should not be called.
GlobalFunctionSniff::$traitFunctions protected property List of global functions that are covered by traits.
GlobalFunctionSniff::process public function Processes this test, when one of its tokens is encountered. Overrides FunctionCall::process