2

可以编译或还原 Jinja2 AST 的一部分吗?

例如,是否可以从从模板中提取的节点列表中调用函数或方法,jinja2.environment或从节点列表中调用函数或方法?jinja2.compiler.generate

例如,给定一个模板y.html

avant-tag
{% xyz %}
tag content {{ 3 + 5 }}
{% endxyz %}
apres-tag

和一个扩展y.py

# -*- coding: utf-8 -*-
from jinja2 import nodes, Environment, FileSystemLoader
from jinja2.ext import Extension

class YExtension(Extension):
    tags = set(['y'])

    def __init__(self, environment):
        super(YExtension, self).__init__(environment)

    def parse(self, parser):
        tag = parser.stream.next()
        body = parser.parse_statements(['name:endy'], drop_needle=True)
        return nodes.Const("<!-- slurping: %s -->" % str(body))

env = Environment(
    loader      = FileSystemLoader('.'),
    extensions  = [YExtension],
    )

print env.get_template('x.html').render()

运行python y.py导致预期输出:

avant-tag
 <!-- slurping: [Output(nodes=[TemplateData(data=u'\n    tag-content '),
   Add(left=Const(value=3), right=Const(value=5)),
   TemplateData(data=u'\n ')])] -->
sous-tag

在该parse方法中,如何可以:

  1. 编译body为 unicode(即tag-content 8);或者,或者,
  2. 恢复body到其原始来源(即tag-content {{ 3 + 5 }})。

作为背景问题,这个问题涉及到两个先前的问题:

  1. Jinja2 在包含后编译扩展;和
  2. 在 Jinja 2 中包含文件的顶部插入 javascript

感谢您的阅读。

布赖恩

4

1 回答 1

1

在该方法中还无法编译为 unicode,parse()因为此时您没有可用的上下文。你当然可以绕过它,但这可能不是最好的方法。

请注意,该parse()步骤通常只在 html 文件上执行一次,之后它将使用解析的字节码来呈现模板。解析步骤的结果可以在给定的环境中呈现。

您根本没有那里可用的上下文,并且在那里获得上下文......非常困难;)

然而,要获得原始来源......没有黑客攻击不会容易得多,但黑客攻击还不错;)

class YExtension(Extension):
    tags = set(['y'])

    def preprocess(self, source, name, filename=None):
        # insert some code here that replaces '{% xyz %}foo bar{% endxyz %}'
        # with something like: '{% xyz %}foo bar{% raw %}foo bar{% endraw %}{% endxyz %}'
        return source

之后,您可以value{% raw %}节点读取文本。确保在此之后将其丢弃,否则它将显示在您的模板中。

于 2010-11-30T01:19:39.307 回答