2000-08-20 17:49:12 +00:00
|
|
|
/* Copyright (C) 1999, 2000 Free Software Foundation
|
1999-08-18 14:16:42 +00:00
|
|
|
|
|
|
|
This file is part of libgcj.
|
|
|
|
|
|
|
|
This software is copyrighted work licensed under the terms of the
|
|
|
|
Libgcj License. Please consult the file "LIBGCJ_LICENSE" for
|
|
|
|
details. */
|
|
|
|
|
|
|
|
package java.net;
|
|
|
|
|
|
|
|
import java.io.*;
|
|
|
|
import java.util.jar.*;
|
2000-11-26 03:58:56 +00:00
|
|
|
import java.util.Enumeration;
|
1999-08-18 14:16:42 +00:00
|
|
|
import java.util.Vector;
|
|
|
|
|
|
|
|
public class URLClassLoader extends ClassLoader
|
|
|
|
{
|
|
|
|
// The URLStreamHandlerFactory
|
|
|
|
URLStreamHandlerFactory factory = null;
|
|
|
|
|
|
|
|
// `path' contains simply the URL's we're using for the searching.
|
|
|
|
private Vector path;
|
|
|
|
|
2000-08-20 17:49:12 +00:00
|
|
|
// If path[n] is a zip/jar, then this holds a JarURLConnection for
|
|
|
|
// that thing, otherwise, path[n] is null.
|
1999-08-18 14:16:42 +00:00
|
|
|
private Vector info;
|
|
|
|
|
|
|
|
private URLStreamHandler getHandler0 (String protocol)
|
|
|
|
{
|
|
|
|
if (factory != null)
|
|
|
|
return factory.createURLStreamHandler(protocol);
|
|
|
|
else
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public URLClassLoader (URL[] urls)
|
|
|
|
{
|
|
|
|
this (urls, null, null);
|
|
|
|
}
|
|
|
|
|
|
|
|
public URLClassLoader (URL[] urls, ClassLoader parent)
|
|
|
|
{
|
|
|
|
this (urls, parent, null);
|
|
|
|
}
|
|
|
|
|
2000-11-26 03:58:56 +00:00
|
|
|
// A File URL may actually be a Jar URL. Convert if possible.
|
|
|
|
private URL jarFileize (URL url)
|
|
|
|
{
|
|
|
|
if (! url.getProtocol ().equals ("jar"))
|
|
|
|
{
|
|
|
|
String f = url.getFile ();
|
|
|
|
|
|
|
|
// If it ends with '/' we'll take it for a directory,
|
|
|
|
// otherwise it's a jar file. This is how JDK 1.2 defines
|
|
|
|
// it, so we will not try to be smart here.
|
|
|
|
if (f.charAt (f.length ()-1) != '/')
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
url = new URL ("jar", "", -1, (url.toExternalForm ())+"!/",
|
|
|
|
getHandler0 ("jar"));
|
|
|
|
}
|
|
|
|
catch (MalformedURLException x)
|
|
|
|
{
|
|
|
|
/* ignore */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return url;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected void addURL (URL url)
|
|
|
|
{
|
|
|
|
JarURLConnection conn = null;
|
|
|
|
|
|
|
|
// Convert a Jar File URL into Jar URL if possible.
|
|
|
|
url = jarFileize (url);
|
|
|
|
|
|
|
|
path.addElement (url);
|
|
|
|
|
|
|
|
if (url.getProtocol ().equals ("jar"))
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
conn = (JarURLConnection) url.openConnection ();
|
|
|
|
}
|
|
|
|
catch (java.io.IOException x)
|
|
|
|
{
|
|
|
|
/* ignore */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
info.addElement (conn);
|
|
|
|
}
|
|
|
|
|
1999-08-18 14:16:42 +00:00
|
|
|
public URLClassLoader (URL[] urls, ClassLoader parent,
|
|
|
|
URLStreamHandlerFactory fac)
|
|
|
|
{
|
|
|
|
super (parent);
|
|
|
|
|
|
|
|
factory = fac;
|
|
|
|
|
|
|
|
if (urls == null || urls.length == 0)
|
|
|
|
{
|
|
|
|
path = new Vector (1);
|
|
|
|
info = new Vector (1);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
path = new Vector (urls.length);
|
|
|
|
info = new Vector (urls.length);
|
|
|
|
|
|
|
|
for (int i = 0; i < urls.length; i++)
|
|
|
|
{
|
2000-11-26 03:58:56 +00:00
|
|
|
// Convert a Jar File URL into a Jar URL is possible.
|
|
|
|
URL u = jarFileize(urls[i]);
|
1999-08-18 14:16:42 +00:00
|
|
|
|
2000-11-26 03:58:56 +00:00
|
|
|
path.addElement (u);
|
1999-08-18 14:16:42 +00:00
|
|
|
|
|
|
|
if (u.getProtocol ().equals ("jar"))
|
|
|
|
{
|
|
|
|
JarURLConnection conn = null;
|
|
|
|
try
|
|
|
|
{
|
|
|
|
conn = (JarURLConnection) u.openConnection ();
|
|
|
|
}
|
|
|
|
catch (java.io.IOException x)
|
|
|
|
{
|
|
|
|
/* ignore */
|
|
|
|
}
|
2000-11-26 03:58:56 +00:00
|
|
|
info.addElement (conn);
|
1999-08-18 14:16:42 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2000-11-26 03:58:56 +00:00
|
|
|
info.addElement (null);
|
1999-08-18 14:16:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2000-11-26 03:58:56 +00:00
|
|
|
|
|
|
|
public URL[] getURLs ()
|
|
|
|
{
|
|
|
|
URL[] urls = new URL[path.size()];
|
|
|
|
path.copyInto (urls);
|
|
|
|
return urls;
|
|
|
|
}
|
1999-08-18 14:16:42 +00:00
|
|
|
|
2000-11-26 03:58:56 +00:00
|
|
|
public Enumeration findResources (String name)
|
1999-08-18 14:16:42 +00:00
|
|
|
{
|
2000-11-26 03:58:56 +00:00
|
|
|
Vector results = new Vector ();
|
|
|
|
|
1999-08-18 14:16:42 +00:00
|
|
|
for (int i = 0; i < path.size(); i++)
|
|
|
|
{
|
2000-11-26 03:58:56 +00:00
|
|
|
URL u = (URL)path.elementAt (i);
|
|
|
|
|
1999-08-18 14:16:42 +00:00
|
|
|
try {
|
|
|
|
JarURLConnection conn = (JarURLConnection) info.elementAt (i);
|
2000-11-26 03:58:56 +00:00
|
|
|
|
1999-08-18 14:16:42 +00:00
|
|
|
if (conn != null)
|
|
|
|
{
|
2000-08-20 17:49:12 +00:00
|
|
|
if (conn.getJarFile().getJarEntry (name) != null)
|
2000-11-26 03:58:56 +00:00
|
|
|
results.addElement (new URL(u, name, getHandler0 (u.getProtocol())));
|
1999-08-18 14:16:42 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
URL p = new URL (u, name, getHandler0 (u.getProtocol()));
|
2000-11-26 03:58:56 +00:00
|
|
|
|
1999-08-18 14:16:42 +00:00
|
|
|
InputStream is = p.openStream();
|
|
|
|
if (is != null)
|
|
|
|
{
|
|
|
|
is.close();
|
2000-11-26 03:58:56 +00:00
|
|
|
results.addElement (p);
|
1999-08-18 14:16:42 +00:00
|
|
|
}
|
|
|
|
}
|
2000-11-26 03:58:56 +00:00
|
|
|
|
1999-08-18 14:16:42 +00:00
|
|
|
// if we get an exception ... try the next path element
|
|
|
|
} catch (IOException x) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2000-11-26 03:58:56 +00:00
|
|
|
|
|
|
|
return results.elements ();
|
1999-08-18 14:16:42 +00:00
|
|
|
}
|
|
|
|
|
2000-11-26 03:58:56 +00:00
|
|
|
public URL findResource (String name)
|
1999-08-18 14:16:42 +00:00
|
|
|
{
|
|
|
|
for (int i = 0; i < path.size(); i++)
|
|
|
|
{
|
2000-11-26 03:58:56 +00:00
|
|
|
URL u = (URL)path.elementAt (i);
|
1999-08-18 14:16:42 +00:00
|
|
|
|
|
|
|
try {
|
|
|
|
JarURLConnection conn = (JarURLConnection) info.elementAt (i);
|
2000-11-26 03:58:56 +00:00
|
|
|
|
1999-08-18 14:16:42 +00:00
|
|
|
if (conn != null)
|
|
|
|
{
|
2000-11-26 03:58:56 +00:00
|
|
|
if (conn.getJarFile().getJarEntry (name) != null)
|
|
|
|
return new URL(u, name, getHandler0 (u.getProtocol()));
|
1999-08-18 14:16:42 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2000-11-26 03:58:56 +00:00
|
|
|
URL p = new URL (u, name, getHandler0 (u.getProtocol()));
|
|
|
|
|
|
|
|
InputStream is = p.openStream();
|
1999-08-18 14:16:42 +00:00
|
|
|
if (is != null)
|
2000-11-26 03:58:56 +00:00
|
|
|
{
|
|
|
|
is.close();
|
|
|
|
return p;
|
|
|
|
}
|
1999-08-18 14:16:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// if we get an exception ... try the next path element
|
|
|
|
} catch (IOException x) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
// and finally, we can implement our class loader functionality.
|
|
|
|
protected Class findClass (String name)
|
|
|
|
throws ClassNotFoundException
|
|
|
|
{
|
|
|
|
if (name == null)
|
|
|
|
throw new ClassNotFoundException ("null");
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
2000-08-20 17:49:12 +00:00
|
|
|
URL u = getResource (name.replace ('.', '/') + ".class");
|
|
|
|
|
|
|
|
if (u == null)
|
1999-08-18 14:16:42 +00:00
|
|
|
throw new ClassNotFoundException (name);
|
2000-08-20 17:49:12 +00:00
|
|
|
|
|
|
|
URLConnection connection = u.openConnection ();
|
|
|
|
InputStream is = connection.getInputStream ();
|
|
|
|
|
|
|
|
int len = connection.getContentLength ();
|
1999-08-18 14:16:42 +00:00
|
|
|
byte[] data = new byte[len];
|
|
|
|
|
|
|
|
int left = len;
|
|
|
|
int off = 0;
|
|
|
|
while (left > 0)
|
|
|
|
{
|
|
|
|
int c = is.read (data, off, len-off);
|
|
|
|
if (c == -1 || c == 0)
|
|
|
|
throw new InternalError ("premature end of file");
|
|
|
|
left -= c;
|
|
|
|
off += c;
|
|
|
|
}
|
|
|
|
|
|
|
|
return defineClass (name, data, 0, len);
|
|
|
|
}
|
|
|
|
catch (java.io.IOException x)
|
|
|
|
{
|
|
|
|
throw new ClassNotFoundException(name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|