Skip to content

Commit b365541

Browse files
committed
Allow nested variable resolving (closes #2)
Transfer CI to github-actions instead of travis-ci
1 parent 5d196bc commit b365541

File tree

7 files changed

+149
-46
lines changed

7 files changed

+149
-46
lines changed

.codeclimate.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -14,4 +14,4 @@ plugins:
1414
enabled: true
1515
config:
1616
tests_patterns:
17-
- test/**
17+
- tests/**

.coveralls.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,3 @@
1-
service_name: travis-ci
1+
service_name: github-actions
22
coverage_clover: build/logs/clover.xml
33
json_path: build/logs/coveralls-upload.json

.github/workflows/main.yml

+48
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,48 @@
1+
name: CI
2+
3+
on: [push, pull_request]
4+
5+
jobs:
6+
tests:
7+
runs-on: ubuntu-latest
8+
strategy:
9+
matrix:
10+
php: ['7.1', '7.2', '7.3', '7.4', '8.0']
11+
12+
name: PHP ${{ matrix.php }} tests
13+
steps:
14+
- name: Checkout
15+
uses: actions/checkout@v2
16+
17+
- name: Setup PHP
18+
uses: shivammathur/setup-php@v2
19+
with:
20+
php-version: ${{ matrix.php }}
21+
coverage: xdebug
22+
23+
- name: Install dependencies
24+
run: composer install
25+
26+
- name: Prepare codeclimate test reporter
27+
if: ${{ matrix.php == '8.0' }}
28+
run: |
29+
curl -L https://codeclimate.com/downloads/test-reporter/test-reporter-latest-linux-amd64 > ./cc-test-reporter
30+
chmod +x ./cc-test-reporter
31+
./cc-test-reporter before-build
32+
33+
- name: Execute tests
34+
run: XDEBUG_MODE=coverage ./vendor/bin/phpunit --coverage-clover=build/logs/clover.xml --testdox
35+
36+
- name: Upload the reports to coveralls.io
37+
if: ${{ matrix.php == '8.0' }}
38+
run: |
39+
composer global require php-coveralls/php-coveralls
40+
php-coveralls -v
41+
env:
42+
COVERALLS_REPO_TOKEN: ${{ secrets.GITHUB_TOKEN }}
43+
44+
- name: Upload the reports to codeclimate
45+
if: ${{ matrix.php == '8.0' }}
46+
run: sudo ./cc-test-reporter after-build -r $CC_TEST_REPORTER_ID
47+
env:
48+
CC_TEST_REPORTER_ID: ddd471f81306655072d8ed9b6330b6851a4d7a63a8767331ba7a4a535374e799

.travis.yml

-32
This file was deleted.

README.md

+19-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
[![Latest Version](https://img.shields.io/packagist/v/wol-soft/php-micro-template.svg)](https://packagist.org/packages/wol-soft/php-micro-template)
22
[![Maintainability](https://api.codeclimate.com/v1/badges/9e3c565c528edb3d58d5/maintainability)](https://codeclimate.com/github/wol-soft/php-micro-template/maintainability)
3-
[![Build Status](https://travis-ci.com/wol-soft/php-micro-template.svg?branch=master)](https://travis-ci.com/wol-soft/php-micro-template)
3+
[![Build Status](https://github.com/wol-soft/php-micro-template/actions/workflows/main.yml/badge.svg)](https://github.com/wol-soft/php-micro-template/actions/workflows/main.yml)
44
[![Coverage Status](https://coveralls.io/repos/github/wol-soft/php-micro-template/badge.svg?branch=master)](https://coveralls.io/github/wol-soft/php-micro-template?branch=master)
55
[![MIT License](https://img.shields.io/packagist/l/wol-soft/php-micro-template.svg)](https://github.com/wol-soft/php-micro-template/blob/master/LICENSE)
66

@@ -116,6 +116,24 @@ Values which are assigned to the template and used directly will be casted to st
116116
{{ myObject.getProperty() }}
117117
```
118118

119+
Your provided data may be a nested array which can be resolved in the template:
120+
121+
```php
122+
$render->renderTemplateString(
123+
'{{ render.productRender.renderProductName(product.details.name) }}',
124+
[
125+
'product' => [
126+
'details' => [
127+
'name' => 'MyProduct',
128+
],
129+
],
130+
'render' => [
131+
'productRender' => new ProductRender(),
132+
]
133+
]
134+
);
135+
```
136+
119137
### Loops
120138

121139
If you assign an array or an iterable object you can use the *foreach* loop to iterate.

src/Render.php

+27-11
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@
1919
*/
2020
class Render
2121
{
22-
private const REGEX_VARIABLE = '(?<variable>\w+)(\.((?<method>\w+)\((?<parameter>[^{}%]*)\)))?';
22+
private const REGEX_VARIABLE = '(?<variable>\w+)(?<nestedVariable>(\.\w+)*)(\.(?<method>\w+)\((?<parameter>[^{}%]*)\))?';
2323

2424
/** @var array */
2525
private $templates = [];
@@ -214,28 +214,44 @@ protected function getTemplate(string $template) : string
214214
*/
215215
protected function getValue(array $matches, array $variables)
216216
{
217-
$variable = $matches['variable'] ?? null;
218-
$method = $matches['method'] ?? null;
217+
$resolved = $variables;
218+
$variablePath = [$matches['variable']];
219219

220-
// first check via isset for faster lookup
221-
if (!isset($variables[$variable]) && !array_key_exists($variable, $variables)) {
222-
throw new UndefinedSymbolException("Unknown variable {$variable}");
220+
if (!empty($matches['nestedVariable'])) {
221+
array_push($variablePath, ...explode('.', trim($matches['nestedVariable'], '.')));
223222
}
224223

225-
if (empty($method)) {
226-
return $variables[$variable];
224+
foreach ($variablePath as $variable) {
225+
// first check via isset for faster lookup
226+
if (!isset($resolved[$variable]) && !array_key_exists($variable, $resolved)) {
227+
throw new UndefinedSymbolException(sprintf('Unknown variable %s', implode('.', $variablePath)));
228+
}
229+
230+
$resolved = $resolved[$variable];
231+
}
232+
233+
if (empty($matches['method'])) {
234+
return $resolved;
227235
}
228236

229-
if (!is_callable([$variables[$variable], $method])) {
230-
throw new UndefinedSymbolException("Function {$method} on object {$variable} not callable");
237+
if (!is_object($resolved)) {
238+
throw new UndefinedSymbolException(
239+
sprintf('Trying to call %s on non-object %s', $matches['method'], implode('.', $variablePath))
240+
);
241+
}
242+
243+
if (!is_callable([$resolved, $matches['method']])) {
244+
throw new UndefinedSymbolException(
245+
sprintf('Function %s on object %s not callable', $matches['method'], implode('.', $variablePath))
246+
);
231247
}
232248

233249
// check if the function to call has a given parameter. In this case resolve the parameter
234250
if (!empty($matches['parameter'])) {
235251
$parameter = $this->extractParameter($matches['parameter'], $variables);
236252
}
237253

238-
return call_user_func_array([$variables[$variable], $method], $parameter ?? []);
254+
return call_user_func_array([$resolved, $matches['method']], $parameter ?? []);
239255
}
240256

241257
/**

tests/RenderTest.php

+53
Original file line numberDiff line numberDiff line change
@@ -29,18 +29,28 @@ public function setUp(): void
2929
public function testRenderNotExistingTemplate(): void
3030
{
3131
$this->expectException(FileSystemException::class);
32+
$this->expectExceptionMessage('Template nonExistingTemplate.template not found');
3233
$this->render->renderTemplate('nonExistingTemplate.template');
3334
}
3435

3536
public function testUndefinedVariable(): void
3637
{
3738
$this->expectException(UndefinedSymbolException::class);
39+
$this->expectExceptionMessage('Unknown variable name');
3840
$this->render->renderTemplate('undefinedVariable.template', ['firstname' => 'John']);
3941
}
4042

43+
public function testMethodOnNonObject(): void
44+
{
45+
$this->expectException(UndefinedSymbolException::class);
46+
$this->expectExceptionMessage('Trying to call getPrice on non-object product');
47+
$this->render->renderTemplate('undefinedMethod.template', ['product' => false]);
48+
}
49+
4150
public function testUndefinedMethod(): void
4251
{
4352
$this->expectException(UndefinedSymbolException::class);
53+
$this->expectExceptionMessage('Function getPrice on object product not callable');
4454
$this->render->renderTemplate('undefinedMethod.template', ['product' => new Product('Wood', true)]);
4555
}
4656

@@ -204,4 +214,47 @@ public function invalidFunctionParameterProvider(): array
204214
['{{ viewHelper.sum(variable object.get()) }}'],
205215
];
206216
}
217+
218+
public function testNestedVariable(): void
219+
{
220+
$vars = [
221+
'person' => [
222+
'name' => [
223+
'firstName' => 'Hans',
224+
'lastName' => 'Schmidt',
225+
],
226+
],
227+
];
228+
229+
$this->assertSame(
230+
'Schmidt, Hans',
231+
$this->render->renderTemplateString('{{ person.name.lastName }}, {{ person.name.firstName }}', $vars)
232+
);
233+
}
234+
235+
public function testNestedMethod(): void
236+
{
237+
$vars = [
238+
'person' => [
239+
'name' => [
240+
'firstName' => 'Hans',
241+
'lastName' => 'Schmidt',
242+
'render' => new class () {
243+
public function renderName(string $firstName, string $lastName): string
244+
{
245+
return "$lastName, $firstName";
246+
}
247+
}
248+
],
249+
],
250+
];
251+
252+
$this->assertSame(
253+
'Schmidt, Hans',
254+
$this->render->renderTemplateString(
255+
'{{ person.name.render.renderName(person.name.firstName, person.name.lastName) }}',
256+
$vars
257+
)
258+
);
259+
}
207260
}

0 commit comments

Comments
 (0)