长亭百川云 - 文章详情

nacos_后台rce分析

漏洞推送

97

2024-08-08

环境搭建

适用 vulhub这个项目来启动漏洞环境

vulhub的nacos版本是 1.4.0,通过github下载源代码

https://github.com/alibaba/nacos/tree/1.4.0

因为vulhub已经默认开放了debug端口,所以直接自己idea jvm远程调试即可

漏洞分析

从 poc来看 问题出在 /nacos/v1/cs/ops/data/removal
 和 /nacos/v1/cs/ops/derby
 这两个接口

removal 代码位于 config/src/main/java/com/alibaba/nacos/config/server/controller/ConfigOpsController.java:133

importDerby方法使用了Secured
注解进行保护

Secured注解的实现在 com/alibaba/nacos/core/auth/AuthFilter.java:88

首先判断了有没有开启认证,如果没有开启就直接pass

如果开启了认证 还会判断 ua是否是 Nacos-Server开头,如果是的话就pass。这就是前几年的未授权漏洞的根源

在vuln 中的nacos是没有开启auth的,所以无需关注认证

漏洞接口代码

@PostMapping(value = "/data/removal")
    @Secured(action = ActionTypes.WRITE, resource = "nacos/admin")
    public DeferredResult<RestResult<String>> importDerby(@RequestParam(value = "file") MultipartFile multipartFile) {
        DeferredResult<RestResult<String>> response = new DeferredResult<>();
        if (!PropertyUtil.isEmbeddedStorage()) {
            response.setResult(RestResultUtils.failed("Limited to embedded storage mode"));
            return response;
        }
        DatabaseOperate databaseOperate = ApplicationUtils.getBean(DatabaseOperate.class);
        WebUtils.onFileUpload(multipartFile, file -> {
            NotifyCenter.publishEvent(new DerbyImportEvent(false));
            databaseOperate.dataImport(file).whenComplete((result, ex) -> {
                NotifyCenter.publishEvent(new DerbyImportEvent(true));
                if (Objects.nonNull(ex)) {
                    response.setResult(RestResultUtils.failed(ex.getMessage()));
                    return;
                }
                response.setResult(result);
            });
        }, response);
        return response;
    }

接受一个文件,临时储存在/tmp目录下,

然后调用databaseOperate.dataImport
方法进行导入,然后继续调用父类的doDataImport方法进行导入

继续调用,在batchUpdate执行了sql语句

这里调用的是spring中的template库来执行sql语句,template中的jdbcurl是

jdbc:derby:/opt/nacos/data/derby-data;create=true

derby 是一个开源的关系型数据库,是嵌入式数据库。于java中H2类似

相关参考文档:

https://db.apache.org/derby/docs/10.17/devguide/rdevdeploy856845.html

https://db.apache.org/derby/docs/10.17/devguide/cdevdeploy21645.html

可以通过远程安装一个java包,将一个或多个 jar 文件添加到数据库后,必须通过在 derby.database.classpath 属性中包含一个或多个 jar 文件来设置数据库 jar 类路径,以使 Derby 能够从 jar 文件加载类。

然后创建一个自定义函数,这个select 这个自定义函数来触发命令执行

select * from (select count(*) as b, S_EXAMPLE_{id}('{command}') as a from config_info) tmp /ROWS FETCH NEXT/

这个时候就产生了一个问题,为什么通过/nacos/v1/cs/ops/data/removal
 已经能执行sql语句了,为什么还要用/nacos/v1/cs/ops/derby
这个接口来执行select语句。

因为这里执行语句的是template.batchUpdate
函数,只能执行update相关语句无法执行select语句。

derby接口执行语句是用的template.queryForList
所以可以执行select语句

修复措施

看官方的最新的代码,应该是默认禁用derby数据库了

通过docker搭建一个最新版运行一下

docker run --name nacos-quick -e MODE=standalone -p 8849:8848 -d nacos/nacos-server:2.0.2

执行poc 确实已经关闭了

相关推荐
关注或联系我们
添加百川云公众号,移动管理云安全产品
咨询热线:
4000-327-707
百川公众号
百川公众号
百川云客服
百川云客服

Copyright ©2024 北京长亭科技有限公司
icon
京ICP备 2024055124号-2