Inxedu 系统代码审计

因酷开源网校系统是由北京因酷时代科技有限公司以下简称(因酷教育软件)研发并推出的国内首家 Java 版开源网校源代码建站系统,并免费提供给非商业用途用户使用,是用户体验最好、运营功能最全、性价比最高的在线教育软件。

该项目是 SSM 框架

项目部署:

1. 配置数据库 (这里我弄的时候存在一个问题:由于 Linux 系统对大小写敏感,所以如果在类 Unix 系统上部署环境,代码中数据库中数据表的大小写会影响,建议将代码中数据表的名字都统一修改为小写)

image-20240706230313281

2. 配置 tomcat

image-20240706230130768

前台账户密码:

lmx193@163.com

111111

后台账户密码:

admin

111111

  • web.xml:程序启动时 tomcat 会首先加载 web.xml 中的配置 。通过 web.xml 完成 DispathcheServlet 的声明,并将我们的请求转发到 springmvc 中。我们可以首先查看 web.xml 中是否配置了全局过滤器。判断是否能够 bypass。

  • applicationContext.xml 是 spring 核心配置文件,这里会加载一些其他的配置文件。

  • sping-mvc.xml 文件中主要的工作是:启动注解、扫描 controller 包注解;静态资源映射;视图解析(defaultViewResolver);文件上传(multipartResolver); 返回消息 json 配置。

审计一套系统,可以先看看 pom.xml 中加载了那些组件 ,如果这些组件中本身存在漏洞,就可以直接利用这些漏洞。

  1. 以文件上传为例,先黑盒寻找并测试文件上传功能点抓包得到对应路由等信息,从而定位到功能实现的具体代码(代码溯源),正向然后审计其是否存在漏洞。
  2. 通过关键字搜索关于上传功能的代码,查看是否存在漏洞,如果存在则反向追踪代码,找到路由

关于 1 day 也是类似的思路,通过网上爆出的路由或者 payload(或者通过看 comment)来在代码中整体搜索定位漏洞点,然后逆向追踪

这里正向追踪:先黑盒寻找上传文件的接口 / 功能点 —>burp 抓包找到接口地址 —> 然后去代码中寻找 (全局搜索 /controller 层找)

登录学员账号,寻找上传点,发现头像上传功能,测试:

image-20240706232650331

看到请求地址为:

image-20240706232753495

去代码中搜索 image/jok4,没搜到:

image-20240706232831158

没搜到原因有几种:

  • 搜索时候注意:比如 image/gok4 这种两个路径可能是拼接起来的 (比如 controller 的类有一个路径 + 方法上也有路径),所以可以分开去搜 imagegok4

  • 这个项目中,这个路径是引入的自己打包的第三方 jar 包中代码中的,IDEA 的搜索是无法搜索 jar 包里面关键字的(实际审计中,可以使用工具进行扫描有无关键字)。实际上这个项目是将处理文件上传的代码封装到了一个 Jar 包然后引用的,一般情况下如果碰到搜不到的情况,就去 pom.xml 里看看有没有声明依赖的 Jar 包

note: 找到代码中接口后(如果看不懂是干啥的,可以结合数据包判断是干啥的功能)

我们定位到了代码位置:

image-20240706233241597

然后继续正向往下分析代码,依次看 controller 层–>service 层–>dao 层 —>mapper 文件

可以定位到 gok4 方法:

1
2
3
4
5
6
7
8
该方法的逻辑如下
1检查上传文件的大小如果超过4M则返回错误信息   //--->uploadfile参数为前端传递的文件
2根据fileType参数判断文件后缀(使用getSuffix方法)是否符合要求如果文件类型不符合或者后缀为jsp则返回错误信息 //--->fileType参数为前端传递
3根据上传文件的后缀和param参数确定上传文件的保存路径  //--->param参数为前端传递
4创建保存文件的目录如果目录不存在)。
5将上传文件保存到指定的路径(通过transferTo())
6返回上传成功的响应信息包括文件路径和状态码
7如果发生异常记录错误日志并返回上传失败的错误信息

image-20240706233426295

跟进 getSuffix 中,发现其代码中除了获取文件后缀其他没什么过滤:

1
2
3
4
    public static String getSuffix(String str) {
        return str.substring(str.lastIndexOf(".") + 1);
    }
}

再结合之前抓包的 url:http://127.0.0.1:8080/image/gok4?&param=temp&fileType=jpg,gif,png,jpeg

将已知的信息串联起来,总结这段代码的判断逻辑为:文件后缀包含 jpg,gif,png,jpeg,且不为 jsp 即进行下一步上传操作。

可以看到 param 和 fileType 为代码中我们可控的

我们的思路:

  1. 上传一个不是 jsp 的且可以解析的,比如 jspx,即 http://127.0.0.1:8080/image/gok4?&param=temp&fileType=jpg,gif,png,jpeg,jspx
  2. 利用 windows 的特性,比如::$DATA,即 http://127.0.0.1:8080/image/gok4?&param=temp&fileType=jpg,gif,png,jpeg,jsp$$DATA

我们使用冰蝎自带的 jsp 测试:

image-20240707095232111

成功上传:

image-20240707095551892

逆向追踪:先通过 "在文件中查找" 来搜索文件上传功能的关键字(搜索时候后缀限制为.java),然后反向回溯功能点

文件上传关键字:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
File
java.io.File
FileUpload
FileUploadBase
FileItemIteratorImpl
FileItemStreamImpl
FileUtils
UploadHandleServlet
FileLoadServlet
FileOutputStream
DiskFileItemFactory
MultipartRequestEntity
MultipartFile
com.oreilly.servlet.MultipartRequest
MultipartHttpServletRequest
org.apache.commons.fileupload
MultipartFile
CommonsMultipartResolver
......

我们搜索 FileUpload

image-20240707095837379

位置 com/inxedu/os/common/controller/VideoUploadController.java

看起来是一个视频上传的功能:

image-20240707100211247

可以直接上传 jsp

我们回溯功能点在哪里(好了,它在 controller 里面,不用回溯了)

可知该上传接口为:/video/uploadvideo

image-20240707100345336

构造:

方法 1: 找到该接口在系统中对应的位置

方法 2:自己构造文件上传的表单 (接口改为该接口即可)(简便)

构造表单:

1
2
3
4
5
<form action="http://192.168.3.214:8080/video/uploadvideo" enctype="multipart/form-data" id="frmUpload" method="post">
<input name="uploadfile" type="file">
<input type ="text" name = "fileType" value="">
<input id="btnUpload" type="submit" value="上传">
</form>

image-20240707100632667

抓包修改

image-20240707103427050

我这里测试一直失败报服务器错误,可能是环境搭建的问题,先不管了

note:解决 IDEA 下 debug 断点进不去:

https://blog.csdn.net/xujie102360/article/details/81476774

https://blog.csdn.net/searlas/article/details/80826777

image-20240707110830424

已知系统使用的是 mybatis,mybatis 最常见的注入就是错误使用 ${} 导致,

所以我们直接搜索 ${,看看哪些是通过 ${} 取值并且没有过滤的

image-20240707104233314

找到了一处,我们逆向追踪调用逻辑,寻找是否含有过滤并且参数我们是否可控。

daoImpl:

image-20240707104350963

dao:

image-20240707104404853

ctrl + 单击,来到 ServiceImpl:

image-20240707104447938

参数为 ids,由形参 articleIds 而来

也没有进行过滤

service:image-20240707104550997

往上走去找 controller:传递的参数为 artidArr,也确实没有过滤

image-20240707104617107

deleteArticle 方法在这里被调用:参数为前端传来的 articelId,我们可控:

image-20240707104735651

image-20240707104813816

可知接口为:/admin/article/delete

打开系统登录后台。找到文章管理部分,选择删除,抓包

image-20240707105658610

直接将数据包放到 sqlmap 中跑:

image-20240707105645898

可以看到这套系统中还有多处存在 SQL 注入漏洞:

image-20240707105811068

待完成

待完成

0%