14-1
约 786 字大约 3 分钟
2025-11-02
CC1、CC6 区别
Commons Collections 1 (CC1) 利用链
CC1 是最经典、最广为人知的 Commons Collections 反序列化利用链。它利用了 InvokerTransformer 这个转换器来反射调用任意方法
利用链核心原理
CC1 利用链的核心思想是通过调用链,最终在反序列化过程中触发 TemplatesImpl 类的 newTransformer 方法,从而执行任意命令
InvokerTransformer:这是核心组件。它的transform()方法能够通过反射调用任意对象的任意方法。攻击者可以利用它来调用Runtime.getRuntime().exec()InstantiateTransformer:这个转换器用于实例化一个对象,其transform()方法会调用构造函数。在利用链中,它常用来实例化InvokerTransformer对象LazyMap:这是一个延迟加载的 Map,它的get()方法会在键不存在时调用一个预设的转换器(Transformer)。攻击者可以将InvokerTransformer作为这个转换器,当对一个不存在的键进行get()操作时,就会触发InvokerTransformer的transform()方法AnnotationInvocationHandler或BadAttributeValueExpException:CC1 利用链通常需要一个入口点,来触发LazyMap的get()方法。在旧版本的 JDK 中,AnnotationInvocationHandler的readObject()方法会在反序列化时自动调用其内部的Proxy对象的invoke()方法,从而间接触发LazyMap的get()。对于新版本的 JDK,由于对AnnotationInvocationHandler进行了限制,攻击者转而利用BadAttributeValueExpException的readObject()方法
Commons Collections 6 (CC6) 利用链
CC6 旨在解决 CC1 在较新版本的库和 JDK 中失效的问题。它抛弃了 CC1 中常用的 InvokerTransformer,转而利用 TiedMapEntry 和 LinkedSet 等新的类来构造利用链
利用链核心原理
CC6 的核心思想是利用 TiedMapEntry 在反序列化时触发 Map 的 get() 方法,最终同样达到命令执行的目的
TiedMapEntry:这是 CC6 利用链的核心。它的toString()方法在调用时会触发其内部Map的get()方法AbstractMap$TansformMapDecorator:这是一个装饰器,它装饰了一个 Map,并用一个Transformer来处理其键值TransformedMap:当向这个 Map 添加元素时,其put()方法会调用一个预设的TransformerLinkedSet:在 CC6 的利用中,攻击者通常会利用LinkedSet的equals()方法,该方法会遍历集合中的元素并调用它们的equals()。通过精心构造LinkedSet,可以使其内部的TiedMapEntry实例的toString()方法被调用InvokerTransformer(再次出现):尽管 CC6 旨在避开InvokerTransformer,但在某些变体中,它仍然可以作为最终的命令执行器。不同的是,CC6 利用链的触发点不再是 CC1 中的LazyMap
| 特性 | Commons Collections 1 (CC1) | Commons Collections 6 (CC6) |
|---|---|---|
| 核心触发器 | LazyMap 的 get() 方法 | TiedMapEntry 的 toString() 方法 |
| 主要攻击类 | InvokerTransformer, LazyMap, AnnotationInvocationHandler | TiedMapEntry, LinkedSet, TransformedMap |
| 核心思想 | 通过 LazyMap 间接调用 InvokerTransformer | 通过 TiedMapEntry 的 toString() 调用 Map 的 get() |
| 适用范围 | 较老的 Commons Collections 库版本,以及旧版 JDK | 较新的 Commons Collections 库版本,解决了 CC1 在新版本中的问题 |
| 链条复杂性 | 相对简单,逻辑直接 | 相对复杂,涉及更多的类和间接调用 |