Qute Templating Engine
Qute is a templating engine designed specifically to meet the Quarkus
needs. The usage of reflection is minimized to reduce the size of native
images. The API combines both the imperative and the non-blocking reactive
style of coding. In the development mode, all files located in
src/main/resources/templates
are watched for changes and modifications are
immediately visible. Furthermore, we try to detect most of the template
problems at build time. In this guide, you will learn how to easily render
templates in your application.
完整源码
We recommend that you follow the instructions in the next sections and create the application step by step. However, you can go right to the completed example.
Clone the Git repository: git clone https://github.com/quarkusio/quarkus-quickstarts.git
, or download
an archive.
The solution is located in the qute-quickstart
directory.
Hello World with Jakarta REST
If you want to use Qute in your Jakarta REST application, you need to add an extension first:
-
either
quarkus-resteasy-reactive-qute
if you are using RESTEasy Reactive:pom.xml<dependency> <groupId>io.quarkus</groupId> <artifactId>quarkus-resteasy-reactive-qute</artifactId> </dependency>
build.gradleimplementation("io.quarkus:quarkus-resteasy-reactive-qute")
-
or
quarkus-resteasy-qute
if you are using RESTEasy Classic:pom.xml<dependency> <groupId>io.quarkus</groupId> <artifactId>quarkus-resteasy-qute</artifactId> </dependency>
build.gradleimplementation("io.quarkus:quarkus-resteasy-qute")
We’ll start with a very simple template:
Hello {name}! (1)
1 | {name} is a value expression that is evaluated when the template is
rendered. |
By default, all files located in the src/main/resources/templates
directory and its subdirectories are registered as templates. Templates are
validated during startup and watched for changes in the development mode.
|
Now let’s inject the "compiled" template in the resource class.
package org.acme.quarkus.sample;
import jakarta.inject.Inject;
import jakarta.ws.rs.GET;
import jakarta.ws.rs.Path;
import jakarta.ws.rs.QueryParam;
import jakarta.ws.rs.Produces;
import jakarta.ws.rs.core.MediaType;
import io.quarkus.qute.TemplateInstance;
import io.quarkus.qute.Template;
@Path("hello")
public class HelloResource {
@Inject
Template hello; (1)
@GET
@Produces(MediaType.TEXT_PLAIN)
public TemplateInstance get(@QueryParam("name") String name) {
return hello.data("name", name); (2) (3)
}
}
1 | If there is no @Location qualifier provided, the field name is used to
locate the template. In this particular case, we’re injecting a template
with path templates/hello.txt . |
2 | Template.data() returns a new template instance that can be customized
before the actual rendering is triggered. In this case, we put the name
value under the key name . The data map is accessible during rendering. |
3 | Note that we don’t trigger the rendering - this is done automatically by a
special ContainerResponseFilter implementation. |
If your application is running, you can request the endpoint:
$ curl -w "\n" http://localhost:8080/hello?name=Martin
Hello Martin!
Type-safe templates
There’s an alternate way to declare your templates in your Java code, which relies on the following convention:
-
Organise your template files in the
/src/main/resources/templates
directory, by grouping them into one directory per resource class. So, if yourItemResource
class references two templateshello
andgoodbye
, place them at/src/main/resources/templates/ItemResource/hello.txt
and/src/main/resources/templates/ItemResource/goodbye.txt
. Grouping templates per resource class makes it easier to navigate to them. -
In each of your resource class, declare a
@CheckedTemplate static class Template {}
class within your resource class. -
Declare one
public static native TemplateInstance method();
per template file for your resource. -
Use those static methods to build your template instances.
Here’s the previous example, rewritten using this style:
We’ll start with a very simple template:
Hello {name}! (1)
1 | {name} is a value expression that is evaluated when the template is
rendered. |
Now let’s declare and use those templates in the resource class.
package org.acme.quarkus.sample;
import jakarta.inject.Inject;
import jakarta.ws.rs.GET;
import jakarta.ws.rs.Path;
import jakarta.ws.rs.QueryParam;
import jakarta.ws.rs.Produces;
import jakarta.ws.rs.core.MediaType;
import io.quarkus.qute.TemplateInstance;
import io.quarkus.qute.CheckedTemplate;
@Path("hello")
public class HelloResource {
@CheckedTemplate
public static class Templates {
public static native TemplateInstance hello(String name); (1)
}
@GET
@Produces(MediaType.TEXT_PLAIN)
public TemplateInstance get(@QueryParam("name") String name) {
return Templates.hello(name); (2)
}
}
1 | This declares a template with path templates/HelloResource/hello . |
2 | Templates.hello() returns a new template instance that is returned from
the resource method. Note that we don’t trigger the rendering - this is done
automatically by a special ContainerResponseFilter implementation. |
Once you have declared a @CheckedTemplate class, we will check that all
its methods point to existing templates, so if you try to use a template
from your Java code and you forgot to add it, we will let you know at build
time :)
|
Keep in mind this style of declaration allows you to reference templates declared in other resources too:
package org.acme.quarkus.sample;
import jakarta.inject.Inject;
import jakarta.ws.rs.GET;
import jakarta.ws.rs.Path;
import jakarta.ws.rs.QueryParam;
import jakarta.ws.rs.Produces;
import jakarta.ws.rs.core.MediaType;
import io.quarkus.qute.TemplateInstance;
@Path("goodbye")
public class GoodbyeResource {
@GET
@Produces(MediaType.TEXT_PLAIN)
public TemplateInstance get(@QueryParam("name") String name) {
return HelloResource.Templates.hello(name);
}
}
Top-level type-safe templates
Naturally, if you want to declare templates at the top-level, directly in
/src/main/resources/templates/hello.txt
, for example, you can declare them
in a top-level (non-nested) Templates
class:
package org.acme.quarkus.sample;
import io.quarkus.qute.TemplateInstance;
import io.quarkus.qute.Template;
import io.quarkus.qute.CheckedTemplate;
@CheckedTemplate
public class Templates {
public static native TemplateInstance hello(String name); (1)
}
1 | This declares a template with path templates/hello . |
Template Parameter Declarations
If you declare a parameter declaration in a template then Qute attempts to validate all expressions that reference this parameter and if an incorrect expression is found the build fails.
Let’s suppose we have a simple class like this:
public class Item {
public String name;
public BigDecimal price;
}
And we’d like to render a simple HTML page that contains the item name and price.
Let’s start again with the template:
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>{item.name}</title> (1)
</head>
<body>
<h1>{item.name}</h1>
<div>Price: {item.price}</div> (2)
</body>
</html>
1 | This expression is validated. Try to change the expression to
{item.nonSense} and the build should fail. |
2 | This is also validated. |
Finally, let’s create a resource class with type-safe templates:
package org.acme.quarkus.sample;
import jakarta.inject.Inject;
import jakarta.ws.rs.GET;
import jakarta.ws.rs.Path;
import jakarta.ws.rs.Produces;
import jakarta.ws.rs.core.MediaType;
import io.quarkus.qute.TemplateInstance;
import io.quarkus.qute.Template;
import io.quarkus.qute.CheckedTemplate;
@Path("item")
public class ItemResource {
@CheckedTemplate
public static class Templates {
public static native TemplateInstance item(Item item); (1)
}
@GET
@Path("{id}")
@Produces(MediaType.TEXT_HTML)
public TemplateInstance get(@PathParam("id") Integer id) {
return Templates.item(service.findItem(id)); (2)
}
}
1 | Declare a method that gives us a TemplateInstance for
templates/ItemResource/item.html and declare its Item item parameter so
we can validate the template. |
2 | Make the Item object accessible in the template. |
When the --parameters compiler argument is enabled, RESTEasy Reactive may
infer the parameter names from the method argument names, making the
@PathParam("id") annotation optional in this case.
|
Template parameter declaration inside the template itself
Alternatively, you can declare your template parameters in the template file itself.
Let’s start again with the template:
{@org.acme.Item item} (1)
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>{item.name}</title> (2)
</head>
<body>
<h1>{item.name}</h1>
<div>Price: {item.price}</div>
</body>
</html>
1 | Optional parameter declaration. Qute attempts to validate all expressions
that reference the parameter item . |
2 | This expression is validated. Try to change the expression to
{item.nonSense} and the build should fail. |
Finally, let’s create a resource class.
package org.acme.quarkus.sample;
import jakarta.inject.Inject;
import jakarta.ws.rs.GET;
import jakarta.ws.rs.Path;
import jakarta.ws.rs.Produces;
import jakarta.ws.rs.core.MediaType;
import io.quarkus.qute.TemplateInstance;
import io.quarkus.qute.Template;
@Path("item")
public class ItemResource {
@Inject
ItemService service;
@Inject
Template item; (1)
@GET
@Path("{id}")
@Produces(MediaType.TEXT_HTML)
public TemplateInstance get(Integer id) {
return item.data("item", service.findItem(id)); (2)
}
}
1 | Inject the template with path templates/item.html . |
2 | Make the Item object accessible in the template. |
Template Extension Methods
Template extension methods are used to extend the set of accessible properties of data objects.
Sometimes, you’re not in control of the classes that you want to use in your template, and you cannot add methods to them. Template extension methods allows you to declare new method for those classes that will be available from your templates just as if they belonged to the target class.
Let’s keep extending on our simple HTML page that contains the item name, price and add a discounted price. The discounted price is sometimes called a "computed property". We will implement a template extension method to render this property easily. Let’s update our template:
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>{item.name}</title>
</head>
<body>
<h1>{item.name}</h1>
<div>Price: {item.price}</div>
{#if item.price > 100} (1)
<div>Discounted Price: {item.discountedPrice}</div> (2)
{/if}
</body>
</html>
1 | if is a basic control flow section. |
2 | This expression is also validated against the Item class and obviously
there is no such property declared. However, there is a template extension
method declared on the TemplateExtensions class - see below. |
Finally, let’s create a class where we put all our extension methods:
package org.acme.quarkus.sample;
import io.quarkus.qute.TemplateExtension;
@TemplateExtension
public class TemplateExtensions {
public static BigDecimal discountedPrice(Item item) { (1)
return item.price.multiply(new BigDecimal("0.9"));
}
}
1 | A static template extension method can be used to add "computed properties" to a data class. The class of the first parameter is used to match the base object and the method name is used to match the property name. |
you can place template extension methods in every class if you annotate them
with @TemplateExtension but we advise to keep them either grouped by
target type, or in a single TemplateExtensions class by convention.
|
Rendering Periodic Reports
Templating engine could be also very useful when rendering periodic
reports. You’ll need to add the quarkus-scheduler
and quarkus-qute
extensions first. In your pom.xml
file, add:
<dependency>
<groupId>io.quarkus</groupId>
<artifactId>quarkus-qute</artifactId>
</dependency>
<dependency>
<groupId>io.quarkus</groupId>
<artifactId>quarkus-scheduler</artifactId>
</dependency>
Let’s suppose we have a SampleService
bean whose get()
method returns a
list of samples.
public class Sample {
public boolean valid;
public String name;
public String data;
}
The template is simple:
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Report {now}</title>
</head>
<body>
<h1>Report {now}</h1>
{#for sample in samples} (1)
<h2>{sample.name ?: 'Unknown'}</h2> (2)
<p>
{#if sample.valid}
{sample.data}
{#else}
<strong>Invalid sample found</strong>.
{/if}
</p>
{/for}
</body>
</html>
1 | The loop section makes it possible to iterate over iterables, maps and streams. |
2 | This value expression is using the elvis operator - if the name is null the default value is used. |
package org.acme.quarkus.sample;
import jakarta.inject.Inject;
import io.quarkus.qute.Template;
import io.quarkus.qute.Location;
import io.quarkus.scheduler.Scheduled;
public class ReportGenerator {
@Inject
SampleService service;
@Location("reports/v1/report_01") (1)
Template report;
@Scheduled(cron="0 30 * * * ?") (2)
void generate() {
String result = report
.data("samples", service.get())
.data("now", java.time.LocalDateTime.now())
.render(); (3)
// Write the result somewhere...
}
}
1 | In this case, we use the @Location qualifier to specify the template path:
templates/reports/v1/report_01.html . |
2 | Use the @Scheduled annotation to instruct Quarkus to execute this method
on the half hour. For more information see the
Scheduler guide. |
3 | The TemplateInstance.render() method triggers rendering. Note that this
method blocks the current thread. |
Qute Reference Guide
To learn more about Qute, please refer to the Qute reference guide.
Qute Configuration Reference
Configuration property fixed at build time - All other configuration properties are overridable at runtime
Type |
Default |
|
---|---|---|
The list of suffixes used when attempting to locate a template file. By default, Environment variable: Show more |
list of string |
|
The list of exclude rules used to intentionally ignore some parts of an expression when performing type-safe validation. An element value must have at least two parts separated by dot. The last part is used to match the property/method name. The prepended parts are used to match the class name. The value Examples:
Environment variable: Show more |
list of string |
|
This regular expression is used to exclude template files from the The matched input is the file path relative from the By default, the hidden files are excluded. The name of a hidden file starts with a dot. Environment variable: Show more |
|
|
The prefix is used to access the iteration metadata inside a loop section. A valid prefix consists of alphanumeric characters and underscores. Three special constants can be used:
Environment variable: Show more |
string |
|
The list of content types for which the Environment variable: Show more |
list of string |
|
The default charset of the templates files. Environment variable: Show more |
|
|
By default, a template modification results in an application restart that triggers build-time validations. This regular expression can be used to specify the templates for which the application is not restarted. I.e. the templates are reloaded and only runtime validations are performed. The matched input is the template path that starts with a template root, and the Environment variable: Show more |
||
The strategy used when a standalone expression evaluates to a "not found" value at runtime and the This strategy is never used when evaluating section parameters, e.g. By default, the Environment variable: Show more |
|
|
Specify whether the parser should remove standalone lines from the output. A standalone line is a line that contains at least one section tag, parameter declaration, or comment but no expression and no non-whitespace character. Environment variable: Show more |
boolean |
|
If set to Note that the Environment variable: Show more |
boolean |
|
The global rendering timeout in milliseconds. It is used if no Environment variable: Show more |
long |
|
If set to Environment variable: Show more |
boolean |
|
The additional map of suffixes to content types. This map is used when working with template variants. By default, the Environment variable: Show more |
|