用python寫自定義模板

模板語法有點像phpphp

 

!/usr/bin/env python
"""


#demo.py.html


<html>
<?py include head.py.html ?>
<body>
    <?py extend test_for.py.html ?>
    <div>
        <pre> test if </pre>
        <?py if 2 > 1 ?>
        <p>Yes, it is</p>
        <?py else ?>
        <p>No it is not</p>
        <?py #endif ?>
    </div>

    <div>
        <pre> test def </pre>
        <?py def p() ?>
            <?py return 'hello' ?>
        <?py #enddef ?>
        Wow! ${p()}
    </div>
</body>
</html>


#head.py.html
<head>
    <title>Demo Py Template</title>
</head>

#test_for.py.html
<div>
    <pre> test for <pre>
    <?py for _ in ['a', 'b', 'c'] ?>
    Hello ${_} #{_.upper()} 
    <?py #endfor ?>
</div>
"""
 
from __future__ import absolute_import, division, print_function, with_statement
import re
import os.path
 
 
class TemplateLoader(object):
    pass
 
 
class DictLoader(object):
    pass
 
 
class Template(object):
 
    #GLOBALS = {}
 
    def __init__(self, tempalte_path):
        self.tempalte_path = tempalte_path
 
        self.globals = {
            't': lambda x: x,
            'xml_escape': xml_escape,
            'html_escape': html_escape,
            'e': escape
        }
 
        self.templates = {
 
        }
 
    def render(self, path, **kwrags):
        # the stemplae system local never be override
        _g = kwargs.update(self.globals)
        code = self.genarate(path, g)
        exec code in _g
        return _g['_tt_render']()
 
    def genarate(self, path, g):
        if path in self.templates:
            return self.templates[path](g)
 
        tempalte_path = os.path.join(self.tempalte_path, path)
        tempalte_func = TemplateParser(tempalte_path).compile()
        self.templates[path] = tempalte_func
        return tempalte_func(g)
 
 
class TemplateParser(object):
 
    PY_TOKEN = re.compile(r'<\?py\s*((?:[^=0-9]).*?)\s*\?>')
    PY_VAR_TOKEN = re.compile(r'(?:[#$])\{(.*?)\}')
 
    def __init__(self, path2template, indent=None, include=False):
        self.path2template = path2template
        self.indent = indent or 1
        self.buffer = ''
        self._include = include
        if not self._include:
            self.buffer += 'def _tt_render():\n'
            self.puts('_buffer=[]')
            self.puts('_append=_buffer.append')
 
    def puts(self, line):
        self.buffer += self.indents + "%s" % (line) + '\n'
 
    def compile(self):
        code = self.parse()
        code = compile(code, '<string>', 'exec', dont_inherit=True)
        return code
 
    def parse(self):
        f = open(self.path2template)
 
        lineno = 1
        while True:
            line = f.readline()
            if not line:
                break
            self.parse_line(line, lineno)
            lineno += 1
        if not self._include:
            self.puts('return "".join(_buffer)')
        f.close()
 
        return self.buffer
 
    def parse_line(self, line, lineno):
 
        # remove the whitespace line and comment line
        if not line.strip() or line.lstrip().startswith('#'):
            return
 
        m = self.PY_TOKEN.search(line)
        if m:
            t = m.group(1)
            parts = t.split(' ', 1)
            if parts[0] in ('from', 'import'):
                self.stmt(t)
                return
 
            if parts[0] == 'include':
                self.include(parts[1])
                return
 
            if parts[0] == 'extend':
                self.extend(parts[1])
                return
 
            if parts[0] in ('for', 'if', 'with', 'def', 'class', 'try'):
                self.contoll(t)
                self.indent += 1
                return
 
            if parts[0] in ('else', 'elif', 'except'):
                self.indent -= 1
                self.contoll(t)
                self.indent += 1
                return
 
            if parts[0] in ('#end', '#endfor', '#endif', '#endtry', '#endclass', '#enddef', '#endwith'):
                self.indent -= 1
                return
 
            self.stmt(t)
            return
 
        # handle var token
        ms = self.PY_VAR_TOKEN.finditer(line)
        if ms:
            a = None
            end = 0
            start = 0
            l = ''
            for m in ms:
                t = m.group(1)
                start = m.start()
 
                b = line[end:m.start()]
                start = m.start()
                end = m.end()
                a = line[end:]
                if l:
                    l += "+ '%s'" % (b) + " + str(%s) " % (t)
                else:
                    l = "'%s'" % (b) + " + str(%s) " % (t)
 
            if a and start:
                l += "+" + " %r" % (a)
            if l:
                self.stmt("_append(" + l + ")")
                return
 
        self.text(line)
 
    def text(self, content):
        content = self.indents + '_append(' + "%r" % (content) + ')\n'
        self.buffer += content
 
    def e(self, formator, t):
        self.puts('_append("' + formator + '"' + ' % ' + t + ")")
 
    def r(self):
        return '%r'
 
    def contoll(self, line):
        self.buffer += self.indents + line + ':\n'
 
    def stmt(self, stmt):
        self.buffer += self.indents + stmt + ' \n'
 
    @property
    def indents(self):
        return '    ' * self.indent
 
    def include(self, template):
        p = TemplateParser(template, include=True)
        self.buffer += p.parse()
 
    def extend(self, template):
        p = TemplateParser(template, self.indent, include=True)
        self.buffer += p.parse()
 
 
if __name__ == '__main__':
    t = TemplateParser('demo.py.html')
 
    code = t.compile()
    ns = {}
    exec code in ns
    # print ns
    print(ns['_tt_render']())
相關文章
相關標籤/搜索