Generate type-safe Gleam modules from text-based template files.
This project provides a Rust program that parses a basic template format and outputs Gleam modules with 'render' functions that can be imported and called to render the template with different parameters.
Matcha is basic but currently feature complete and well tested. If the repository looks inactive it is because it is stable.
Template languages like Matcha are useful to have in general and are particularly useful for generating unstructured text. Matcha generates well typed code that avoids issues that some more dynamic templating systems have.
That said, if you're planning to generate structured output, it is sensible to use a better suited approach. For data formats, this should be serialization and deserialization libraries. For formats like HTML, you would likely be better using a library like lustre or nakai.
Download pre-built binaries for the latest release from the Releases page.
Build from source with:
cargo install --path .
Run:
matcha
At the root of your project and it will walk your project folder structure and compile any template files it finds.
Template files should have a .matcha
extension. Templates are compiled into .gleam
files that can
be imported like any other regular module. The modules expose a render
function, that returns a
String
, and render_builder
function that returns a StringBuilder
.
Some errors, mostly syntax, will be picked up by the Rust code but it is possible to generate invalid modules and so the Gleam compiler will pick up further errors.
The syntax is inspired by Jinja.
You can use {>
syntax to add with
statements to declare parameters and their assoicated types
for the template and the generated render function. All parameters must be declared with with
statements.
{> with greeting as String
{> with name as String
{{ greeting }}, {{ name }}
You can use {{ name }}
syntax to insert the value of name
into the rendered template.
{> with name as String
Hello {{ name }}
You can use {[ name ]}
syntax to insert a string builder value into the rendered template. This
has the advantage of using
string_builder.append_builder
in the rendered template and so it more efficient for inserting content that is already in a
StringBuilder
. This can be used to insert content from another template.
{> with name as StringBuilder
{[ name ]}
You can use {% %}
blocks to create an if-statement using the if
, else
and endif
keywords.
The else
is optional.
{> with is_admin as Bool
{% if is_admin %}Admin{% else %}User{% endif %}
You can use {% %}
blocks to create for-loops using the for
, in
and endfor
keywords.
{> with list as List(String)
<ul>
{% for entry in list %}
<li>{{ entry }}</li>
{% endfor %}
</ul>
Additionally you can use the as
keyword to associate a type with the items being iterated over.
This is necessary if you're using a complex object.
{> import organisation.{type Organisation}
{> import membership.{type Member}
{> with org as Organisation
<ul>
{% for user as Member in organisation.members %}
<li>{{ user.name }}</li>
{% endfor %}
</ul>
You can use the {>
syntax to add import statements to the template. These are used to import types
to use with the with
syntax below to help Gleam check variables used in the template.
{> import my_user.{type MyUser}
You can use the {> fn ... {> endfn
syntax to add a local function to your template:
{> fn full_name(second_name: String)
Lucy {{ second_name }}
{> endfn
The function always returns a StringBuilder
value so you must use {[ ... ]}
syntax to insert
them into templates. The function body has its last new line trimmed, so the above function called
as full_name("Gleam")
would result in Lucy Gleam
and not \nLucy Gleam\n
or any other
variation. If you want a trailing new line in the output then add an extra blank line before the {> endfn
.
The function declaration has no impact on the final template as all lines are removed from the final text.
Like in normal code, functions make it easier to deal with repeated components within your template.
{> fn item(name: String)
<li class="px-2 py-1 font-bold">{{ name }}</li>
{> endfn
<ul>
{[ item(name: "Alice") ]}
{[ item(name: "Bob") ]}
{[ item(name: "Cary") ]}
</ul>
You can use the pub
keyword to declare the function as public in which case other modules will be
able to import it from gleam module compiled from the template.
{> pub fn user_item(name: String)
<li class="px-2 py-1 font-bold">{{ name }}</li>
{> endfn
If a template only includes function declarations and no meaningful template content then matcha
will not add the render
and render_builder
. Instead the module will act as a library of
functions where each function body is a template.
A template like:
{> import my_user.{type User}
{> with user_obj as User
Hello{% if user_obj.is_admin %} Admin{% endif %}
is compiled to a Gleam module:
import gleam/string_builder.{StringBuilder}
import gleam/list
import my_user.{User}
pub fn render_builder(user_obj user_obj: User) -> StringBuilder {
let builder = string_builder.from_string("")
let builder = string_builder.append(builder, "Hello")
let builder = case user_obj.is_admin {
True -> {
let builder = string_builder.append(builder, " Admin")
builder
}
False -> builder
}
let builder = string_builder.append(builder, "
")
builder
}
pub fn render(user_obj user_obj: User) -> String {
string_builder.to_string(render_builder(user_obj: user_obj))
}
Which you can import and call render
or render_builder
on with the appropriate arguments.
Rust tests can be run with cargo test
. They use insta for snapshots.
Gleam tests can be run with cargo run && gleam test
.