7 releases

0.3.3 Dec 10, 2024
0.3.2 Oct 29, 2024
0.3.1 Sep 29, 2024
0.2.0 Sep 28, 2023
0.1.0 Jul 3, 2023

#182 in Procedural macros

Download history 110/week @ 2024-09-17 215/week @ 2024-09-24 58/week @ 2024-10-01 15/week @ 2024-10-08 132/week @ 2024-10-29 3/week @ 2024-11-05 2/week @ 2024-11-12 22/week @ 2024-11-19 5/week @ 2024-11-26 29/week @ 2024-12-03 176/week @ 2024-12-10 23/week @ 2024-12-17 22/week @ 2024-12-24

222 downloads per month
Used in 2 crates (via stilts-macros)

MIT license

43KB
1K SLoC

Stilts

Documentation Package

Stilts is a templating language inspired by Askama. That means it features all the type safety the rust compiler will give you. It is still loosly related to Jinja, however it has adopted a more rust-like expression syntax.

I started this project because rust is a great language, and Jinja2 is the template engine I was most used to. I however felt that one thing Jinja always lacked was expressivity, in the sense that you couldn't write arbitrary python code into the template. This limitation is preserved in other libraries like Askama, Tera, and minijinja. While those projects are all great, I personally found them lacking in usability.

This project while much younger than Askama and other template engines, is in a state that I believe to be very usable, in fact it is being used in production for multiple projects in my office.

If you have suggestions, features, questions, or anything else please feel free to open an issue. I could especially use extra help writing unit tests for stuff, and would be open to working with others to get that done.

Guide

Check out the book for in depth documentation.

How it works

Stilts uses a procedural derive macro on a struct to generate template rendering code which results in the template code being checked by the rust compiler for correctness.

A Quick Example

Here is what some rust code defining a template looks like

use stilts::Template;

#[derive(Template)] // This derive macro will generate the template rendering code
#[stilts(path = "index.html")] // based on the contents of the given template path
struct IndexPage<'a> {
    some_data: &'a str, // The fields on a struct are the variables 
}                       // you want to use in your template

Here is what index.html could look like

<!DOCTYPE html>
<html>
    <body>
        <h1>My Index Page</h1>
        {% some_data %} <!--This will print some_data to the template here using 
                            the types implementaion of the Display trait-->
    </body>
</html>

About Stilts Expressions

In Stilts an expression is made up of two variants single and block expressions. A single expression is anything inside the delimiters {%%}. Whereas a block expression has an opening single expression and a closing single expression with some kind of content in-between. For example:

{% for i in 0..10 %}
    This will be repeated 10 times
    {% i %}
    And I can put other expressions inside
{% end %}

Something you might notice if you have used Askama or Jinja before is that normally to render the value of something you have to use a different set of delimiters namely something like {{ some_data }}. But here in Stilts there is only one set of delimiters {% some_data %}.

So how do we determine when the user wants to render something or just write some code? Well the answer is by determining if the rust code inside the delimiters is an expression or a statement. In rust an expression always produces a value and a statement doesn't.

Therefore if as a user we want to write some code but not render it to the template all we have to do is add a semicolon. {% some_data; %} is a statement now not an expression which means Stilts will insert that into the template rendering code as code to run and not render it to the resulting template. This is familiar to many rust developers as it is the same way we can omit the return keyword inside of functions by just ending it with an expression.

What this also means is that you can put any arbitrary rust expression or statement inside of the delimiters. For example {% let myval = some_data.split(' ').filter(|s| s != "abcd"); %}. But you aren't limited to a single line either you can split them into as many lines as you want.

{% fn my_useful_func() {
    // do some stuff
} %}

Features

  • The syntax is similar to Jinja but also more closely tied to rust
  • Utilize rust's type system to verify your code
  • Achieves performance comparable to Askama which is already very good
  • Good error messages and formatting especially with optional fancy feature
  • Works on stable rust

Supported Template Constructs

  • Inheritance
  • for loops
  • if/ else if /else
  • match expressions
  • includes statements
  • variables (as a result of allowing arbitrary rust expressions and statements)
  • useful extension traits imported into scope
  • Configurable Opt-out HTML escaping

Goals

  • Create a templating language that is both familiar to jinja users and rust developers
  • Have good error reporting and messages with useful error locations.

Other Stuff

In the tooling directory there is a work-in-progress tree-sitter parser implementation. If you know better how to get that working that would be super cool.


lib.rs:

Stilts Lang is the parser for the stilts language Parse a full template with parse_template the output is is an ast whose definition is in the types module

Dependencies

~3–13MB
~158K SLoC