blob: cdb0dd998266d5c7688a342e26c0ce0c1020ed4f (
plain) (
tree)
|
|
extern crate askama;
extern crate proc_macro;
extern crate syn;
use proc_macro::TokenStream;
use std::fs::File;
use std::io::Read;
use std::path::{Path, PathBuf};
fn get_path_from_attrs(attrs: &Vec<syn::Attribute>) -> String {
for attr in attrs {
if attr.name() == "template" {
match attr.value {
syn::MetaItem::List(_, ref inner) => {
match inner[0] {
syn::NestedMetaItem::MetaItem(ref item) => {
match item {
&syn::MetaItem::NameValue(ref key, ref val) => {
assert_eq!(key.as_ref(), "path");
match val {
&syn::Lit::Str(ref s, _) => { return s.clone(); },
_ => panic!("template path must be a string"),
}
},
_ => panic!("template annotation must contain key/value pair"),
}
},
_ => panic!("template annotation must contain item"),
}
},
_ => panic!("template annotation must be of List type"),
}
}
}
panic!("template annotation not found");
}
fn get_template_source(tpl_file: &str) -> String {
let root = ::std::env::var("CARGO_MANIFEST_DIR").unwrap();
let mut path = PathBuf::from(root);
path.push("templates");
path.push(Path::new(tpl_file));
let mut f = File::open(path).unwrap();
let mut s = String::new();
f.read_to_string(&mut s).unwrap();
s
}
#[proc_macro_derive(Template, attributes(template))]
pub fn derive_template(input: TokenStream) -> TokenStream {
let source = input.to_string();
let ast = syn::parse_derive_input(&source).unwrap();
let _ctx = match ast.body {
syn::Body::Struct(ref data) => data,
_ => panic!("#[derive(Template)] can only be used with structs"),
};
let path = get_path_from_attrs(&ast.attrs);
let src = get_template_source(&path);
let tokens = askama::parser::parse(&src);
askama::generator::generate(&ast, &tokens).parse().unwrap()
}
|