Skip to content

Commit d5a1a0b

Browse files
committed
Fix ArrayAccess access, add access to public object properties
1 parent 21ad8ac commit d5a1a0b

File tree

3 files changed

+156
-18
lines changed

3 files changed

+156
-18
lines changed

README.md

+14
Original file line numberDiff line numberDiff line change
@@ -134,6 +134,20 @@ $render->renderTemplateString(
134134
);
135135
```
136136

137+
Also, public properties of objects may be accessed from the template:
138+
139+
```php
140+
$person = new stdClass();
141+
$person->name = 'Hans';
142+
143+
$render->renderTemplateString(
144+
'{{ person.name }}',
145+
[
146+
'person' => $person,
147+
]
148+
);
149+
```
150+
137151
By default, a used variable which is not provided will result in an `UndefinedSymbolException`. You can register a callback function via `onResolveError` to handle unresolved variable errors. The callback function must implement the signature `function (string $unresolvedVariable): string`. The provided `$unresolvedVariable` will contain the whole expression which failed to resolve (eg. `myUnresolvedVariable`, `myUnresolvedObject.render(var1, var2)`).
138152

139153
```php

src/Render.php

+40-14
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44

55
namespace PHPMicroTemplate;
66

7+
use ArrayAccess;
78
use PHPMicroTemplate\Exception\FileSystemException;
89
use PHPMicroTemplate\Exception\SyntaxErrorException;
910
use PHPMicroTemplate\Exception\UndefinedSymbolException;
@@ -238,20 +239,7 @@ protected function getValue(array $matches, array $variables)
238239
array_push($variablePath, ...explode('.', trim($matches['nestedVariable'], '.')));
239240
}
240241

241-
foreach ($variablePath as $variable) {
242-
// first check via isset for faster lookup
243-
if (!isset($resolved[$variable]) && !array_key_exists($variable, $resolved)) {
244-
if ($this->resolveErrorCallback) {
245-
return ($this->resolveErrorCallback)($matches['expression']);
246-
}
247-
248-
throw new UndefinedSymbolException(sprintf('Unknown variable %s', implode('.', $variablePath)));
249-
}
250-
251-
$resolved = $resolved[$variable];
252-
}
253-
254-
if (empty($matches['method'])) {
242+
if (!$this->resolveNestedVariable($resolved, $variablePath, $matches) || empty($matches['method'])) {
255243
return $resolved;
256244
}
257245

@@ -275,6 +263,44 @@ protected function getValue(array $matches, array $variables)
275263
return call_user_func_array([$resolved, $matches['method']], $parameter ?? []);
276264
}
277265

266+
/**
267+
* Resolve nested variable access and object property access
268+
*
269+
* @param mixed $resolved
270+
* @param array $variablePath
271+
* @param array $matches
272+
*
273+
* @return bool
274+
*
275+
* @throws UndefinedSymbolException
276+
*/
277+
protected function resolveNestedVariable(&$resolved, array $variablePath, array $matches): bool
278+
{
279+
foreach ($variablePath as $variable) {
280+
if (is_object($resolved) && !($resolved instanceof ArrayAccess)) {
281+
if (isset($resolved->$variable)) {
282+
$resolved = $resolved->$variable;
283+
284+
continue;
285+
}
286+
} elseif (isset($resolved[$variable]) || (is_array($resolved) && array_key_exists($variable, $resolved))) {
287+
$resolved = $resolved[$variable];
288+
289+
continue;
290+
}
291+
292+
if ($this->resolveErrorCallback) {
293+
$resolved = ($this->resolveErrorCallback)($matches['expression']);
294+
295+
return false;
296+
}
297+
298+
throw new UndefinedSymbolException(sprintf('Unknown variable %s', implode('.', $variablePath)));
299+
}
300+
301+
return true;
302+
}
303+
278304
/**
279305
* Index a control structure in a given template section so a handling of nested control structures of the same
280306
* type can be offered

tests/RenderTest.php

+102-4
Original file line numberDiff line numberDiff line change
@@ -4,12 +4,15 @@
44

55
namespace PHPMicroTemplate\Tests;
66

7+
use ArrayAccess;
8+
use ArrayObject;
79
use PHPMicroTemplate\Exception\FileSystemException;
810
use PHPMicroTemplate\Exception\SyntaxErrorException;
911
use PHPMicroTemplate\Exception\UndefinedSymbolException;
1012
use PHPMicroTemplate\Render;
1113
use PHPMicroTemplate\Tests\Objects\Product;
1214
use PHPUnit\Framework\TestCase;
15+
use stdClass;
1316

1417
/**
1518
* Class RenderTest
@@ -139,18 +142,28 @@ public function testWhitespaceTolerance(): void
139142

140143
/**
141144
* Test multiple loops following each other
145+
*
146+
* @dataProvider loopDataProvider
142147
*/
143-
public function testMultipleLoops(): void
148+
public function testMultipleLoops($products): void
149+
{
150+
$result = $this->render->renderTemplate('multipleLoops.template', ['products' => $products]);
151+
152+
$this->assertXmlStringEqualsXmlFile(__DIR__ . '/Expectations/multipleLoops', $result);
153+
}
154+
155+
public function loopDataProvider(): array
144156
{
145157
$products = [
146158
new Product('Hammer', true),
147159
new Product('Nails', true),
148160
new Product('Wood', true),
149161
];
150162

151-
$result = $this->render->renderTemplate('multipleLoops.template', ['products' => $products]);
152-
153-
$this->assertXmlStringEqualsXmlFile(__DIR__ . '/Expectations/multipleLoops', $result);
163+
return [
164+
'array' => [$products],
165+
'ArrayObject' => [new ArrayObject($products)],
166+
];
154167
}
155168

156169
/**
@@ -283,4 +296,89 @@ public function resolveErrorDataProvider()
283296
'object function call with parameters' => ['person.renderName(firstname, lastname)'],
284297
];
285298
}
299+
300+
/**
301+
* @dataProvider nonExistingPropertyDataProvider
302+
*/
303+
public function testAccessExistingProperty($person): void
304+
{
305+
$this->assertSame(
306+
'Schmidt, Hans',
307+
$this->render->renderTemplateString(
308+
'{{ person.lastName }}, {{ person.firstName }}',
309+
[
310+
'person' => $person,
311+
]
312+
)
313+
);
314+
}
315+
316+
/**
317+
* @dataProvider nonExistingPropertyDataProvider
318+
*/
319+
public function testAccessNonExistingPropertyFails($person): void
320+
{
321+
$this->expectException(UndefinedSymbolException::class);
322+
$this->expectExceptionMessage('Unknown variable person.age');
323+
324+
$this->render->renderTemplateString(
325+
'{{ person.age }}',
326+
[
327+
'person' => $person,
328+
]
329+
);
330+
}
331+
332+
public function nonExistingPropertyDataProvider(): array
333+
{
334+
return [
335+
'array' => [
336+
[
337+
'firstName' => 'Hans',
338+
'lastName' => 'Schmidt',
339+
],
340+
],
341+
'arrayAccess' => [$this->getArrayAccessObject()],
342+
'object' => [$this->getPersonObject()],
343+
];
344+
}
345+
346+
private function getArrayAccessObject(): ArrayAccess
347+
{
348+
return new class () implements ArrayAccess {
349+
private $data = [
350+
'firstName' => 'Hans',
351+
'lastName' => 'Schmidt',
352+
];
353+
354+
public function offsetExists($offset)
355+
{
356+
return array_key_exists($offset, $this->data);
357+
}
358+
359+
public function offsetGet($offset)
360+
{
361+
return $this->data[$offset];
362+
}
363+
364+
public function offsetSet($offset, $value)
365+
{
366+
$this->data[$offset] = $value;
367+
}
368+
369+
public function offsetUnset($offset)
370+
{
371+
unset($this->data[$offset]);
372+
}
373+
};
374+
}
375+
376+
public function getPersonObject(): stdClass
377+
{
378+
$person = new stdClass();
379+
$person->lastName = 'Schmidt';
380+
$person->firstName = 'Hans';
381+
382+
return $person;
383+
}
286384
}

0 commit comments

Comments
 (0)