Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/jstachio/jstachio
Java type safe statically compiled mustache
https://github.com/jstachio/jstachio
annotation-processor dropwizard java jooby mustache spring template-engine type-safe
Last synced: about 1 month ago
JSON representation
Java type safe statically compiled mustache
- Host: GitHub
- URL: https://github.com/jstachio/jstachio
- Owner: jstachio
- License: bsd-3-clause
- Created: 2022-10-13T18:18:17.000Z (about 2 years ago)
- Default Branch: main
- Last Pushed: 2024-09-13T09:25:23.000Z (2 months ago)
- Last Synced: 2024-09-27T03:42:10.511Z (about 2 months ago)
- Topics: annotation-processor, dropwizard, java, jooby, mustache, spring, template-engine, type-safe
- Language: Java
- Homepage:
- Size: 2.59 MB
- Stars: 257
- Watchers: 3
- Forks: 10
- Open Issues: 27
-
Metadata Files:
- Readme: README.md
- License: LICENSE
- Security: SECURITY.md
Awesome Lists containing this project
README
[![Maven Central](https://maven-badges.herokuapp.com/maven-central/io.jstach/jstachio/badge.svg)](https://maven-badges.herokuapp.com/maven-central/io.jstach/jstachio)
[![Github](https://github.com/jstachio/jstachio/actions/workflows/maven.yml/badge.svg)](https://github.com/jstachio/jstachio/actions)A type-safe Java Mustache templating engine.
Templates are compiled into readable Java source code and value bindings are statically checked.
## Documentation
* **[Latest SNAPSHOT JStachio doc](https://jstach.io/jstachio/)**
* **[Current released JStachio doc](https://jstach.io/doc/jstachio/current/apidocs)**The doc is also on javadoc.io but is not aggregated like the above.
The aggregated javadoc is the preferred documentation and the rest of this readme
is mainly for ~~propaganda~~ marketing purposes.For previous releases:
https://jstach.io/doc/jstachio/VERSION/apidocs
Where `VERSION` is the version you want.
## Why choose JStachio
Covered in [why_jstachio_is_better.md](why_jstachio_is_better.md).
## Features
* Logicless [Mustache (v 1.3)](https://github.com/mustache/spec) syntax.
* Full support of non-optional Mustache spec v1.3.0 requirements (including whitespace)
* Optional inheritance support with some caveats
* Optional lambda support with some differences due to static nature
* Get [JEP 430](https://openjdk.org/jeps/430) like support today but wth even more power.
* Templates are compiled into Java code* Value bindings are statically checked.
* Methods, fields and getter-methods can be referenced in templates.
* Friendly error messages with context.
* Zero configuration. No plugins or tweaks are required.
Everything is done with standard javac with any IDE and/or build-system.
* Non-HTML templates are supported. Set of supported escaping content types is extensible.
* Layouts are supported via the Mustache inheritance spec.
* Fallback render service extension point via ServiceLoader* Seamlessly Fallback to reflection based runtime rendering via [JMustache](https://github.com/samskivert/jmustache) and [mustache.java](https://github.com/spullara/mustache.java) (useful for development and changing templates in real time)
* If you are not a fan of generated code you can still use JStachio to type check your mustache templates.
* Customize allowed types that can be outputted otherwise compiler error (to avoid toString on classes that do not have a friendly toString).
* Formatter for custom `toString` of variables at runtime
* Add extra `implements` interfaces to generated code for trait like add ons (`@JStacheInterfaces`)
* Powerful Lambda support
* `Map` support
* `Optional>` support
* Compatible with [JMustache](https://github.com/samskivert/jmustache#-first-and--last) and [Handlebars](https://handlebarsjs.com/api-reference/data-variables.html#root) list index extensions (like `-first`, `-last`, `-index`)
* It is by far the [fastest Java Mustache-like template engine as well one of the fastest in general](#performance).
* Zero dependencies other than JStachio itself
* An absolutely zero runtime dependency option is avaialable (as in all the code needed is generated and not even jstachio is needed during runtime). No need to use Maven shade for annotation processors and other zero dep projects. Also useful for Graal VM native projects for as minimal footprint as possible.
* First class support for Spring Framework (as in the project itself will provide plugins as opposed to an aux project)## Performance
**It is not a goal of this project to be the fastest java templating engine!**
(however it is currently the fastest that I know when this readme was last updated)
Not that peformance matters much with templating languges
(it is rarely the bottleneck) but JStachio is very fast:https://github.com/agentgt/template-benchmark
### String Output
![Template Comparison](https://github.com/agentgt/template-benchmark/raw/utf8/results.png)
### UTF-8 byte Output with extended characters
![Template Comparison](https://github.com/agentgt/template-benchmark/raw/utf8/results-utf8.png)
## Quick Example
```java
@JStache(template = """
{{#people}}
{{message}} {{name}}! You are {{#ageInfo}}{{age}}{{/ageInfo}} years old!
{{#-last}}
That is all for now!
{{/-last}}
{{/people}}
""")
public record HelloWorld(String message, List people) implements AgeLambdaSupport {
}public record Person(String name, LocalDate birthday) {
}public record AgeInfo(long age, String date) {
}public interface AgeLambdaSupport {
@JStacheLambda
default AgeInfo ageInfo(Person person) {
long age = ChronoUnit.YEARS.between(person.birthday(), LocalDate.now());
String date = person.birthday().format(DateTimeFormatter.ISO_DATE);
return new AgeInfo(age, date);
}}
@Test
public void testPerson() throws Exception {
Person rick = new Person("Rick", LocalDate.now().minusYears(70));
Person morty = new Person("Morty", LocalDate.now().minusYears(14));
Person beth = new Person("Beth", LocalDate.now().minusYears(35));
Person jerry = new Person("Jerry", LocalDate.now().minusYears(35));
String actual = JStachio.render(new HelloWorld("Hello alien", List.of(rick, morty, beth, jerry)));
String expected = """
Hello alien Rick! You are 70 years old!
Hello alien Morty! You are 14 years old!
Hello alien Beth! You are 35 years old!
Hello alien Jerry! You are 35 years old!
That is all for now!
""";
assertEquals(expected, actual);}
```## Installation
### Maven
```xml
0.6.0-SNAPSHOT
...
io.jstach
jstachio
${io.jstach.version}
...
org.apache.maven.plugins
maven-compiler-plugin
3.8.1
17
17
io.jstach
jstachio-apt
${io.jstach.version}
```
*N.B. The annotations jar (jstachio-annotation) is pulled in transitively*
### Gradle
```gradle
dependencies {
implementation 'io.jstach:jstachio:VERSION'
annotationProcessor 'io.jstach:jstachio-apt:VERSION'
}
```Examples
--------### user.mustache ###
```hbs
{{#name}}Name: {{.}}, Name Length is {{length}}
{{/name}}Age: {{ age }}
Achievements:
- {{.}}
{{#array}}
{{/array}}
{{^array}}
No achievements
{{/array}}
Items:
- {{value}}
{{#list1}}
{{/list1}}
```
### User.java
Following class can be used to provide actual data to fill into above template.
```java
@JStache(
// points to src/main/resources/user.mustache file
path = "user.mustache",
// or alternatively you can inline the template
template = "",
)
public record User(String name, int age, String[] array, List> list) {
public static class Item {
private final T value;
public Item(T value) {
this.value = value;
}
T value() {
return value;
}
}
}
```
### Rendering
New class `UserRenderer` will be mechanically generated with the above code.
This class can be used to render template filled with actual data. To render template following code can be used:
```java
class Main {
public static void main(String[] args) throws IOException {
User user = new User("John Doe", 21, new String[] {"Knowns nothing"}, list);
StringBuilder appendable = new StringBuilder();
JStachio.render(user, appendable);
}
}
```
The result of running this code will be
```html
Name: John Doe, Name Length is 8
Age: 21
Achievements:
- Knowns nothing
Items:
- helmet
- shower
```
Referencing non existent fields, or fields with non renderable type, all result in compile-time errors. These errors are reported at your project's compile-time alone with other possible errors in java sources.
```
target/classes/user.mustache:5: error: Field not found in current context: 'age1'
Age: {{ age1 }} ({{birthdate}})
^
symbol: mustache directive
location: mustache template
```
```
target/classes/user.mustache:5: error: Unable to render field: type error: Can't render data.birthdate expression of java.util.Date type
Age: {{ age }} ({{birthdate}})
^
symbol: mustache directive
location: mustache template
```
See `test/examples` project for more examples.
## Java specific extensions
### Enum matching support
Basically enums have boolean keys that are the enums name (`Enum.name()`) that can be used as conditional sections.
Assume `light` is an enum like:
```java
public enum Light {
RED,
GREEN,
YELLOW
}
```
You can conditinally select on the enum like a pattern match:
```hbs
{{#light.RED}}
STOP
{{/light.RED}}
{{#light.GREEN}}
GO
{{/light.GREEN}}
{{#light.YELLOW}}
Proceeed with caution
{{/light.YELLOW}}
```
### Index support
JStachio is compatible with both handlebars and JMustache index keys for iterable sections.
* `-first` is boolean that is true when you are on the first item
* `-last` is a boolean that is true when you are on the last item in the iterable
* `-index` is a one based index. The first item would be `1` and not `0`
### Lambda support
JStachio supports lambda section calls in a similar manner to JMustache. Just tag your methods
with `@JStacheLambda` and the returned models will be used to render the contents of the lambda section.
The top of the context stack can be passed to the lambda.
JStachio unlike the spec does not support returning dynamic templates that are then rendered against the context stack.
However dynamic output can be achieved by the caller changing the contents of the lambda section as the contents of the
section act as an inline template.
Design
------
The idea is to create templating engine combining [mustache](https://jgonggrijp.gitlab.io/wontache/mustache.5.html) logicless philosophy
with Java's single responsibility and static-typing.
Full compile-time check of syntax and data-binding is the main requirement.
Currently Java-code is generated for templates. Generated Java-code should never fail to compile.
If it is impossible to generate valid Java-code from some template,
friendly compile-time error pointing to template file should be generated.
Users should never be exposed to generated Java-code.
Original mustache uses Javascript-objects to define rendering context.
Fields of selected Javascript-objects are binded with template fields.
Static mustache uses Java-objects to define rendering context.
Binding of template fields is defined and checked at compile-time.
Missing fields are compile-time error.
License
-------
JStachio is under BSD 3-clause license.