【问题标题】:Jinja2 extension outputs escaped html instead of html tagJinja2 扩展输出转义的 html 而不是 html 标签
【发布时间】:2015-03-31 14:05:51
【问题描述】:

我正在尝试编写一个简单的jinja2 扩展,它将在页面中呈现一个带有一些属性和内容属性的<meta> 标记。它看起来像这样:

from jinja2 import nodes
from jinja2.ext import Extension

class MetaExtension(Extension):
    """
    returns a meta tag of key, value
    >> env = jinja2.Environment(extensions=[MetaExtension])
    >> env.from_string('{% meta "key", "value" %}').render()
       u'<meta property="keyword" content="value" />'
   """

    # we'll use it in the template with
    tags = set(['meta'])

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

    def parse(self, parser):
        tag = parser.stream.next()
        args = [parser.parse_expression()]

        if parser.stream.skip_if('comma'):
            args.append(parser.parse_expression())

        return nodes.Output([self.call_method('_render', args)]).set_lineno(tag.lineno)

    def _render(self, *args):
        return '<meta property="{}" content="{}" />'.format(args[0], args[1])

meta = MetaExtension

在烧瓶应用中使用它:

# register the extension in the app file
from flask import Flask
app = Flask(__name__)
app.jinja_env.add_extension('flask_meta.MetaExtension')

还有

<!-- use it in the template -->
<head>
  {% meta "key", "value" %}
</head>

虽然标签在控制台中渲染得很好,但当我在烧瓶应用程序中使用它时,它会转义 html 并将转义的标签输出到页面。所以,而不是

<meta property="keyword" content="value" />

我明白了

&lt;meta property=&#34;key&#34; content=&#34;value&#34; /&gt;

我做错了什么?谢谢。

更新:请参阅下面的答案。

【问题讨论】:

  • 您应该将您的解决方案添加为答案并将其标记为正确答案。

标签: python jinja2 flask-extensions html-escape


【解决方案1】:

这段代码产生...

<meta property="('this', 'that')" content="('this2', 'that2')" />

试试这个...

from jinja2 import nodes
from flask import Flask
from jinja2.ext import Extension

app = Flask(__name__)

class MetaExtension(Extension):
        """
        returns a meta tag of key, value
        >> env = jinja2.Environment(extensions=[MetaExtension])
        >> env.from_string('{% meta "key", "value" %}').render()
           u'<meta property="keyword" content="value" />'
       """

        # we'll use it in the template with
        tags = set(['meta'])

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

        def parse(self, parser):
            tag = parser.stream.next()
            args = [parser.parse_expression()]

            if parser.stream.skip_if('comma'):
                args.append(parser.parse_expression())

            return nodes.Output([self.call_method('_render', args)]).set_lineno(tag.lineno)

        def _render(self, *args):
            return '<meta property="{0}" content="{1}" />'.format(args[0], args[1])

meta = MetaExtension


@app.route("/")
def index():
    #return 'hi'
    try:
        return MetaExtension._render('', ("this", 'that') , ("this2", 'that2') )
    except  Exception as e:
        print(str(e))

app.jinja_env.add_extension(meta)

if __name__ == "__main__":
    app.run()

希望这会有所帮助!

【讨论】:

  • 谢谢。我的目标是一个简单的 API,结果发现我的错误是返回 nodes.Output 而不是 nodes.CallBlock - 用解决方案更新了我的问题。
【解决方案2】:

原来我应该使用nodes.CallBlock 而不是nodes.Output。应用该更改,parse_render 函数现在看起来像:

...
def parse(self, parser):
    tag = parser.stream.next()
    args = [parser.parse_expression()]

    if parser.stream.skip_if('comma'):
        args.append(parser.parse_expression())

    return nodes.CallBlock(self.call_method('_render', args),
            [], [], []).set_lineno(tag.lineno)

def _render(self, value, name, *args, **kwargs):
    return '<meta property="{}" content="{}" />'.format(value, name)

希望这对某人有所帮助,因为 Jinja2 扩展参考(尤其是对于非块标签)并不容易获得 :)

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2016-11-14
    • 2019-04-27
    • 2012-09-15
    • 2017-07-31
    • 2012-04-21
    • 2016-07-15
    相关资源
    最近更新 更多