EntityFieldsViaMagicReflectionExtension.php
Namespace
mglaman\PHPStanDrupal\ReflectionFile
-
vendor/
mglaman/ phpstan-drupal/ src/ Reflection/ EntityFieldsViaMagicReflectionExtension.php
View source
<?php
namespace mglaman\PHPStanDrupal\Reflection;
use LogicException;
use PHPStan\Reflection\ClassReflection;
use PHPStan\Reflection\PropertiesClassReflectionExtension;
use PHPStan\Reflection\PropertyReflection;
use PHPStan\TrinaryLogic;
use PHPStan\Type\ObjectType;
use function array_key_exists;
/**
* Allows field access via magic methods
*
* See \Drupal\Core\Entity\ContentEntityBase::__get and ::__set.
*
* @todo split into Entity and FieldItem specifics.
*/
class EntityFieldsViaMagicReflectionExtension implements PropertiesClassReflectionExtension {
public function hasProperty(ClassReflection $classReflection, string $propertyName) : bool {
// @todo Have this run after PHPStan\Reflection\Annotations\AnnotationsPropertiesClassReflectionExtension
// We should not have to check for the property tags if we could get this to run after PHPStan's
// existing annotation property reflection.
if ($classReflection->hasNativeProperty($propertyName) || array_key_exists($propertyName, $classReflection->getPropertyTags())) {
// Let other parts of PHPStan handle this.
return false;
}
foreach ($classReflection->getAncestors() as $ancestor) {
if (array_key_exists($propertyName, $ancestor->getPropertyTags())) {
return false;
}
}
// We need to find a way to parse the entity annotation so that at the minimum the `entity_keys` are
// supported. The real fix is Drupal developers _really_ need to start writing @property definitions in the
// class doc if they don't get `get` methods.
if ($classReflection->implementsInterface('Drupal\\Core\\Entity\\ContentEntityInterface')) {
// @todo revisit if it's a good idea to be true.
// Content entities have magical __get... so it is kind of true.
return true;
}
if (self::classObjectIsSuperOfInterface($classReflection->getName(), self::getFieldItemListInterfaceObject())
->yes()) {
return FieldItemListPropertyReflection::canHandleProperty($classReflection, $propertyName);
}
return false;
}
public function getProperty(ClassReflection $classReflection, string $propertyName) : PropertyReflection {
if ($classReflection->implementsInterface('Drupal\\Core\\Entity\\EntityInterface')) {
return new EntityFieldReflection($classReflection, $propertyName);
}
if (self::classObjectIsSuperOfInterface($classReflection->getName(), self::getFieldItemListInterfaceObject())
->yes()) {
return new FieldItemListPropertyReflection($classReflection, $propertyName);
}
throw new LogicException($classReflection->getName() . "::{$propertyName} should be handled earlier.");
}
public static function classObjectIsSuperOfInterface(string $name, ObjectType $interfaceObject) : TrinaryLogic {
return $interfaceObject->isSuperTypeOf(new ObjectType($name));
}
protected static function getFieldItemListInterfaceObject() : ObjectType {
return new ObjectType('Drupal\\Core\\Field\\FieldItemListInterface');
}
}
Classes
Title | Deprecated | Summary |
---|---|---|
EntityFieldsViaMagicReflectionExtension | Allows field access via magic methods |