aboutsummaryrefslogblamecommitdiffstats
path: root/askama_derive/src/generator.rs
blob: b1babceff91acac1cef3df3160e6d7a4d0deb829 (plain) (tree)
1
2
3
4
5
6
7
8
9
10
                         

             

                  

                


                
 
                           
                                                                
     



                                                 












                                                     

     
                                  





                                           

                             
 








                                    
                                       
                                    
                                                                  
                           

     
                                       



                                                                             

     





                                               



                                                            
                                                                   




                            




                            

     


                               
 

 
                                                               
                                   


                       
                
 
use parser::{Expr, Node};
use std::str;

struct Generator {
    buf: String,
    indent: u8,
    start: bool,
}

impl Generator {

    fn new() -> Generator {
        Generator { buf: String::new(), indent: 0, start: true }
    }

    fn init(&mut self, name: &str) {
        self.write("impl askama::Template for ");
        self.write(name);
        self.writeln(" {");
        self.indent();
        self.writeln("fn render(&self) -> String {");
        self.indent();
        self.writeln("let mut buf = String::new();");
    }

    fn indent(&mut self) {
        self.indent += 1;
    }

    fn dedent(&mut self) {
        self.indent -= 1;
    }

    fn write(&mut self, s: &str) {
        if self.start {
            for _ in 0..(self.indent * 4) {
                self.buf.push(' ');
            }
            self.start = false;
        }
        self.buf.push_str(s);
    }

    fn writeln(&mut self, s: &str) {
        if s.is_empty() {
            return;
        }
        self.write(s);
        self.buf.push('\n');
        self.start = true;
    }

    fn visit_lit(&mut self, s: &[u8]) {
        self.write("buf.push_str(");
        self.write(&format!("{:#?}", str::from_utf8(s).unwrap()));
        self.writeln(");");
    }

    fn visit_var(&mut self, s: &[u8]) {
        let var_name = str::from_utf8(s).unwrap();
        let code = format!("std::fmt::Write::write_fmt(\
            &mut buf, format_args!(\"{{}}\", self.{})).unwrap();", var_name);
        self.writeln(&code);
    }

    fn visit_expr(&mut self, expr: &Expr) {
        match expr {
            &Expr::Var(s) => self.visit_var(s),
        }
    }

    fn handle(&mut self, tokens: &Vec<Node>) {
        for n in tokens {
            match n {
                &Node::Lit(val) => { self.visit_lit(val); },
                &Node::Expr(ref val) => { self.visit_expr(&val); },
            }
        }
    }

    fn finalize(&mut self) {
        self.writeln("buf");
        self.dedent();
        self.writeln("}");
        self.dedent();
        self.writeln("}");
    }

    fn result(self) -> String {
        self.buf
    }

}

pub fn generate(ctx_name: &str, tokens: &Vec<Node>) -> String {
    let mut gen = Generator::new();
    gen.init(ctx_name);
    gen.handle(tokens);
    gen.finalize();
    gen.result()
}