![Travis Build Status] ![Appveyor Build Status] ![Latest Version]
[gtmpl-rust] provides the [Golang text/template] engine for Rust. This enables seamless integration of Rust application into the world of devops tools around [kubernetes], [docker] and whatnot.
Add the following dependency to your Cargo manifest…
toml
[dependencies]
gtmpl = "0.5.5"
and look at the docs: * gtmpl at crates.io * gtmpl documentation * golang documentation
It's not perfect, yet. Help and feedback is more than welcome.
Basic template: ```rust extern crate gtmpl; use gtmpl;
fn main() { let output = gtmpl::template("Finally! Some {{ . }} for Rust", "gtmpl"); assert_eq!(&output.unwrap(), "Finally! Some gtmpl for Rust"); } ```
Adding custom functions: ```rust
extern crate gtmpl; extern crate gtmplvalue; use gtmplvalue::Function; use gtmpl::{template, Value};
fn main() {
gtmplfn!(
fn add(a: u64, b: u64) -> Result
Passing a struct as context: ```rust
extern crate gtmpl;
extern crate gtmplderive; extern crate gtmplvalue;
struct Foo { bar: u8 }
fn main() { let foo = Foo { bar: 42 }; let output = gtmpl::template("The answer is: {{ .bar }}", foo); assert_eq!(&output.unwrap(), "The answer is: 42"); } ```
Invoking a method on a context: ```rust
extern crate gtmpl;
extern crate gtmplderive; extern crate gtmplvalue;
use gtmpl::{Func, Value};
fn plusone(args: &[Value]) -> Result
struct AddMe { num: u8, plus_one: Func }
fn main() { let addme = AddMe { num: 42, plusone }; let output = gtmpl::template("The answer is: {{ .plusone }}", addme); assert_eq!(&output.unwrap(), "The answer is: 43"); } ```
This is work in progress. Currently the following features are not supported:
html
, js
printf
is not yet fully stable, but should support all sane inputEven though it was never intended to extend the syntax of Golang text/template there might be some convenient additions:
Enable gtmpl_dynamic_template
in your Cargo.toml
:
```toml
[dependencies.gtmpl]
version = "0.5.5"
features = ["gtmpldynamictemplate"]
```
Now you can have dynamic template names for the template
action.
```rust extern crate gtmpl; use gtmpl::{Context, Template};
fn main() { let mut template = Template::default(); template .parse( r#" {{- define "tmpl1"}} some {{ end -}} {{- define "tmpl2"}} some other {{ end -}} there is {{- template (.) -}} template "#, ) .unwrap();
let context = Context::from("tmpl2").unwrap();
let output = template.render(&context);
assert_eq!(output.unwrap(), "there is some other template".to_string());
} ```
The following syntax is used: ``` {{template (pipeline)}} The template with the name evaluated from the pipeline (parenthesized) is executed with nil data.
{{template (pipeline) pipeline}} The template with the name evaluated from the first pipeline (parenthesized) is executed with dot set to the value of the second pipeline. ```
We use [gtmplvalue]'s Value as internal data type. [gtmplderive] provides a
handy derive
macro to generate the From
implementation for Value
.
See:
Why? Dear god, why? I can already imagine the question coming up why anyone would ever do this. I wasn't a big fan of Golang templates when i first had to write some custom formatting strings for docker. Learning a new template language usually isn't something one is looking forward to. Most people avoid it completely. However, it's really useful for automation if you're looking for something more lightweight than a full blown DSL.
The main motivation for this is to make it easier to write devops tools in Rust that feel native. [docker] and [helm] ([kubernetes]) use golang templates and it feels more native if tooling around them uses the same.