Skip to content

Commit

Permalink
Rely on reflection to access null properties
Browse files Browse the repository at this point in the history
  • Loading branch information
nicolas-grekas committed Oct 31, 2024
1 parent 494f010 commit 5e1b6fe
Showing 1 changed file with 47 additions and 6 deletions.
53 changes: 47 additions & 6 deletions src/Extension/CoreExtension.php
Original file line number Diff line number Diff line change
Expand Up @@ -1624,11 +1624,13 @@ public static function getAttribute(Environment $env, Source $source, $object, $
{
// array
if (Template::METHOD_CALL !== $type) {
$arrayItem = \is_bool($item) || \is_float($item) ? (int) $item : $item;
$arrayItem = \is_bool($item) || \is_float($item) ? (int) $item : $item = (string) $item;

if (((\is_array($object) || $object instanceof \ArrayObject) && (isset($object[$arrayItem]) || \array_key_exists($arrayItem, (array) $object)))
|| ($object instanceof \ArrayAccess && isset($object[$arrayItem]))
) {
if (match (true) {
\is_array($object) => \array_key_exists($arrayItem, $object),
$object instanceof \ArrayAccess => $object->offsetExists($arrayItem),
default => false,
}) {
if ($isDefinedTest) {
return true;
}
Expand Down Expand Up @@ -1697,7 +1699,23 @@ public static function getAttribute(Environment $env, Source $source, $object, $

// object property
if (Template::METHOD_CALL !== $type) {
if (isset($object->$item) || \array_key_exists((string) $item, (array) $object)) {
static $propertyCheckers = [];

if (isset($object->$item)
|| ($propertyCheckers[$object::class][$item] ??= self::getPropertyChecker($object::class, $item))($object, $item)
) {
if ($isDefinedTest) {
return true;
}

if ($sandboxed) {
$env->getExtension(SandboxExtension::class)->checkPropertyAllowed($object, $item, $lineno, $source);
}

return $object->$item;
}

if ($object instanceof \DateTimeInterface && \in_array($item, ['date', 'timezone', 'timezone_type'], true)) {
if ($isDefinedTest) {
return true;
}
Expand All @@ -1706,7 +1724,7 @@ public static function getAttribute(Environment $env, Source $source, $object, $
$env->getExtension(SandboxExtension::class)->checkPropertyAllowed($object, $item, $lineno, $source);
}

return isset($object->$item) ? $object->$item : ((array) $object)[(string) $item];
return ((array) $object)[$item];
}

if (\defined($object::class.'::'.$item)) {
Expand Down Expand Up @@ -2055,4 +2073,27 @@ public static function parseAttributeFunction(Parser $parser, Node $fakeNode, $a

return new GetAttrExpression($args[0], $args[1], $args[2] ?? null, Template::ANY_CALL, $line);
}

private static function getPropertyChecker(string $class, string $property): \Closure
{
static $classReflectors = [];

$class = $classReflectors[$class] ??= new \ReflectionClass($class);

if (!$class->hasProperty($property)) {
static $propertyExists;

return $propertyExists ??= \Closure::fromCallable('property_exists');
}

$property = $class->getProperty($property);

if (!$property->isPublic()) {
static $false;

return $false ??= static fn () => false;
}

return static fn ($object) => $property->isInitialized($object);
}
}

0 comments on commit 5e1b6fe

Please sign in to comment.