Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add proposal for template renderer #1280

Draft
wants to merge 7 commits into
base: master
Choose a base branch
from
209 changes: 209 additions & 0 deletions proposed/template-renderer.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,209 @@
Common Interface for Rendering Templates
========================================

This document describes a common interface for template renderers.

The goal set by `TemplateRendererInterface` is to standardize how frameworks, libraries and CMSs
render their template so that projects are more free to use the template renderer they prefer.
alexander-schranz marked this conversation as resolved.
Show resolved Hide resolved

The key words "MUST", "MUST NOT", "REQUIRED", "SHALL", "SHALL NOT", "SHOULD",
"SHOULD NOT", "RECOMMENDED", "MAY", and "OPTIONAL" in this document are to be
interpreted as described in [RFC 2119][].

The word `implementor` in this document is to be interpreted as someone
implementing the `TemplateRendererInterface` in a [template engine][] library.
Users of template renderer are referred to as `user`.
The word `enduser` in this document is to be interpreted as someone using
a library, framework, cms created by a `user`.
alexander-schranz marked this conversation as resolved.
Show resolved Hide resolved

[RFC 2119]: http://tools.ietf.org/html/rfc2119

## Goal

Having common interfaces for rendering templates allows developers to create libraries that can interact with many frameworks and other libraries in a common fashion.

Some examples may use the Interface:
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think details should be moved to a meta-document.


- Content Management Systems (Sulu CMS, Drupal, Typo3, Contao CMS, ...)
- E-Commerce Platforms (Sylius, Spryker)
- Newsletter/Mail Libraries (Symfony Mailer, Abstractions over Mailchimp and other Newsletter tools)
- Anything following ["Separating content from presentation"](https://en.wikipedia.org/wiki/Separation_of_content_and_presentation) where the presentation is project specific.

Some examples may implement the Interface or providing a bridge:
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yii has alike interface https://github.com/yiisoft/view/blob/master/src/TemplateRendererInterface.php with the difference that we're passing a rendering context as $this for nested template rendering.

Copy link
Author

@alexander-schranz alexander-schranz Jun 1, 2022

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thx for mentioning yii views. Looking at the example it works the same way ->render($template, $params): https://github.com/yiisoft/view/blob/master/docs/basic-functionality.md#rendering

That yii uses then in the template $this is totally fine the TemplateRendererInterface doesn't force how the the variables need to access in the template. If it is {variable}, {{variable}} or <?php echo $this->variable ?> is up to the template renderer.

I will add it also the the list. Please check my details in the PR description about it if I did write nothing wrong about it.


- Blade (Laravel Framework)
- Laminas View
- Latte (Nette Framework)
- Smarty
- Sulu CMS
- Twig (Symfony Framework)
- Typo3

## Definitions

* **Template** - A string representation of a template.
* **Context** - A array of context data given into to the rendered template.
alexander-schranz marked this conversation as resolved.
Show resolved Hide resolved
* **TemplateRenderer** - A service rendering which will render the template with the given context and return the rendered content.
alexander-schranz marked this conversation as resolved.
Show resolved Hide resolved

### Template

A template MUST be a string representation of a given template supported by the template renderer. It MAY be a file path
to the template file, but it can also be a virtual name or path supported only by a specific template renderer. The
template is not limited by specific characters by definition but a template renderer MAY support only specific one.

### Context

A context is MUST be an array of the available variables given to the template renderer. The array keys represent the
alexander-schranz marked this conversation as resolved.
Show resolved Hide resolved
available variables and MUST be typed as string. The array values represent the variables value and can be anything and
alexander-schranz marked this conversation as resolved.
Show resolved Hide resolved
are so typed by mixed. The context SHOULD NOT be an object to support also basic implementation of a template renderer.
alexander-schranz marked this conversation as resolved.
Show resolved Hide resolved

### TemplateRenderer

A template renderer is a service object implementing the `TemplateRendererInterface`. It MUST be responsible to render a
supported template by a OPTIONAL context. It MUST return the rendered content of the template as a string.
If a template was not found by the template renderer, an Exception implementing the `TemplateNotFoundExceptionInterface`
MUST be thrown. The template renderer SHALL NOT in any case output/print/stream something directly to php output.
A implementor of a template renderer is allowed to support as context also an object, but a user is REQUIRED whennusing
The `TemplateRendererInterface` to give the context as array to support also simple template renderers.
alexander-schranz marked this conversation as resolved.
Show resolved Hide resolved

## Usage
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This section definitely belongs to metadata document.


While the implementor of `TemplateRendererInterface` MUST make sure that the template renderer behave like defined
above. The user of `TemplateRendererInterface` MUST allow when providing a library, application, cms for an enduser
using the Interface to give or configure the `template` used by the template renderer. This can be achieved with any
type of injection or configuration.

Basic implementor implementation:

```php
<?php

use Psr\TemplateRenderer\TemplateRendererInterface;

class TemplateRenderer implements TemplateRendererInterface
{
public function render(string $template, array $context = []): string
{
switch ($template) {
case 'home.tpl';

return $this->renderTemplate('<p>{{name}}</p>', $context);
default:
throw new TemplateNotFoundException($template);
}
}

/**
* @param array<string, mixed> $context
* @return string
*/
private function renderTemplate(string $content, array $context): string
{
foreach ($context as $key => $value) {
$content = str_replace('{{' . $key . '}}', $context[$key], $content);
}

return $content;
}
}

use Psr\TemplateRenderer\TemplateNotFoundExceptionInterface;

class TemplateNotFoundException extends \RuntimeException implements TemplateNotFoundExceptionInterface
{
public function __construct(private string $template, ?\Throwable $previous = null)
{
parent::__construct('Template not found: "' . $template . '"', 0 , $previous);
}

public function getTemplate(): string
{
return $this->template;
}
}
```

Basic user implementation:

```php
<?php

use Psr\TemplateRenderer\TemplateRendererInterface;

class SomeController
{
public function __construct(
private TemplateRendererInterface $templateRenderer,
private string $template,
)

public function someAction(): string
{
$context = [/* load something */];

return $this->templateRenderer->render($this->template, $context);
}
}
```

Example enduser implementation A:

```php
<?php

$controller = new Controller($twig, '@Context/pages/mail.html.twig');
$controller->someAction();
```

Example enduser implementation B:

```php
<?php

$controller = new Controller($blade, 'mail');
$controller->someAction();
```

Example enduser implementation C:

```php
<?php

$controller = new Controller($latte, 'mail.latte');
$controller->someAction();
```

## Interfaces

### TemplateRendererInterface

```php
<?php

namespace Psr\TemplateRenderer;

interface TemplateRendererInterface {
/**
* Render the template with the given context data.
*
* @param string $template
* @param array<string, mixed> $context
*
* @return string
*/
public function render(string $template, array $context = []): string;
}
```

### TemplateNotFoundExceptionInterface

```php
<?php

namespace Psr\TemplateRenderer;

interface TemplateNotFoundExceptionInterface
alexander-schranz marked this conversation as resolved.
Show resolved Hide resolved
{
public function getTemplate(): string;
}
```