/*
* Copyright 2014 TWO SIGMA OPEN SOURCE, LLC
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.twosigma.beaker.jvm.classloader;
import com.google.common.io.ByteStreams;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.MalformedURLException;
import java.net.URL;
import java.net.URLClassLoader;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
public class DynamicClassLoaderSimple extends ClassLoader {
private final static Logger logger = LoggerFactory.getLogger(DynamicClassLoaderSimple.class.getName());
private URLClassLoader myloader;
private String outDir;
protected final Map<String, Class<?>> classes;
public DynamicClassLoaderSimple(ClassLoader classLoader) {
super(classLoader);
classes = Collections.synchronizedMap( new HashMap<String, Class<?>>() );
}
public void addJars(List<String> dirs) {
List<URL> urlList = new ArrayList<>();
for (String dir : dirs) {
try {
urlList.add(Paths.get(dir).toUri().toURL());
} catch (MalformedURLException e) {
logger.error(e.getMessage());
}
}
myloader = new URLClassLoader(urlList.toArray(new URL[urlList.size()]), null);
}
public void addDynamicDir(String o) {
outDir = o + File.separator;
}
private Class<?> getClass(String cname, byte[] bytes, boolean resolveIt) {
Class<?> result = defineClass( cname, bytes, 0, bytes.length );
if (result == null) {
return null;
}
/*
* Preserve package name.
*/
if (result.getPackage() == null) {
int lastDotIndex = cname.lastIndexOf( '.' );
String packageName = (lastDotIndex >= 0) ? cname.substring( 0, lastDotIndex) : "";
if(!packageName.isEmpty())
definePackage( packageName, null, null, null, null, null, null, null );
}
if (resolveIt)
resolveClass( result );
return result;
}
public Class<?> loadClass(String name) throws ClassNotFoundException {
// try parent first
try {
return getParent().loadClass(name);
} catch (ClassNotFoundException cnfe) {
}
// try java file
if(!name.startsWith("java")) {
String cname = formatClassName(name);
String fname = outDir + cname;
File f = new File(fname);
if (f.exists() && f.isFile()) {
FileInputStream fis;
try {
//System.out.println("found on dynloader");
fis = new FileInputStream(f);
byte[] bytes = ByteStreams.toByteArray(fis);
Class<?> result = getClass(name, bytes, true);
if (result != null) {
return result;
}
} catch (Exception e) {
e.printStackTrace();
}
}
// try our custom classpath
if (myloader != null) {
Class<?> result = classes.get( name );
if (result != null) {
return result;
}
InputStream is = myloader.getResourceAsStream(cname);
if (is != null) {
try {
byte[] bytes = ByteStreams.toByteArray(is);
result = getClass(name, bytes, true);
if (result != null) {
classes.put( name, result );
return result;
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
}
throw new ClassNotFoundException();
}
public URL getResource(String name) {
URL c;
c = getParent().getResource(name);
if (c == null && myloader != null) {
c = myloader.getResource(name);
}
return c;
}
public InputStream getResourceAsStream(String name) {
InputStream c;
c = getParent().getResourceAsStream(name);
if (c == null && myloader != null) {
c = myloader.getResourceAsStream(name);
}
return c;
}
public Enumeration<URL> getResources(String name) throws IOException
{
Enumeration<URL> c;
c = getParent().getResources(name);
//System.out.println("get resources "+name);
if ((c == null || !c.hasMoreElements()) && myloader != null) {
try {
c = myloader.getResources(name);
} catch(IOException e) {
}
}
return c;
}
protected String formatClassName(String className) {
className = className.replace( '/', '~' );
// '/' is used to map the package to the path
className = className.replace( '.', '/' ) + ".class";
className = className.replace( '~', '/' );
return className;
}
}