Friday, June 26, 2009

Dynamically add jar for JDBC

Java Almanac (exampledepot.com) has example of loading class not in the classpath. For example, if I want to load driver class from the Oracle JDBC Driver jar:
File file = new File("C:\\oracle\\ora92\\jdbc\\lib\\ojdbc14.jar");
// file.toURL() directly is unsafe - does not escape characters illegal in URLs
URL url = file.toURI().toURL();
URL[] urls = new URL[]{url};
ClassLoader cl = new URLClassLoader(urls);
Class cls = cl.loadClass("oracle.jdbc.driver.OracleDriver");

However, if you then call
Class.forName("oracle.jdbc.driver.OracleDriver");
Connection con = DriverManager.getConnection(…);
The calls will fail.

How do tools such as DBVisualizer dynamically add jar files to the classpath and then work with JDBC? DBVisualizer even figures out there are 2 driver classes in ojdbc14.jar and allows you to select which one to load.
This part is easy. It was looking for files like %Driver.class:
JarFile jar = new JarFile(file);
Enumeration allFiles = jar.entries();
while (allFiles.hasMoreElements()) {
JarEntry jarEntry = allFiles.nextElement();
String name = jarEntry.getName();
if (name.endsWith("Driver.class")) {
System.out.println(jarEntry.getName());
}
}

But how to add the jar to classpath so that Class.forName(..) will work?

The important thing to note is that Class.forName() is using the SystemClassLoader.
The groovy Sql class implements loadDriver with:
Class.forName(driverClassname);
and then the thread context class loader ...
Thread.currentThread().getContextClassLoader().loadClass(driverClassName);
and then the class loader that loaded the Sql class itself ...
Sql.class.getClassLoader().loadClass(driverClassName);

So the trick is really to modify the System Class Loader.
http://forums.sun.com/thread.jspa?threadID=300557&start=0&tstart=0 provides a good solution.

---------------------------------------------------------------------------------------
import java.net.URL;
import java.net.URLClassLoader;
import java.io.IOException;
import java.io.File;
import java.lang.reflect.Method;

/**
* The System Class Loader (ClassLoader.getSystemClassLoader()) is a subclass of URLClassLoader.
* It can therefore be casted into a URLClassLoader and used as one.
*
* URLClassLoader has a protected method addURL(URL url),
* which you can use to add files, jars, web addresses - any valid URL in fact.
*
* Since the method is protected you need to use reflection to invoke it.
*/
public class ClasspathAppender {

private static final Class[] parameters = new Class[]{URL.class};

public static void addFile(String s) throws IOException {
File f = new File(s);
addFile(f);
}//end method

public static void addFile(File f) throws IOException {
//addURL(f.toURL());
addURL(f.toURI().toURL());
}//end method


public static void addURL(URL u) throws IOException {

URLClassLoader sysloader = (URLClassLoader)ClassLoader.getSystemClassLoader();
Class sysclass = URLClassLoader.class;

try {
Method method = sysclass.getDeclaredMethod("addURL",parameters);
method.setAccessible(true);
method.invoke(sysloader,new Object[]{ u });
} catch (Throwable t) {
t.printStackTrace();
throw new IOException("Error, could not add URL to system classloader");
}//end try catch

}//end method

}//end class

No comments: