Extension for PHPStan to allow analysis of Drupal code.
PHPStan is able to discover symbols by using autoloading provided by Composer. However, Drupal does not provide autoloading information for modules and themes. This project registers those namespaces so that PHPStan can properly discover symbols in your Drupal code base automatically.
Note
With Drupal 11.2, Drupal core is now using PHPStan 2.0. The 1.x branch of phpstan-drupal will be supported until Drupal 10 loses security support when Drupal 12 is released.
When you are using phpstan/extension-installer, phpstan.neon will be automatically included.
Manual installation
If you don't want to use phpstan/extension-installer, include extension.neon in your project's PHPStan config:
includes:
- vendor/mglaman/phpstan-drupal/extension.neon
To include Drupal specific analysis rules, include this file:
includes:
- vendor/mglaman/phpstan-drupal/rules.neon
Ask for assistance in the discussions or #phpstan channel on Drupal Slack.
To exclude tests from analysis, add the following parameter
parameters:
excludePaths:
- *Test.php
- *TestBase.php
This project depends on phpstan/phpstan-deprecation-rules which adds deprecation rules. We provide Drupal-specific
deprecated scope resolvers.
To only handle deprecation testing, use a phpstan.neon like this:
parameters:
customRulesetUsed: true
reportUnmatchedIgnoredErrors: false
# Ignore phpstan-drupal extension's rules.
ignoreErrors:
- '#\Drupal calls should be avoided in classes, use dependency injection instead#'
- '#Plugin definitions cannot be altered.#'
- '#Missing cache backend declaration for performance.#'
- '#Plugin manager has cache backend specified but does not declare cache tags.#'
includes:
- vendor/mglaman/phpstan-drupal/extension.neon
- vendor/phpstan/phpstan-deprecation-rules/rules.neon
To disable deprecation rules while using phpstan/extension-installer, you can do the following:
{
"extra": {
"phpstan/extension-installer": {
"ignore": [
"phpstan/phpstan-deprecation-rules"
]
}
}
}See the extension-installer documentation for more information: https://github.com/phpstan/extension-installer#ignoring-a-particular-extension
These rules are disabled by default to avoid unexpected failures in a patch release. They graduate to the default ruleset in a future minor version. Include bleedingEdge.neon to enable all of them at once, or enable them individually:
parameters:
drupal:
rules:
# Enforces that OOP hook implementations using the Hook attribute have the
# correct method signature for hook_form_alter, hook_form_FORM_ID_alter, etc.
# Requires Drupal 10.3+ (Hook attribute).
hookRules: true
# Flags non-abstract test classes whose names do not end with "Test".
testClassSuffixNameRule: true
# Flags properties that are private or read-only in classes using
# DependencySerializationTrait, which does not support them.
dependencySerializationTraitPropertyRule: true
# Flags calls to AccessResult static methods (::allowed(), ::forbidden(), etc.)
# whose argument type already makes the condition always true or always false.
accessResultConditionRule: true
# Flags addCacheableDependency() calls whose argument does not implement
# CacheableDependencyInterface.
cacheableDependencyRule: true
# Flags logger channel objects (from LoggerChannelFactoryInterface::get()) that
# are assigned to a property in a class using DependencySerializationTrait,
# which cannot serialize logger channels correctly.
loggerFromFactoryPropertyAssignmentRule: true
# Flags direct injection of EntityStorageInterface (or a subtype) into a
# constructor. Inject EntityTypeManagerInterface and call getStorage() instead.
entityStorageDirectInjectionRule: trueYou can disable the ClassExtendsInternalClassRule rule by adding the following to your phpstan.neon:
parameters:
drupal:
rules:
classExtendsInternalClassRule: falseYou can disable various extensions. This is useful when contributing to Drupal Core to improve its types.
parameters:
drupal:
extensions:
entityFieldsViaMagicReflection: true
entityFieldMethodsViaMagicReflection: true
entityQuery: true
entityRepository: true
stubFiles: trueBoth options are enabled by default.
bleedingEdge.neon enables all opt-in rules plus hook deprecation checks against .api.php files. New rules land here first before graduating to the default ruleset in a minor release.
includes:
- vendor/mglaman/phpstan-drupal/bleedingEdge.neonWhat it currently enables:
checkCoreDeprecatedHooksInApiFiles— reports hook implementations deprecated in Drupal core.api.phpfilescheckContribDeprecatedHooksInApiFiles— reports hook implementations deprecated in contrib module.api.phpfileshookRules— validates OOP hook method signatures (requires Drupal 10.3+)testClassSuffixNameRule— non-abstract test class names must end withTestdependencySerializationTraitPropertyRule— flags private or read-only properties in classes usingDependencySerializationTraitaccessResultConditionRule— flags always-true/always-falseAccessResultconditionscacheableDependencyRule— flagsaddCacheableDependency()calls with non-cacheable argumentsloggerFromFactoryPropertyAssignmentRule— flags logger channels assigned to properties in classes usingDependencySerializationTraitentityStorageDirectInjectionRule— flags direct injection of entity storage into a constructor; injectEntityTypeManagerInterfaceand callgetStorage()instead
Note
checkDeprecatedHooksInApiFiles is deprecated. Use checkCoreDeprecatedHooksInApiFiles and checkContribDeprecatedHooksInApiFiles instead.
TodoCommentWithIssueUrlRule is an opt-in rule for Drupal contrib CI pipelines. When running PHPStan as part of a GitLab merge request, it reports an error for any @todo comment that contains a drupal.org issue URL matching the current issue — for example:
// @todo Remove once https://drupal.org/i/3456789 is resolved.This prevents issue-specific TODOs from being accidentally merged without resolution.
The rule auto-detects the current issue NID from standard GitLab CI environment variables:
CI_MERGE_REQUEST_SOURCE_BRANCH_NAME(e.g.3456789-my-feature)CI_MERGE_REQUEST_SOURCE_PROJECT_PATH(e.g.issue/mymodule-3456789)
It is silent when neither variable is set, so it is safe to include in a shared config.
The rule is not registered by default. To enable it, add it to your project's phpstan.neon:
rules:
- mglaman\PHPStanDrupal\Rules\Drupal\TodoCommentWithIssueUrlRuleNote
When using the Drupal GitLab CI templates,
adding extra rules requires a custom phpstan.neon that includes the default configuration, since adding additional
rules is not supported directly through the template variables.
Both drupal.org/i/{nid} and drupal.org/project/{project}/issues/{nid} URL formats are recognized.
phpstan-drupal provides custom PHPDoc types that can be used to improve type safety in Drupal code.
The entity-type-id type represents a valid Drupal entity type ID string (e.g. 'node', 'user', 'taxonomy_term'). PHPStan will report an error when a constant string that is not a known entity type ID is passed where entity-type-id is expected.
Drupal coding standards require keeping PHPStan-specific types in @phpstan-param and @phpstan-return tags rather than in the standard @param and @return tags:
/**
* Loads an entity by its entity type ID and entity ID.
*
* @param string $entityTypeId
* The entity type ID.
* @param int|string $id
* The entity ID.
*
* @phpstan-param entity-type-id $entityTypeId
*/
public function loadEntity(string $entityTypeId, int|string $id): ?EntityInterface {
return $this->entityTypeManager->getStorage($entityTypeId)->load($id);
}For return types:
/**
* Returns the entity type ID.
*
* @return string
* The entity type ID.
*
* @phpstan-return entity-type-id
*/
public function getEntityTypeId(): string {
return $this->entityTypeId;
}Known entity type IDs are sourced from the drupal.entityMapping parameter. See Entity storage mappings for how to register custom entity types so their IDs are also recognized.
The EntityTypeManagerGetStorageDynamicReturnTypeExtension service helps map dynamic return types. This inspects the
passed entity type ID and tries to return a known storage class, besides the default EntityStorageInterface. The
default mapping can be found in extension.neon. For example:
parameters:
drupal:
entityMapping:
block:
class: Drupal\block\Entity\Block
storage: Drupal\Core\Config\Entity\ConfigEntityStorage
node:
class: Drupal\node\Entity\Node
storage: Drupal\node\NodeStorage
taxonomy_term:
class: Drupal\taxonomy\Entity\Term
storage: Drupal\taxonomy\TermStorage
user:
class: Drupal\user\Entity\User
storage: Drupal\user\UserStorage
To add support for custom entities, you may add the same definition in your project's phpstan.neon. See the following
example for adding a mapping for Search API:
parameters:
drupal:
entityMapping:
search_api_index:
class: Drupal\search_api\Entity\Index
storage: Drupal\search_api\Entity\SearchApiConfigEntityStorage
search_api_server:
class: Drupal\search_api\Entity\Server
storage: Drupal\search_api\Entity\SearchApiConfigEntityStorage
Similarly, the EntityStorageDynamicReturnTypeExtension service helps to determine the type of the entity which is
loaded, created etc.. when using an entity storage.
For instance when using
$node = \Drupal::entityTypeManager()->getStorage('node')->create(['type' => 'page', 'title' => 'foo']);It helps with knowing the type of the $node variable is Drupal\node\Entity\Node.
The default mapping can be found in extension.neon:
parameters:
drupal:
entityMapping:
block:
class: Drupal\block\Entity\Block
storage: Drupal\Core\Config\Entity\ConfigEntityStorage
node:
class: Drupal\node\Entity\Node
storage: Drupal\node\NodeStorage
taxonomy_term:
class: Drupal\taxonomy\Entity\Term
storage: Drupal\taxonomy\TermStorage
user:
class: Drupal\user\Entity\User
storage: Drupal\user\UserStorageTo add support for custom entities, you may add the same definition in your project's phpstan.neon likewise.
Contributed modules can provide their own mapping that can be automatically registered with a user's code base when
they use the phpstan/extension-installer. The extension installer scans installed package's composer.json for a
value in extra.phpstan. This will automatically bundle the defined include that contains an entity mapping
configuration.
For example, the Paragraphs module could have the following entity_mapping.neon file:
parameters:
drupal:
entityMapping:
paragraph:
class: Drupal\paragraphs\Entity\Paragraph
paragraphs_type:
class: Drupal\paragraphs\Entity\ParagraphsTypeThen in the composer.json for Paragraphs, the entity_mapping.neon would be provided as a PHPStan include
{
"name": "drupal/paragraphs",
"description": "Enables the creation of Paragraphs entities.",
"type": "drupal-module",
"license": "GPL-2.0-or-later",
"require": {
"drupal/entity_reference_revisions": "~1.3"
},
"extra": {
"phpstan": {
"includes": [
"entity_mapping.neon"
]
}
}
}
